您的位置:首页 > 房产 > 建筑 > 网页背景图片素材_动漫网站设计源代码_自己怎么免费做百度推广_外贸网站有哪些

网页背景图片素材_动漫网站设计源代码_自己怎么免费做百度推广_外贸网站有哪些

2025/1/1 11:05:49 来源:https://blog.csdn.net/gzjimzhou/article/details/144298565  浏览:    关键词:网页背景图片素材_动漫网站设计源代码_自己怎么免费做百度推广_外贸网站有哪些
网页背景图片素材_动漫网站设计源代码_自己怎么免费做百度推广_外贸网站有哪些

在这里插入图片描述

《C++ 异步妙韵》
C++ 异步展新章, async 开启并行航。 函数异步去闯荡, 不必同步苦守望。

future 犹如希望光, 异步结果有处藏。 等待之时心不慌, 获取成果见真章。

promise 把诺来担当, 结果设定不迷茫。 与 future 紧捆绑, 异步流程稳起航。

await 恰似静候郎, future 前面来站岗。 异步任务渐明朗, 代码优雅韵悠长。

一、c++异步的简单用法

以下是一个使用asyncawait(模拟其语义)、futurepromise的示例代码:

#include <iostream>
#include <future>
#include <thread>
#include <chrono>// 模拟await操作,等待future完成并获取结果
template<typename T>
T await(std::future<T>& fut) {return fut.get();
}// 一个耗时的计算任务
int heavyComputation(int num) {std::this_thread::sleep_for(std::chrono::seconds(2));  // 模拟耗时操作return num * 2;
}int main() {// 创建一个promise和对应的futurestd::promise<int> prom;std::future<int> fut = prom.get_future();// 使用async启动异步任务std::async(std::launch::async, [&]() {int result = heavyComputation(5);  // 执行耗时计算prom.set_value(result);  // 设置promise的值,使future可获取结果});// 模拟await等待异步任务完成并获取结果int finalResult = await(fut);std::cout << "最终结果: " << finalResult << std::endl;return 0;
}

在这个示例中:

  • async用于启动一个异步任务,这里传入一个lambda表达式,在其中执行heavyComputation这个耗时操作。
  • promise用来在异步任务完成时设置结果,future则用于获取这个结果。异步任务计算出heavyComputation(5)的结果后,通过promiseset_value设置值,这样对应的future就可以获取到该结果。
  • 自定义的await函数接受一个future,通过调用futureget函数等待future完成并获取其结果,就如同在等待异步任务的最终成果,体现了代码在异步操作过程中的等待与获取结果的逻辑,类似于诗中所描述的各个角色在异步流程中的作用与交互。

请注意,实际的await在C++ 中可能有更复杂的应用场景和语法(例如在C++ 20协程中),这里只是一个简单的模拟来解释相关概念的协同工作。

二、使用过程中的容易犯错的案例

《并发编程警言》
async 并行路,启动策略顾。 async 若 defer,串行非正途。

future 结果处,获取莫多步。 异常应先捕,重复调用误。

promise 诺来付,生命周期护。 设值锁别堵,匹配莫疏忽。

并发编程酷,细节心中驻。

以下是使用asyncfuturepromise时常见的 5 个较难发现和解决的错误案例:

案例一:未正确处理 future 的异常

#include <iostream>
#include <future>
#include <thread>
#include <vector>// 一个可能抛出异常的函数
int divide(int a, int b) {if (b == 0) {throw std::runtime_error("除数不能为 0");}return a / b;
}int main() {std::vector<std::future<int>> futures;// 启动多个异步任务for (int i = 0; i < 5; ++i) {// 错误:未正确处理 future 可能出现的异常futures.push_back(std::async(std::launch::async, divide, 10, i));}// 尝试获取结果for (auto& f : futures) {try {std::cout << f.get() << " ";} catch (const std::exception& e) {std::cout << "捕获异常: " << e.what() << " ";}}return 0;
}

