您的位置:首页 > 财经 > 产业 > 新媒体营销论文_做app多少钱_简述企业网站如何推广_百度网站推广排名优化

新媒体营销论文_做app多少钱_简述企业网站如何推广_百度网站推广排名优化

2025/4/17 1:49:05 来源:https://blog.csdn.net/z3363/article/details/146489059  浏览:    关键词:新媒体营销论文_做app多少钱_简述企业网站如何推广_百度网站推广排名优化
新媒体营销论文_做app多少钱_简述企业网站如何推广_百度网站推广排名优化

协程

c++20的协程三大标签:“性能之优秀”,“开发之灵活”,“门槛之高”

在讲解c++的协程使用前,我们需要先明白协程是什么,协程可以理解为用户态的线程,它需要由程序来进行调度,如上下文切换与调度设计都需要程序来设计,并且协程运行在单个线程中,这就成就了协程的低成本,更直接一点的解释可以说,协程就是一种可以被挂起与恢复的特殊函数。
协程并不会真正地“脱离”当前的线程,它只是让控制流从一个函数流转到另一个地方,然后再回来。这个过程是 轻量级非阻塞 的。
协程可以分为两种类型对称型与非对称型协程:

  • 非对称协程:控制权的转移是单向的,总是从调用者协程转移到被调用者协程,并且被调用者协程必须将控制权返回给调用者协程。
  • 对称协程:控制权可以在任意两个协程之间直接转移。协程在执行过程中可以选择将控制权交给其他任何协程,而不依赖于特定的调用层次关系。

c++协程的基本概念

