Со времен простейшей формы подъёма. Токен асинхронного завершения операции Asio является функцией обратного вызова, мы можем применить ту же тактику для Asio, что и для нашего гипотетического.AsyncAPI
Асинхронные операции.
К счастью, нам это не нужно. Повышаю. Asio использует механизм, с помощью которого абонент может настроить поведение уведомления любой операции асинхронизации. Поэтому мы можем построитьтокен завершения, который при переходе наувеличивается. Asioасинхронизирует работу, запрашивает блокировку вызывающего волокна.
Типичная асинхронная функция Асио может выглядеть примерно так:
template < ..., class CompletionToken >
deduced_return_type
async_something( ... , CompletionToken&& token)
{
handler_type<CompletionToken, ...>::type handler(token)
;
async_result<decltype(handler)> result(handler)
;
return result.get()
;
}
Мы будем задействовать этот механизм, который основан на специализации Asio’shandler_type<>
шаблон дляТокен завершения
типа и подпись конкретного обратного вызова. Остальная часть этого обсуждения будет ссылаться наasync_something()
в качестве рассматриваемой функции Asio async.
В реализации, описанной ниже, используются средства более низкого уровня, чемобещание
ибудущее
, потому чтообещание
механизм плохо взаимодействует сio_service::стоп
. Он даетнарушенное обещание
исключения.
бустер::волокна::асио::выходвыход
является завершением токена такого рода.доходность
является примеромдоходность
:
class yield_t {
public:
yield_t() = default;
yield_t operator[]( boost::system::error_code & ec) const {
yield_t tmp;
tmp.ec_ = & ec;
return tmp;
}
boost::system::error_code * ec_{ nullptr };
};
на самом деле только заполнитель, способ вызвать повышение. Азио кастомизация. Он может связыватьбустер::система::error_code
для использования фактическим обработчиком.
доходность
объявляется как:
thread_local yield_t yield{};
Настройка Asio осуществляется посредством специализацииboost::asio::handler_type<>
дляyield_t
:
template< typename ReturnType >
struct handler_type< fibers::asio::yield_t, ReturnType( boost::system::error_code) >
{ typedef fibers::asio::detail::yield_handler< void > type; };
(На самом деле вDetail/yield.hppесть четыре различные специализации, по одной для каждой из четырех подписей Asio async callback, которые мы ожидаем.)
Вышеуказанное предписывает Asio использоватьyield_handler
в качестве фактического обработчика для операции асинхронизации, к которойдоходность
передается. Там’s дженерикyield_handler<T>
реализация иyield_handler<пустота>
специализация. Начнем с<void>
специализация:
template<>
class yield_handler< void >: public yield_handler_base {
public:
explicit yield_handler( yield_t const& y) :
yield_handler_base{ y } {
}
void operator()() {
( * this)( boost::system::error_code() );
}
using yield_handler_base::operator();
};
async_something
, проконсультировавшись сhandler_type<>
специализацией признаков, представляет собойyield_handler<void>
, который будет передан в качестве фактического обратного вызова для операции асинхронизации.yield_handler
& #8217;s конструктор принимаетyield_t
экземплярyield
объект передан функции асинхронизации] и передает его вyield_handler_base
:
class yield_handler_base {
public:
yield_handler_base( yield_t const& y) :
ctx_{ boost::fibers::context::active() },
yt_( y ) {
}
void operator()( boost::system::error_code const& ec) {
BOOST_ASSERT_MSG( ycomp_,
"Must inject yield_completion* "
"before calling yield_handler_base::operator()()");
BOOST_ASSERT_MSG( yt_.ec_,
"Must inject boost::system::error_code* "
"before calling yield_handler_base::operator()()");
yield_completion::lock_t lk{ ycomp_->mtx_ };
ycomp_->completed_ = true;
* yt_.ec_ = ec;
if ( fibers::context::active() != ctx_ ) {
fibers::context::active()->set_ready( ctx_);
}
}
boost::fibers::context * ctx_;
yield_t yt_;
yield_completion * ycomp_{ nullptr };
};
yield_handler_base
хранит копиюyield_t
экземпляра — который, как показано выше, содержит толькоerror_code*
. Он также захватывает контекст
* для текущего волокна, вызывая контекст::active()
.
Вы заметите, чтоyield_handler_base
имеет еще один элемент данныхycomp_
, который инициализируется доnullptr
его конструктором и #8212; хотя егооператор[]
метод опирается наycomp_
, являющийся ненулевым. Подробнее об этом в один момент.
Сконструировавyield_handler<void>
,async_something[]async_resulthandler_typeтипа
: в этом случаеasync_result<250]void
. Он передаетyield_handler<void>
экземпляр новомуasync_result
экземпляру.
template<>
class async_result< boost::fibers::asio::detail::yield_handler< void > > :
public boost::fibers::asio::detail::async_result_base {
public:
typedef void type;
explicit async_result( boost::fibers::asio::detail::yield_handler< void > & h):
boost::fibers::asio::detail::async_result_base{ h } {
}
};
Естественно, это приводит нас прямо кasync_result_base
:
class async_result_base {
public:
explicit async_result_base( yield_handler_base & h) {
h.ycomp_ = & this->ycomp_;
if ( ! h.yt_.ec_) {
h.yt_.ec_ = & ec_;
}
}
void get() {
ycomp_.wait();
if ( ec_) {
throw_exception( boost::system::system_error{ ec_ } );
}
}
private:
boost::system::error_code ec_{};
yield_completion ycomp_{};
};
Вот какyield_handler_base::ycomp_
становится ненулевым:async_result_base
& #8217; конструктор вводит указатель обратно в свойyield_completion
член.
Напомним, что каноническийвыход_t
экземплярвыход
инициализирует свойошибку_код*
членec_
донульприт
. Если этот экземпляр передаетсяasync_Something[]
по-прежнемуnullptr
, копия, хранящаяся вyield_handler_base
, также будет иметь нульec_
.async_result_base
& #8217;s yield_handler_base& #8217;sec_ошибка_code
член.
Сейчас сцена установлена.async_something[]
инициирует фактическую операцию асинхронизации, организуя ееyield_handler<void>
экземпляр по завершении. Let’s говорят, ради аргумента, что фактическая операция асинхронизации’s callback имеет подписьvoiderror_code
.
Но так как это’s асинхронная операция, управление возвращается сразу кasync_Something
.async_Something
вызываетasync_result<yield_handler<void>>::получает(]
и вернет свое возвращаемое значение.
async_result<yield_handler<void>>::get
наследуетasync_result_base::get
.
async_result_base::получить()
немедленно вызываетвыход_завершение::ждать()
.
struct yield_completion {
typedef fibers::detail::spinlock mutex_t;
typedef std::unique_lock< mutex_t > lock_t;
mutex_t mtx_{};
bool completed_{ false };
void wait() {
lock_t lk{ mtx_ };
if ( ! completed_) {
fibers::context::active()->suspend( lk);
}
}
};
Предполагая, что ожидающая операция асинхронизации еще не завершена,yield_completion::completed_
по-прежнему будетложным
, аждать[]
вызоветконтекст::suspend()
на текущем волокне.
Другие волокна теперь будут иметь возможность работать.
Спустя некоторое время асинхронная операция завершается. Он называетyield_handler<void>::оператор()error_codeconst&]
с ошибкой_code
, указывающей либо на успех, либо на неудачу. Мы и #8217 рассмотрим оба случая.
yield_handler<void>
эксплицитно наследуетоператор()error_codeconst&]
отyield_handler_base
.
yield_handler_base::оператор()error_codeconst&]
первые наборыyield_completion::завершено_
истинно. Таким образом, еслиasync_что-то()
& #8217; операция асинхронизации завершается немедленно & #8212; еслиyield_handler_base::оператор()
называется еще до того, какasync_result_base::[]
& #8212; вызывающее волокнонеприостанавливается.
Фактическийerror_code
, создаваемый операцией асинхронизации, затем хранится через сохраненныйуказатель yield_t::ec_
. Еслиasync_Something[]
& #8217;s caller используется (например)выходmy_ec
для связывания локальногоerror_code
экземпляра, фактическое значениеerror_code
сохраняется в переменной caller’. В противном случае он хранится вasync_result_base::ec_
.
Если сохраненный контекст волокнаyield_handler_base::ctx_
уже не работает, он помечается как готовый к запуску, передавая его в контекст::set_ready()
. Затем управление возвращается отyield_handler_base::оператор[]
: Обратный звонок закончен.
Со временем это волокно возобновляется. Контроль возвращается из контекста::suspend()
вyield_completion::Wait()
, который возвращается вasync_result_base::get()
.
- Если первоначальный абонент передал
выходmy_ec
async_что-тодля связывания локальногоerror_code, товыход_handler_base::оператор[]
сохранил свойerror_code
для абонента’smy_ec
, оставивasync_result_base::ec_
инициализированным к успеху. - Если первоначальный абонент передал
, то
async_Something[]
без привязки к локальномуerror_code
переменная, затемyield_handler_base::оператор[]
сохранил свойerror_code
вasync_result_base::ec_
. Еслиошибка_код
является успехом, то все хорошо. - В противном случае — первоначальный абонент не связывал локальный
error_code
иyield_handler_base::оператор[]
был вызванerror_code
с указанием ошибки —async_result_base::получить[]
бросаетsystem_error
с этимerror_code
.
Случай, в которомasync_something[]
& #8217; завершение обратного вызова имеет сигнатуруvoid
.yield_handler<void>::оператор()
вызывает механизм выше с& #8220;успех& #8221;ошибка_code
.
Завершение обратного вызова с подписьюvoiderror_code,T
(то есть: помимоerror_code
, обратный вызов получает некоторый элемент данных) обрабатывается несколько иначе. Для этого вида подписиhandler_type<>::type
указываетyield_handler<T>
(дляT
, кромеvoid
.
Ayield_handler<T>
резервируетзначение_
указатель на значение типаT
:
template< typename T >
class yield_handler: public yield_handler_base {
public:
explicit yield_handler( yield_t const& y) :
yield_handler_base{ y } {
}
void operator()( T t) {
(*this)( boost::system::error_code(), std::move(t) );
}
void operator()( boost::system::error_code const& ec, T t) {
BOOST_ASSERT_MSG( value_,
"Must inject value ptr "
"before caling yield_handler<T>::operator()()");
* value_ = std::move( t);
yield_handler_base::operator()( ec);
}
T * value_{ nullptr };
};
Этот указатель инициализируется доnullptr
.
Когдаasync_something
мгновенноasync_reult<yield_handler<T>>
:
template< typename T >
class async_result< boost::fibers::asio::detail::yield_handler< T > > :
public boost::fibers::asio::detail::async_result_base {
public:
typedef T type;
explicit async_result( boost::fibers::asio::detail::yield_handler< T > & h) :
boost::fibers::asio::detail::async_result_base{ h } {
h.value_ = & value_;
}
type get() {
boost::fibers::asio::detail::async_result_base::get();
return std::move( value_);
}
private:
type value_{};
};
Этоaync_result<>
специализация резервирует член типаT
для получения переданного элемента данных и устанавливаетвыход_handler<T>::значение_
для указания на его собственный член данных.
async_result<yield_handler<T>>
переопределенияполучают
. Оверрайд-звонкиasync_result_base::получают()
, поэтому вызывающее волокно приостанавливается, как описано выше.
yield_handler<>::оператор[] []error_code,T
сохраняет свое пройденноеT
значение в
async_result<T>>]::значение_.
Затем он передает управлениеyield_handler_base::оператор()error_code
для борьбы с пробуждением исходного волокна, как описано выше.
Когдаaync_result<yield_handler<T>>::[]
возобновляется, он возвращает сохраненноезначение_
async_something
и, в конечном счете,async_something
[8217].
Случай обратного вызова подписиvoid
обрабатывается, имеяyield_handler<T>:оператор[]T
задействуютvoidошибку_code
машины, пройдя& #8220;успех& #8221;ошибку_code
.
Приведенный выше исходный код содержится вyield.hppиdetail/yield.hpp.