解释:在这个例子中,当i0时,divide函数会抛出异常。由于async创建的异步任务是独立运行的,如果不进行特殊处理,异常可能会在后台线程中未被捕获而导致程序异常终止。虽然在main函数中通过get尝试捕获异常,但如果有多个future,这种方式会使代码变得复杂且难以维护。正确的做法可以在异步任务的函数内部进行异常处理,或者使用std::futurewait_for结合std::future_status来检查任务状态并处理异常。

案例二:future 被多次获取结果

#include <iostream>
#include <future>int add(int a, int b) {return a + b;
}int main() {std::future<int> f = std::async(std::launch::async, add, 3, 5);// 第一次获取结果int result1 = f.get();std::cout << "第一次结果: " << result1 << std::endl;// 错误:再次获取结果,会导致程序错误try {int result2 = f.get();std::cout << "第二次结果: " << result2 << std::endl;} catch (const std::future_error& e) {std::cout << "捕获 future 错误: " << e.what() << std::endl;}return 0;
}

解释:futureget函数是阻塞的并且只能获取一次结果。一旦获取了结果,future就进入了已就绪状态,再次调用get会抛出std::future_error异常。在实际应用中,如果不小心多次调用get,可能会导致程序逻辑错误且难以排查,尤其是在复杂的代码结构中,可能忘记了future已经获取过结果。

案例三:async 任务的生命周期管理不当

