boost asio 需要使用智能指针而不能使用普通指针的原因

来源:互联网 发布:站长域名查询 编辑:程序博客网 时间:2024/05/19 05:04

当使用asio做服务器时,需要一个会话管理器来管理所有的会话,类似下面这样:

class SessionMgr{private:boost::shared_mutex _rw_mtx;boost::unordered_map<uint32, boost::shared_ptr<Session> > _session_list;boost::unordered_map<uint32, boost::shared_ptr<Session> >::iterator _iterator;public:void AddSession(boost::shared_ptr<Session> session);void DelSession(boost::shared_ptr<Session> session);boost::shared_ptr<Session> GetSession(uint32 session_key);boost::shared_ptr<Session> GetFirstSession();boost::shared_ptr<Session> GetNextSession();};

此类中使用会话指针时,不能直接用Session *来作为容器的元素,需要使用智能指针boost::shared_ptr<Session>来作为boost::unordered_map的元素,

原因是网络通信使用了异步机制,程序随时会把session对象的指针交给系统底层使用,由系统回调session对象的函数,比如:

void Session::HandleWrite(const boost::system::error_code &error){if(error){cout << "[ERROR] func:" << __FUNCTION__ << ", error:" << error.value() << "," << error.message() << endl;DeleteSelf();return;}if(session_info.need_close){return;}_socket.async_read_some(boost::asio::buffer(session_info.pkg_buf, MAX_SESSION_PKG_SIZE),boost::bind(&Session::HandleRead, shared_from_this(), boost::asio::placeholders::error, boost::asio::placeholders::bytes_transferred));}
在处理数据包写入完成时,会异步读取数据,将session对象自身的指针和函数HandleRead的地址传送给系统底层,系统读取成功时会自动回调这个成员函数。

如果是普通指针,那么会话管理器SessionMgr在删除会话时会立即释放掉Session对象,那么系统底层在回调成员函数时就会发生错误。

所以需要使用智能指针,在需要关闭会话时,只需要将Session指针从SessionMgr的容器中移除,不需要释放Session对象,再做一个标志表明会话需要删除就可以了。

在Session处理函数中判断标志,如果需要关闭的标志存在,则取消下一步的动作,比如,异步读取,异步写入等等,这样,Session指针的引用次数就不会再增加,当引用次数为0时,Session对象就会自动释放了。


原创粉丝点击