std::enable_shared_from_this消除异步回调引发的野指针问题

     std::enable_shared_from_this这个C++组件完美解决了异步回调发生时宿主对象销毁的问题。C++提供了lambda表达式和各种异步函数,std::thread,std::async或者其他框架api都提供了异步回调方法。特别是lambda在异步应用中实在是太方便了,比如发送一个http请求,接口可能是这样的api::get(url, [](response){ //解析response})。用起来方便,阅读起来也容易理解,但是同时也带来了问题,那就是回调回来的时候宿主类对象已经销毁了,但是大多数情况下我们再处理网络结果的时候需要用到宿主类对象的一些数据,这就造成了问题,宿主对象已经销毁了,他的成员数据也一起销毁了,继续调用只会导致无法预期的问题,所以这里我们有三种应对方法:

第一,宿主类销毁后,切断回调和宿主之前的关系.C++不提供这样的方法,需要自己设计,自己重新设置回调指针为空,调用回调函数的地方判断回调指针为空后就不调用。这种方法比较麻烦,需要手动设置,违反了简易自动化的原则。如果使用QT的话会比较方便,因为QT的信号槽机制会检测接收对象是否销毁。

第二,如果回调没有回来,延长宿主对象的生命周期,继承std::enable_shared_from_this可以做到这一点,那就是让lambda捕获宿主对象的shared_ptr指针(share_from_this()),这样在回调回来之前宿主对象不会销毁。但是问题也很明显,宿主对象的生命周期就依赖于这些回调被调用的时间,有些场景不合适。

第三, 是最好的方式,那就是检测对象是否销毁,如果销毁就不调用,如果没有继续执行,同样继承std::enable_shared_from_this可以做到这一点。让回调lambda捕获宿主对象的弱引用指针(weak_from_this()),在回调回来以后,检测weak_ptr是否可用,可用则获取shared_ptr保证宿主生命周期,然后执行其他方法。不可用直接返回。

#include 
#include 
#include 
#include 
#include 

class A : public std::enable_shared_from_this
{
public:
	A() { std::cout << __FUNCTION__ << std::endl; };
	~A() { std::cout << __FUNCTION__ << std::endl; };
	void Start()
	{
		auto weak = weak_from_this();
		std::thread t([weak]() { //捕获宿主弱引用指针
			std::this_thread::sleep_for(std::chrono::seconds(10));
			auto self = weak.lock();
			if (self) //判断宿主是否销毁
			{
				self->PrintSelf();
			}
		});
		t.detach();
	}
	void PrintSelf()
	{
		std::cout << printInfo << std::endl;
	}
private:
	std::string printInfo = " its my self.";
};

int main()
{
	{
		std::shared_ptr ptrA = std::make_shared(); //继承std::enable_shared_from_this类生命周期由shared_ptr托管
		ptrA->Start();
	}//ptrA超出作用域,对象销毁
	int wait;
	std::cin >> wait;
}

 

你可能感兴趣的:(C++)