#include <iostream>
#include <future>class MyClass {
public:MyClass() {// 启动异步任务f = std::async(std::launch::async, [this]() {// 模拟一些耗时操作for (int i = 0; i < 1000000; ++i) {// 错误:在异步任务中持有 MyClass 的 this 指针,可能导致生命周期问题if (i % 1000 == 0) {std::cout << "MyClass 内部数据: " << data << " ";}}});}~MyClass() {// 尝试获取异步任务结果,可能导致死锁或未定义行为try {f.get();} catch (const std::exception& e) {std::cout << "捕获异常: " << e.what() << " ";}}private:int data = 10;std::future<void> f;
};int main() {{MyClass obj;// 在此处 MyClass 的析构函数会被调用,可能引发问题}return 0;
}

解释:在MyClass的构造函数中启动了一个异步任务,并且在异步任务中使用了this指针访问MyClass的数据成员。如果MyClass对象在异步任务完成之前被销毁,会导致this指针悬空,产生未定义行为。同时,在析构函数中尝试获取异步任务的结果,如果异步任务还未完成,可能会导致死锁(例如,如果async使用的是std::launch::deferred策略且析构函数在主线程中调用)或者其他异常情况。正确的做法是确保MyClass对象的生命周期长于异步任务,或者在异步任务中使用std::shared_ptr来管理MyClass对象的生命周期。

案例四:错误的线程启动策略导致性能问题

#include <iostream>
#include <future>
#include <thread>
#include <vector>
#include <chrono>// 一个简单的任务函数
void task() {std::this_thread::sleep_for(std::chrono::seconds(1));std::cout << "任务完成" << std::endl;
}int main() {std::vector<std::future<void>> futures;// 错误:使用 std::launch::deferred 启动多个任务,导致串行执行for (int i = 0; i < 5; ++i) {futures.push_back(std::async(std::launch::deferred, task));}// 等待所有任务完成for (auto& f : futures) {f.wait();}return 0;
}

解释:std::launch::deferred策略会将任务延迟到futurewaitget函数被调用时才执行。在这个例子中,创建了多个future并都使用了std::launch::deferred策略,当循环调用wait时,任务实际上是串行执行的,而不是并行执行,这违背了使用async进行异步并行任务处理的初衷,可能导致性能严重下降。应该根据任务的性质和需求选择合适的async启动策略,如std::launch::async表示立即在新线程中启动任务(如果可能),或者使用std::launch::async | std::launch::deferred让系统根据情况选择。

案例五:promise 和 future 不匹配导致死锁

#include <iostream>
#include <future>
#include <thread>
#include <mutex>std::mutex mtx;
// 函数用于设置 promise 的值
void setValue(std::promise<int>& prom) {std::lock_guard<std::mutex> guard(mtx);// 错误:在持有锁的情况下设置 promise 的值,可能导致死锁prom.set_value(42);
}int main() {std::promise<int> prom;std::future<int> fut = prom.get_future();// 启动一个线程来设置 promise 的值std::thread t(setValue, std::ref(prom));// 尝试获取 future 的值,可能导致死锁std::cout << fut.get() << std::endl;t.join();return 0;
}

解释:在setValue函数中,在持有互斥锁mtx的情况下设置promise的值。如果在main函数中,主线程在获取future的值(fut.get)时,由于fut.get会阻塞直到promise被设置值,而此时设置值的线程在等待锁,就会导致死锁。正确的做法是避免在设置promise值时持有可能被future获取操作阻塞的锁,或者重新设计锁的使用范围和时机。

三、高安全实时系统编程中的注意事项

《并发编程慎行歌》
async 任务勿轻搁,资源引用莫悬着。 局部变量若用过,智能指针来护呵。

并发数量细量度,线程池设防逾度。 任务狂抛资源枯,合理管控稳舟渡。

超时取消应有术,原子变量作标鹄。 实时系统限时务,机制完备免踟蹰。

异常处理要周普,内外兼顾莫粗疏。 任务抛错勿疏忽,安全稳行心不怵。

数据同步莫相忤,共享资源锁防护。 竞争若起乱如荼,同步机制除迷雾。
以下是在高安全可靠实时系统开发中使用 asyncfuturepromise 的一些注意事项及示例:

注意事项一:避免资源泄漏与悬空引用

在异步任务中,如果不小心使用了指向局部变量或即将被销毁对象的引用或指针,可能会导致资源泄漏或悬空引用问题。

#include <iostream>
#include <future>class Resource {
public:Resource() { std::cout << "Resource created\n"; }~Resource() { std::cout << "Resource destroyed\n"; }
};std::future<void> badAsyncFunction() {Resource localResource;  // 局部资源// 错误:返回一个包含局部资源引用的 future,当 future 被获取结果时可能导致资源已被销毁return std::async(std::launch::async, [&]() {std::this_thread::sleep_for(std::chrono::seconds(1));// 这里引用了局部资源,可能导致悬空引用std::cout << "Using localResource in async task\n"; });
}int main() {std::future<void> f = badAsyncFunction();// 等待异步任务完成f.wait();return 0;
}

正确的做法是确保异步任务中使用的资源具有合适的生命周期,例如使用智能指针来管理资源:

#include <iostream>
#include <future>
#include <memory>class Resource {
public:Resource() { std::cout << "Resource created\n"; }~Resource() { std::cout << "Resource destroyed\n"; }
};std::future<void> goodAsyncFunction() {// 使用智能指针管理资源std::shared_ptr<Resource> sharedResource = std::make_shared<Resource>();return std::async(std::launch::async, [sharedResource]() {std::this_thread::sleep_for(std::chrono::seconds(1));std::cout << "Using sharedResource in async task\n"; });
}int main() {std::future<void> f = goodAsyncFunction();f.wait();return 0;
}

注意事项二:控制任务的并发度与资源使用

在高安全可靠的实时系统中,无节制地创建大量异步任务可能导致系统资源耗尽,如线程过多导致上下文切换开销过大或内存不足。

#include <iostream>
#include <future>
#include <vector>// 一个简单但可能消耗资源的任务
void resourceConsumingTask() {std::vector<int> largeVector(1000000);  // 分配较大内存// 模拟一些计算操作for (int i = 0; i < largeVector.size(); ++i) {largeVector[i] = i * i;}
}int main() {// 错误:创建过多的异步任务可能耗尽系统资源std::vector<std::future<void>> futures;for (int i = 0; i < 100; ++i) {  // 假设 100 个任务过多futures.push_back(std::async(std::launch::async, resourceConsumingTask));}// 等待所有任务完成for (auto& f : futures) {f.wait();}return 0;
}