c++提供的协程是典型的非对称协程,c++中的协程有两个特定条件:

  1. 协程函数必须包含至少一个协程关键字(co_awaitco_yieldco_return
    • co_await: 主要用于暂停协程的执行,同时等待某个异步操作完成。在协程执行到 co_await 表达式时,它会将控制权交还给调用者,待异步操作完成后,协程再恢复执行。
    • co_yield: 主要用于生成器模式,它会产生一个值,然后暂停协程的执行。每次调用生成器的迭代器时,协程会恢复执行,直到下一个 co_yield 或者协程结束。
    • co_return: 用于结束协程的执行,并返回一个值(如果协程有返回类型)。当协程执行到 co_return 时,它会销毁自身的栈帧,并将控制权交还给调用者。
      这里先简单介绍协程的基本概念和使用,后面会结合更多代码解释每一个点
  2. 返回值必须是实现了promise_type结构体的一个类或者结构体;
struct Task {struct promise_type {Task get_return_object() { return {}; }std::suspend_never initial_suspend() { return {}; }std::suspend_never final_suspend() noexcept { return {}; }void return_void() {}void unhandled_exception() {}};
};Task task() {std::cout << "Starting coroutine operation..." << std::endl;co_await std::suspend_always{};std::cout << "Coroutine operation completed." << std::endl;
}int main() {task();std::cout << "Main function continues..." << std::endl;return 0;
}// 输出
Starting coroutine operation...
Main function continues...

这个代码简单演示了协程的使用,我们可以看到输出只有两条,并没有输出Coroutine operation completed.,这个程序的运行流程是main函数调用task后,先保存当前上下文,再将当前线程的上下文切换成task的,接下来输出一条内容运行到co_await时,再次保存当前上下文,不过这次会将上下文切换回协程的调用者也就是main,我们后续没有再回到task运行最后一条命令,这里也就没有相关输出了
协程函数的返回类型有特殊要求。返回类型主要用于表示协程的句柄或包装器,它提供了一种方式来管理协程的生命周期,以及与协程进行交互。promise_type主要用来定义协程的生命周期和行为。

  • promise_type必须实现以下关键方法:
    • get_return_object():该函数在协程开始执行之前被调用,其作用是返回协程的返回对象
    • initial_suspend():在协程开始执行时调用,用于决定协程是否在开始时就暂停。std::suspend_never 表示协程不会在开始时暂停,会立即执行。
    • final_suspend():在协程结束执行时调用,用于决定协程是否在结束时暂停。std::suspend_never 表示协程不会在结束时暂停,会立即销毁。
    • unhandled_exception():处理未捕获的异常。
      下面的方法可选
  • return_void() 当调用co_return时会触发这个函数,它触发于final_suspend()之前
  • yield_value() 主要用于生成器返回值
    我们再来明确一下调用时刻,get_return_object()是在协程运行前就被调用了,当存放task();这一行时就会调用get_return_object()函数,因为我们这个演示比较简单不会返回如何东西,我们会在下面的演示中详细介绍,initial_suspend()是在协程开始执行时调用,这个时刻是介于get_return_object()与真正开始运行之间的时候,这些函数名必须一模一样

生成器

template<typename T>
struct Generator {struct promise_type {T value_;auto get_return_object() { return Generator{this}; }auto initial_suspend() noexcept { return std::suspend_always{}; }auto final_suspend() noexcept { return std::suspend_always{}; }void unhandled_exception() { std::terminate(); }auto yield_value(T val) {value_ = val;return std::suspend_always{};}};using Handle = std::coroutine_handle<promise_type>;Handle handle_;explicit Generator(promise_type* p) : handle_(Handle::from_promise(*p)) {}~Generator() { if (handle_) handle_.destroy(); }T next() {if (!handle_.done()) handle_.resume();return handle_.promise().value_;}
};// 斐波那契生成器
Generator<int> fibonacci() {int a = 0, b = 1;while (true) {co_yield a;int temp = a;a = b;b = temp + b;}
}int main() {auto fib = fibonacci();for (int i = 0; i < 10; ++i) {std::cout << fib.next() << " "; }
}// 输出 
0 1 1 2 3 5 8 13 21 34

这个代码使用了一种叫做生成器的特殊类,它需要我们自己实现,作用是获得一次协程的返回值,fibonacci中使用co_yield返回了a,之前我们介绍过co_yield它的作用是,产生一个值然后暂停协程的执行
这里的promise_type结构体多了两个东西,yield_value()函数的作用是把传入的值存储到 value_ 中,并且返回 std::suspend_always{},使得协程暂停。当使用co_yield关键字时就相当于调用了这个函数,T value_用于存放我们的返回值
promise_type结构体外的东西,都是我们可以自定义的这里最重要的就是,代码中的std::coroutine_handle<>,它是一个协程句柄表示一个协程实例的句柄,允许开发者手动控制协程的恢复、销毁或访问其内部状态,它通常实现为一个指针,直接指向协程的状态帧 ,C++20 协程机制中的核心工具类,用于直接操作协程的生命周期和执行流程。它提供了一种底层但灵活的方式来管理协程的状态。我们介绍几个常用的函数

  • Handle::from_promise() 这是一个静态成员函数,用于从 promise_type 对象创建一个协程句柄。在协程的 promise_type 中,通常会使用这个函数来获取协程句柄,以便在返回对象中保存。
  • resume() 恢复协程的执行。如果协程当前处于暂停状态,调用这个函数会让协程从暂停点继续执行,直到下一个暂停点或者协程结束。
  • destroy() 销毁协程,释放协程占用的资源。在协程执行完毕或者不再需要时,应该调用这个函数来避免内存泄漏。
  • bool done() 检查协程是否已经完成。如果协程已经执行完毕,返回 true;否则返回 false
  • promise_type& promise() 返回与协程句柄关联的 promise_type 对象的引用。通过这个引用,你可以访问和修改协程的状态和数据。
    next()函数用于返回值并恢复协程的执行,这里我们通过get_return_object()构建并返回我们的生成器

自定义可等待对象

自定义可等待对象需要满足特定的接口要求,主要涉及await_readyawait_suspendawait_resume这三个成员函数。

  • await_ready:用于判断是否可以立即恢复协程的执行。若返回true,协程会马上恢复;若返回false,协程就会被挂起。
  • await_suspend:在协程挂起时调用,这里可以执行异步操作。在示例中,使用一个新线程来模拟异步操作,操作完成后恢复协程。
  • await_resume:在协程恢复执行时调用,返回异步操作的结果。
template <typename T>
struct FutureAwaitable {std::future<T>& future;bool await_ready() const noexcept {return future.wait_for(std::chrono::seconds(0)) == std::future_status::ready;}void await_suspend(std::coroutine_handle<> handle) const {std::thread([this, handle]() mutable {future.wait();handle.resume();}).detach();}T await_resume() {return future.get();}
};template <typename T>
FutureAwaitable<T> co_awaitable(std::future<T>& future) {return {future};
}struct AsyncTask {struct promise_type {std::promise<int> promise;std::future<int> result = promise.get_future();auto get_return_object() { return AsyncTask{this}; }auto initial_suspend() { return std::suspend_never{}; }auto final_suspend() noexcept { return std::suspend_always{}; }void return_value(int val) {promise.set_value(val);}void unhandled_exception() {std::terminate();}};using Handle = std::coroutine_handle<promise_type>;Handle handle_;explicit AsyncTask(promise_type* p) : handle_(Handle::from_promise(*p)) {}~AsyncTask() {if (handle_) {handle_.destroy();}}int get() {return handle_.promise().result.get();}
};AsyncTask simulate_async_io() {auto future = std::async([] {std::this_thread::sleep_for(std::chrono::seconds(1));return 42;});auto result = co_await co_awaitable(future);co_return result;
}int main() {auto task = simulate_async_io();std::cout << "Waiting..." << std::endl;std::cout << "Result: " << task.get() << std::endl; // 输出: 42return 0;
}

这个代码演示了co_await结合自定义等待对象,实现的异步调用,当我们触发co_await时会等待异步操作的完成,并获得返回值,在等待时协程会挂起让出cpu资源
await_suspend()函数中我们在另一个线程中恢复了协程,协程的执行线程由恢复它的线程决定。handle.resume() 是恢复协程执行的关键操作,它会从协程上次挂起的位置接着执行。由于 handle.resume() 是在新创建的 std::thread 线程里被调用的,所以协程恢复后会继续在这个新线程里执行后续代码。
return_value()yield_value()函数类似,它是用来处理co_return关键字的返回值的

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com