1
0
mirror of https://github.com/tearshark/librf.git synced 2024-10-02 00:00:11 +08:00
librf/tutorial/test_async_modern_cb.cpp

425 lines
15 KiB
C++
Raw Normal View History

2020-03-26 17:26:39 +08:00
2019-04-21 09:05:04 +08:00
#include "librf.h"
#include <chrono>
#include <iostream>
#include <string>
#include <thread>
#include <future>
//原旨主义的异步函数,其回调写法大致如下
2019-04-21 09:05:04 +08:00
template<typename _Input_t, typename _Callable_t>
void tostring_async_originalism(_Input_t&& value, _Callable_t&& token)
2019-04-21 09:05:04 +08:00
{
std::thread([callback = std::move(token), value = std::forward<_Input_t>(value)]
2019-04-21 09:05:04 +08:00
{
callback(std::to_string(value));
}).detach();
}
//使用原旨主义的方式扩展异步方法来支持future
template<typename _Input_t>
auto tostring_async_originalism_future(_Input_t&& value)
{
std::promise<std::string> _promise;
std::future<std::string> _future = _promise.get_future();
std::thread([_promise = std::move(_promise), value = std::forward<_Input_t>(value)]() mutable
{
_promise.set_value(std::to_string(value));
}).detach();
return std::move(_future);
}
//----------------------------------------------------------------------------------------------------------------------
//以下演示如何通过现代回调(Modern Callback) 使用回调适配器模型,
//将异步回调函数扩展到支持future模式调用链模式以及协程。
//首先准备modern_call_return_void_t和modern_callback_adapter_t给异步函数使用
//通过一个间接的类来解决返回void的语法问题以便于优化返回值
struct modern_call_return_void_t
{
void get(){}
};
2019-04-21 09:05:04 +08:00
//回调适配器的模板类
//_Callable_t 要符合 _Signature_t 签名
//这个类除了转移token外不做任何有效的工作
//有效工作等待特列化的类去做
template<typename _Callable_t, typename _Signature_t>
2019-04-21 09:05:04 +08:00
struct modern_callback_adapter_t
{
using callback_type = _Callable_t;
using return_type = modern_call_return_void_t;
2019-04-21 09:05:04 +08:00
static std::tuple<callback_type, return_type> traits(_Callable_t&& token)
2019-04-21 09:05:04 +08:00
{
return { std::forward<_Callable_t>(token), {} };
2019-04-21 09:05:04 +08:00
}
};
//一个使用回调处理结果的异步函数,会涉及以下概念:
2019-04-21 09:05:04 +08:00
//_Input_t异步函数的输入参数
//_Signature_t: 此异步回调的函数签名应当满足void(_Exception_t, _Result_t...)或者void(_Result_t...)’类型;
//_Callable_t回调函数或标记如果是回调函数则需要符合_Signature_t的签名类型。这个回调必须调用一次且只能调用一次
2019-04-21 09:05:04 +08:00
//_Return_t异步函数的返回值
//_Result_t...:异步函数完成后的结果值,作为回调函数的入参部分;这个参数可以有零至多个;
2019-04-21 09:05:04 +08:00
//_Exception_t回调函数的异常 如果不喜欢异常的则忽略这个部分,但就得异步代码将异常处置妥当;
//
//在回调适配器模型里_Input_t/_Result_t/_Exception_t(可选)是异步函数提供的功能所固有的部分_Callable_t/_Return_t
//部分并不直接使用而是通过适配器去另外处理。这样给予适配器一次扩展到future模式调用链模式的机会以及支持协程的机会。
//
//tostring_async 演示了在其他线程里将_Input_t的输入值转化为std::string类型的_Result_t。
//然后调用_Signature_t为 void(std::string &&) 类型的 _Callable_t。
//忽视异常处理故没有_Exception_t。
2019-04-21 09:05:04 +08:00
//
template<typename _Input_t, typename _Callable_t>
auto tostring_async(_Input_t&& value, _Callable_t&& token)
2019-04-21 09:05:04 +08:00
{
//适配器类型
2020-02-16 15:06:09 +08:00
using _Adapter_t = modern_callback_adapter_t<typename resumef::remove_cvref_t<_Callable_t>, void(std::string)>;
//通过适配器获得兼容_Signature_t类型的真正的回调以及返回值_Return_t
auto adapter = _Adapter_t::traits(std::forward<_Callable_t>(token));
2019-04-21 09:05:04 +08:00
//callback与token未必是同一个变量甚至未必是同一个类型
std::thread([callback = std::move(std::get<0>(adapter)), value = std::forward<_Input_t>(value)]
2019-04-21 09:05:04 +08:00
{
using namespace std::literals;
std::this_thread::sleep_for(0.1s);
callback(std::to_string(value));
2019-04-21 09:05:04 +08:00
}).detach();
//返回适配器的_Return_t变量
return std::move(std::get<1>(adapter)).get();
2019-04-21 09:05:04 +08:00
}
//或者宏版本写法
#define MODERN_CALLBACK_TRAITS(_Token_value, _Signature_t) \
2020-02-16 15:06:09 +08:00
using _Adapter_t = modern_callback_adapter_t<typename resumef::remove_cvref_t<_Callable_t>, _Signature_t>; \
auto _Adapter_value = _Adapter_t::traits(std::forward<_Callable_t>(_Token_value))
#define MODERN_CALLBACK_CALL() std::move(std::get<0>(_Adapter_value))
#define MODERN_CALLBACK_RETURN() return std::move(std::get<1>(_Adapter_value)).get()
template<typename _Input_t, typename _Callable_t>
auto tostring_async_macro(_Input_t&& value, _Callable_t&& token)
2019-04-21 09:05:04 +08:00
{
MODERN_CALLBACK_TRAITS(token, void(std::string));
2019-04-21 09:05:04 +08:00
std::thread([callback = MODERN_CALLBACK_CALL(), value = std::forward<_Input_t>(value)]
2019-04-21 09:05:04 +08:00
{
callback(std::to_string(value));
2019-04-21 09:05:04 +08:00
}).detach();
MODERN_CALLBACK_RETURN();
2019-04-21 09:05:04 +08:00
}
//----------------------------------------------------------------------------------------------------------------------
2019-04-21 09:05:04 +08:00
//下面演示如何扩展tostring_async函数以支持future模式
//future库有多种但应当都提供遵循promise/future对兼容std::promise/std::future用法
//这样的话可以做一个更加通用的支持future的callback类
2019-04-21 09:05:04 +08:00
//实现use_future_callback_t的基类避免写一些重复代码
template<typename _Promise_traits, typename _Result_t>
struct use_future_callback_base_t
{
//回调函数的结果类型,已经排除掉了异常参数
using result_type = _Result_t;
//通过_Promise_traits获取真正的promise类型
using promise_type = typename _Promise_traits::template promise_type<result_type>;
2019-04-21 09:05:04 +08:00
//此类持有一个std::promise<_Result_t>,便于设置值和异常
//而将与promise关联的future作为返回值_Return_t让tostring_async返回。
mutable promise_type _promise;
auto get_future() const
{
return this->_promise.get_future();
}
};
//此类的实例作为真正的callback交给异步回调函数替换token。
//在实际应用中需要针对是否有异常参数结果值为01多个等情况做特殊处理故还需要通过更多的偏特化版本来支持。
//具体的异常参数需要根据实际应用去特里化。这里仅演示通过std::exception_ptr作为异常传递的情况。
template<typename...>
struct use_future_callback_t;
//无异常无结果的callback类型void()
template<typename _Promise_traits>
struct use_future_callback_t<_Promise_traits> : public use_future_callback_base_t<_Promise_traits, void>
{
using use_future_callback_base_t<_Promise_traits, void>::use_future_callback_base_t;
2019-04-21 09:05:04 +08:00
void operator()() const
2019-04-21 09:05:04 +08:00
{
this->_promise.set_value();
2019-04-21 09:05:04 +08:00
}
};
//有异常无结果的callback类型void(exception_ptr)
template<typename _Promise_traits>
struct use_future_callback_t<_Promise_traits, std::exception_ptr> : public use_future_callback_base_t<_Promise_traits, void>
{
using use_future_callback_base_t<_Promise_traits, void>::use_future_callback_base_t;
2019-04-21 09:05:04 +08:00
void operator()(std::exception_ptr eptr) const
2019-04-21 09:05:04 +08:00
{
if (!eptr)
this->_promise.set_value();
2019-04-21 09:05:04 +08:00
else
this->_promise.set_exception(std::move(eptr));
2019-04-21 09:05:04 +08:00
}
};
//无异常单结果的callback类型void(_Result_t)
template<typename _Promise_traits, typename _Result_t>
struct use_future_callback_t<_Promise_traits, _Result_t> : public use_future_callback_base_t<_Promise_traits, _Result_t>
2019-04-21 09:05:04 +08:00
{
using use_future_callback_base_t<_Promise_traits, _Result_t>::use_future_callback_base_t;
2019-04-21 09:05:04 +08:00
template<typename Arg>
void operator()(Arg && arg) const
2019-04-21 09:05:04 +08:00
{
this->_promise.set_value(std::forward<Arg>(arg));
}
};
2019-04-21 09:05:04 +08:00
//有异常单结果的callback类型void(std::exception_ptr, _Result_t)
template<typename _Promise_traits, typename _Result_t>
struct use_future_callback_t<_Promise_traits, std::exception_ptr, _Result_t> : public use_future_callback_base_t<_Promise_traits, _Result_t>
{
using use_future_callback_base_t<_Promise_traits, _Result_t>::use_future_callback_base_t;
template<typename Arg>
void operator()(std::exception_ptr eptr, Arg && arg) const
{
if (!eptr)
this->_promise.set_value(std::forward<Arg>(arg));
else
this->_promise.set_exception(std::move(eptr));
2019-04-21 09:05:04 +08:00
}
};
//无异常多结果的callback类型void(_Result_t...)
template<typename _Promise_traits, typename... _Result_t>
struct use_future_callback_t<_Promise_traits, _Result_t...> : public use_future_callback_base_t<_Promise_traits, std::tuple<_Result_t...> >
{
using use_future_callback_base_t<_Promise_traits, std::tuple<_Result_t...> >::use_future_callback_base_t;
template<typename... Args>
void operator()(Args&&... args) const
{
static_assert(sizeof...(Args) == sizeof...(_Result_t), "");
this->_promise.set_value(std::make_tuple(std::forward<Args>(args)...));
}
};
//有异常多结果的callback类型void(std::exception_ptr, _Result_t...)
template <typename _Promise_traits, typename... _Result_t>
struct use_future_callback_t<_Promise_traits, std::exception_ptr, _Result_t...> : public use_future_callback_base_t<_Promise_traits, std::tuple<_Result_t...> >
{
using use_future_callback_base_t<_Promise_traits, std::tuple<_Result_t...> >::use_future_callback_base_t;
template<typename... Args>
void operator()(std::exception_ptr eptr, Args&&... args) const
{
static_assert(sizeof...(Args) == sizeof...(_Result_t), "");
if (!eptr)
this->_promise.set_value(std::make_tuple(std::forward<Args>(args)...));
else
this->_promise.set_exception(std::move(eptr));
}
};
2019-04-21 09:05:04 +08:00
//与use_future_callback_t配套的获得_Return_t的类
template<typename _Future_traits, typename _Result_t>
struct use_future_return_t
2019-04-21 09:05:04 +08:00
{
using result_type = _Result_t;
using future_type = typename _Future_traits::template future_type<result_type>;
future_type _future;
2019-04-21 09:05:04 +08:00
use_future_return_t(future_type && ft)
: _future(std::move(ft)) {}
2019-04-21 09:05:04 +08:00
future_type get()
2019-04-21 09:05:04 +08:00
{
return std::move(_future);
2019-04-21 09:05:04 +08:00
}
};
//利用use_future_callback_t + use_future_return_t 实现的callback适配器
template<typename _Token_as_callable_t, typename... _Result_t>
struct modern_callback_adapter_impl_t
{
using traits_type = _Token_as_callable_t;
using callback_type = use_future_callback_t<traits_type, _Result_t...>;
using result_type = typename callback_type::result_type;
using return_type = use_future_return_t<traits_type, result_type>;
static std::tuple<callback_type, return_type> traits(const _Token_as_callable_t& /*没人关心这个变量*/)
{
callback_type callback{};
auto future = callback.get_future();
return { std::move(callback), std::move(future) };
}
};
//----------------------------------------------------------------------------------------------------------------------
//一、做一个使用std::promise/std::future的辅助类。
//这个类还负责萃取promise/future对的类型。
struct std_future_t
{
template<typename _Result_t>
using promise_type = std::promise<_Result_t>;
template<typename _Result_t>
using future_type = std::future<_Result_t>;
};
//二、申明这个辅助类的全局变量。不申明这个变量也行就是每次要写use_future_t{},麻烦些。
//以后就使用std_future替代tostring_async的token参数了。
//这个参数其实不需要实质传参,最后会被编译器优化没了。
//仅仅是要指定_Callable_t的类型为std_future_t
//从而在tostring_async函数内使用偏特化的modern_callback_adapter_t<std_future_t, ...>而已。
constexpr std_future_t std_future{};
//三、偏特化_Callable_t为std_future_t类型的modern_callback_adapter_t
//真正的回调类型是use_future_callback_t返回类型_Return_t是use_future_return_t。
//配合use_future_callback_t的promise<result_type>和use_future_return_t的future<result_type>正好组成一对promise/future对。
//promise在真正的回调里设置结果值
//future返回给调用者获取结果值。
template<typename R, typename... _Result_t>
struct modern_callback_adapter_t<std_future_t, R(_Result_t...)> : public modern_callback_adapter_impl_t<std_future_t, _Result_t...>
2019-04-21 09:05:04 +08:00
{
};
//----------------------------------------------------------------------------------------------------------------------
//同理可以制作支持C++20的协程的下列一系列类其实这才是我的最终目的
struct use_librf_t
{
template<typename _Result_t>
using promise_type = resumef::awaitable_t<_Result_t>;
2019-04-21 09:05:04 +08:00
template<typename _Result_t>
using future_type = resumef::future_t<_Result_t>;
};
constexpr use_librf_t use_librf{};
2019-04-21 09:05:04 +08:00
template<typename R, typename... _Result_t>
struct modern_callback_adapter_t<use_librf_t, R(_Result_t...)> : public modern_callback_adapter_impl_t<use_librf_t, _Result_t...>
{
2019-04-21 09:05:04 +08:00
};
//所以我现在的看法是支持异步操作的库尽可能如此设计回调。这样便于支持C++20的协程。以及future::then这样的任务链。
//这才是“摩登C++”!
//----------------------------------------------------------------------------------------------------------------------
2019-04-21 09:05:04 +08:00
//使用范例
//演示异步库有多个异步回调函数只要按照Modern Callback范式去做回调就不再需要写额外的代码就可以适配到future+librf以及更多的其他库
template<typename _Ty1, typename _Ty2, typename _Callable_t>
auto add_async(_Ty1&& val1, _Ty2&& val2, _Callable_t&& token)
{
MODERN_CALLBACK_TRAITS(token, void(decltype(val1 + val2)));
std::thread([=, callback = MODERN_CALLBACK_CALL()]
{
using namespace std::literals;
std::this_thread::sleep_for(0.1s);
callback(val1 + val2);
}).detach();
MODERN_CALLBACK_RETURN();
}
//演示异步库有多个异步回调函数只要按照Modern Callback范式去做回调就不再需要写额外的代码就可以适配到future+librf以及更多的其他库
template<typename _Ty1, typename _Ty2, typename _Callable_t>
auto muldiv_async(_Ty1&& val1, _Ty2&& val2, _Callable_t&& token)
{
MODERN_CALLBACK_TRAITS(token, void(std::exception_ptr, decltype(val1 * val2), decltype(val1 / val2)));
std::thread([=, callback = MODERN_CALLBACK_CALL()]
{
using namespace std::literals;
std::this_thread::sleep_for(0.1s);
auto v1 = val1 * val2;
if (val2 == 0)
callback(std::make_exception_ptr(std::logic_error("divided by zero")), v1, 0);
else
callback(nullptr, v1, val1 / val2);
}).detach();
MODERN_CALLBACK_RETURN();
}
2019-04-21 09:05:04 +08:00
void resumable_main_modern_cb()
{
using namespace std::literals;
//使用lambda作为异步回调函数传统用法
tostring_async_originalism(-1.0, [](std::string && value)
{
std::cout << value << std::endl;
});
std::this_thread::sleep_for(0.5s);
2019-04-21 09:05:04 +08:00
tostring_async(1.0, [](std::string && value)
{
std::cout << value << std::endl;
});
std::this_thread::sleep_for(0.5s);
2019-04-21 09:05:04 +08:00
std::cout << "......" << std::endl;
//支持future的用法
std::future<std::string> f1 = tostring_async_originalism_future(5);
2019-04-21 09:05:04 +08:00
std::cout << f1.get() << std::endl;
std::future<std::string> f2 = tostring_async(6.0f, std_future);
2019-04-21 09:05:04 +08:00
std::cout << f2.get() << std::endl;
//支持librf的用法
GO
{
#ifndef __clang__
try
#endif
{
int val = co_await add_async(1, 2, use_librf);
std::cout << val << std::endl;
//muldiv_async函数可能会抛异常取决于val是否是0
//异常将会带回到本协程里的代码所以需要try-catch
auto ab = co_await muldiv_async(9, val, use_librf);
//C++17:
//auto [a, b] = co_await muldiv_async(9, val, use_librf);
std::string result = co_await tostring_async(std::get<0>(ab) + std::get<1>(ab), use_librf);
std::cout << result << std::endl;
}
#ifndef __clang__
catch (const std::exception& e)
{
std::cout << "exception signal : " << e.what() << std::endl;
}
catch (...)
{
std::cout << "exception signal : who knows?" << std::endl;
}
#endif
2019-04-21 09:05:04 +08:00
};
2019-04-21 09:05:04 +08:00
resumef::this_scheduler()->run_until_notask();
}