应该根据系统资源和任务需求合理控制并发度,例如使用线程池来管理任务执行:

#include <iostream>
#include <future>
#include <vector>
#include <thread>
#include <mutex>
#include <queue>// 线程池类
class ThreadPool {
public:ThreadPool(size_t numThreads) {for (size_t i = 0; i < numThreads; ++i) {workers.push_back(std::thread([this]() {while (true) {std::function<void()> task;{std::unique_lock<std::mutex> lock(queueMutex);condition.wait(lock, [this]() {return stop ||!tasks.empty();});if (stop && tasks.empty()) {return;}task = std::move(tasks.front());tasks.pop();}task();}}));}}~ThreadPool() {{std::unique_lock<std::mutex> lock(queueMutex);stop = true;}condition.notify_all();for (std::thread& worker : workers) {worker.join();}}template<typename F, typename... Args>auto enqueue(F&& f, Args&&... args) -> std::future<typename std::result_of<F(Args...)>::type> {using return_type = typename std::result_of<F(Args...)>::type;auto task = std::make_shared<std::packaged_task<return_type()>>(std::bind(std::forward<F>(f), std::forward<Args>(args)...));std::future<return_type> res = task->get_future();{std::unique_lock<std::mutex> lock(queueMutex);tasks.push([task]() {(*task)();});}condition.notify_one();return res;}private:std::vector<std::thread> workers;std::queue<std::function<void()>> tasks;std::mutex queueMutex;std::condition_variable condition;bool stop = false;
};// 资源消耗任务
void resourceConsumingTask() {std::vector<int> largeVector(1000000);for (int i = 0; i < largeVector.size(); ++i) {largeVector[i] = i * i;}
}int main() {// 创建一个合理大小的线程池,例如 4 个线程ThreadPool pool(4);std::vector<std::future<void>> futures;for (int i = 0; i < 10; ++i) {  // 提交 10 个任务到线程池futures.push_back(pool.enqueue(resourceConsumingTask));}// 等待所有任务完成for (auto& f : futures) {f.wait();}return 0;
}

注意事项三:处理任务超时与取消

在实时系统中,任务可能需要在规定时间内完成,如果超时则可能需要采取相应措施,如取消任务或进行错误处理。

#include <iostream>
#include <future>
#include <chrono>// 一个可能耗时的任务
int longRunningTask(int seconds) {std::this_thread::sleep_for(std::chrono::seconds(seconds));return seconds;
}int main() {std::future<int> f = std::async(std::launch::async, longRunningTask, 5);// 等待 3 秒检查任务状态std::future_status status = f.wait_for(std::chrono::seconds(3));if (status == std::future_status::ready) {std::cout << "任务在 3 秒内完成,结果为: " << f.get() << std::endl;} else if (status == std::future_status::timeout) {// 错误:没有合适的任务取消机制,仅能判断超时std::cout << "任务超时,没有处理取消操作\n";// 可以尝试在此处进行一些清理或错误报告操作}return 0;
}

一种处理方式是结合 std::atomic 变量来实现简单的任务取消机制:

#include <iostream>
#include <future>
#include <chrono>
#include <atomic>// 一个可取消的可能耗时任务
int cancellableLongRunningTask(int seconds, std::atomic<bool>& cancelFlag) {for (int i = 0; i < seconds; ++i) {if (cancelFlag.load()) {// 任务被取消std::cout << "任务被取消\n";return -1;}std::this_thread::sleep_for(std::chrono::seconds(1));}return seconds;
}int main() {std::atomic<bool> cancelFlag(false);std::future<int> f = std::async(std::launch::async, cancellableLongRunningTask, 5, std::ref(cancelFlag));// 等待 3 秒检查任务状态std::future_status status = f.wait_for(std::chrono::seconds(3));if (status == std::future_status::ready) {std::cout << "任务在 3 秒内完成,结果为: " << f.get() << std::endl;} else if (status == std::future_status::timeout) {// 设置取消标志cancelFlag.store(true);try {// 尝试获取结果,如果任务已被取消会抛出异常std::cout << "任务超时,已取消,结果为: " << f.get() << std::endl;} catch (const std::future_error& e) {std::cout << "捕获到 future 错误: " << e.what() << " 因为任务被取消\n";}}return 0;
}

注意事项四:确保异常安全

在异步任务中抛出的异常如果未被正确处理,可能导致程序不稳定或资源泄漏。

#include <iostream>
#include <future>// 一个可能抛出异常的任务
int errorProneTask(int value) {if (value == 0) {throw std::runtime_error("Invalid value");}return 10 / value;
}int main() {std::future<int> f = std::async(std::launch::async, errorProneTask, 0);// 错误:没有正确处理可能抛出的异常,可能导致程序崩溃try {std::cout << "结果为: " << f.get() << std::endl;} catch (const std::runtime_error& e) {std::cout << "捕获到异常: " << e.what() << std::endl;}// 如果在异常处理后有资源清理等操作,可能因异常未正确传播而被遗漏return 0;
}

正确的做法是在异步任务内部尽可能处理异常,并在外部也进行适当的异常处理与资源清理:

#include <iostream>
#include <future>// 一个异常安全处理的任务
int errorProneTask(int value) {try {if (value == 0) {throw std::runtime_error("Invalid value");}return 10 / value;} catch (const std::runtime_error& e) {// 在任务内部记录错误日志或进行一些局部清理操作std::cerr << "任务内部捕获到异常: " << e.what() << std::endl;// 可以根据情况重新抛出异常或返回错误码throw; }
}int main() {std::future<int> f = std::async(std::launch::async, errorProneTask, 0);try {std::cout << "结果为: " << f.get() << std::endl;} catch (const std::runtime_error& e) {std::cout << "捕获到异常: " << e.what() << std::endl;// 在这里可以进行全局的资源清理或错误报告操作}return 0;
}

注意事项五:保证数据一致性与同步

当多个异步任务访问共享数据时,需要进行合适的数据同步操作,否则可能导致数据竞争和不一致性。

#include <iostream>
#include <future>
#include <vector>
#include <mutex>// 共享数据
int sharedData = 0;
std::mutex dataMutex;// 一个修改共享数据的任务
void incrementData() {for (int i = 0; i < 1000; ++i) {// 错误:没有正确的锁保护,可能导致数据竞争sharedData++; }
}int main() {std::vector<std::future<void>> futures;for (int i = 0; i < 10; ++i) {futures.push_back(std::async(std::launch::async, incrementData));}// 等待所有任务完成for (auto& f : futures) {f.wait();}std::cout << "最终共享数据值: " << sharedData << std::endl;return 0;
}

应该使用互斥锁或其他同步机制来保护共享数据:

#include <iostream>
#include <future>
#include <vector>
#include <mutex>// 共享数据
int sharedData = 0;
std::mutex dataMutex;// 一个修改共享数据的任务
void incrementData() {for (int i = 0; i < 1000; ++i) {// 使用互斥锁保护共享数据std::lock_guard<std::mutex> guard(dataMutex);sharedData++; }
}int main() {std::vector<std::future<void>> futures;for (int i = 0; i < 10; ++i) {futures.push_back(std::async(std::launch::async, incrementData));}// 等待所有任务完成for (auto& f : futures) {f.wait();}std::cout << "最终共享数据值: " << sharedData << std::endl;return 0;
}

在高安全可靠实时系统开发中,要充分考虑这些注意事项,以确保系统的稳定性、正确性和高效性。

版权声明:

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

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