您的位置:首页 > 新闻 > 资讯 > 建筑装修设计网站大全_app制作平台大全_福州seo按天收费_网址最全的浏览器

建筑装修设计网站大全_app制作平台大全_福州seo按天收费_网址最全的浏览器

2025/1/10 9:22:30 来源:https://blog.csdn.net/weixin_74113106/article/details/142577245  浏览:    关键词:建筑装修设计网站大全_app制作平台大全_福州seo按天收费_网址最全的浏览器
建筑装修设计网站大全_app制作平台大全_福州seo按天收费_网址最全的浏览器

文章目录

  • 一.概念辨析
    • 1.什么是连接?为什么要自己封装 Connection
  • 二.实现思路
  • 三.代码实践

一.概念辨析

1.什么是连接?为什么要自己封装 Connection

连接就是对 TCP 连接的封装,Connection 和 TcpConnectionPtr 是一一对应的关系。我们之所以要自己封装 Connenction,因为多个信道共用一个 TCP 连接,我们要管理这种从属关系,便于查找和删除信道

二.实现思路

成员变量:

  1. muduo 库中的 TCP 连接
  2. 虚拟机句柄
  3. 协议处理句柄
  4. 消费者管理句柄
  5. 线程池句柄
    以上这 5 个成员都用于初始化信道
  6. 信道管理句柄(每个 Connection 都私有一份)
    成员方法:
  7. 创建信道
  8. 删除信道
  9. 根据信道 id 获取信道

三.代码实践

#pragma once
#include "muduo/protobuf/codec.h"
#include "muduo/protobuf/dispatcher.h"
#include "muduo/base/Logging.h"
#include "muduo/base/Mutex.h"
#include "muduo/net/EventLoop.h"
#include "muduo/net/TcpClient.h"
#include "muduo/net/EventLoopThread.h"
#include "muduo/base/CountDownLatch.h"
#include "../common/ThreadPool.hpp"
#include "Channel.hpp"
#include <functional>
#include <iostream>
namespace ns_connection
{using ThreadPoolPtr = std::shared_ptr<ns_tp::ThreadPool>;using ProtobufCodecPtr = std::shared_ptr<ProtobufCodec>;using ProtobufDispatcherPtr = std::shared_ptr<ProtobufDispatcher>;using ChannelPtr = std::shared_ptr<ns_channel::Channel>;using ThreadPoolPtr = std::shared_ptr<ns_tp::ThreadPool>;using CommonResponsePtr = std::shared_ptr<ns_protocol::CommomResponse>;using PushMessageResonsePtr = std::shared_ptr<ns_protocol::PushMessageResponse>;/************ Connection是对底层用于通信的TCP套接字封装(muduo库中的TcpConnectionPtr)* 一个Connection中包含多个信道,当Connection关闭,信道也会销毁* ******************/class Connection{private:muduo::net::EventLoopThread _loopThread;muduo::CountDownLatch _latch;muduo::net::TcpClient _client;muduo::net::TcpConnectionPtr _connPtr;ProtobufDispatcherPtr _distpatcherPtr;ProtobufCodecPtr _codecPtr;ns_channel::ChannelManager _channelManager;ThreadPoolPtr _threadPoolPtr;public:Connection(const std::string &serverIp, int serverPort, const ThreadPoolPtr &threadPoolPtr): _loopThread(),_latch(1),_client(_loopThread.startLoop(), muduo::net::InetAddress(serverIp, serverPort), "client"),_connPtr(),_channelManager(),_threadPoolPtr(threadPoolPtr){// 构造成员_distpatcherPtr = std::make_shared<ProtobufDispatcher>((std::bind(&Connection::onUnknownMessage,this,std::placeholders::_1,std::placeholders::_2,std::placeholders::_3)));_codecPtr = std::make_shared<ProtobufCodec>((std::bind(&ProtobufDispatcher::onProtobufMessage,_distpatcherPtr.get(),std::placeholders::_1,std::placeholders::_2,std::placeholders::_3)));// 给Client注册两个回调函数_client.setConnectionCallback(std::bind(&Connection::onConnection, this, std::placeholders::_1));_client.setMessageCallback(std::bind(&ProtobufCodec::onMessage, _codecPtr.get(), std::placeholders::_1,std::placeholders::_2, std::placeholders::_3));_distpatcherPtr->registerMessageCallback<ns_protocol::CommomResponse>(std::bind(&Connection::onCommonResponse,this, std::placeholders::_1,std::placeholders::_2,std::placeholders::_3));_distpatcherPtr->registerMessageCallback<ns_protocol::PushMessageResponse>(std::bind(&Connection::onRecvMessage,this, std::placeholders::_1,std::placeholders::_2,std::placeholders::_3));connect();}void connect(){_client.connect(); // 非阻塞_latch.wait();}ChannelPtr openChannel(){// 只是在本地建立了信道auto channelPtr = _channelManager.openChannel(_connPtr, _codecPtr, _threadPoolPtr);// 通过该信道发送建立信道的请求,要服务端也建立对应的信道if (!channelPtr->openChannel()){LOG(WARNING) << "打开信道失败" << endl;// 关闭本地的信道,防止内存泄漏_channelManager.closeChannel(channelPtr->_id);}return channelPtr;}void closeChannel(const ChannelPtr& channelPtr){// 发送关闭信道的请求,让服务端关闭信道channelPtr->closeOpenChannel();// 把本地信道关掉_channelManager.closeChannel(channelPtr->_id);}private:// 给_client设置的回调void onConnection(muduo::net::TcpConnectionPtr connPtr){if (connPtr->connected()){_connPtr = connPtr;_latch.countDown();}else{_connPtr.reset();}}void onUnknownMessage(const muduo::net::TcpConnectionPtr &connPtr,const MessagePtr &resp, muduo::Timestamp time){LOG(WARNING) << "未知响应" << endl;}// 业务处理函数void onCommonResponse(const muduo::net::TcpConnectionPtr &connPtr,const CommonResponsePtr &respPtr, muduo::Timestamp time){//LOG(DEBUG) << "收到CommonResponse, respId: " << respPtr->response_id() << endl;std::string channeId = respPtr->channel_id();auto channelPtr = _channelManager.getChannel(channeId);channelPtr->putCommonResponse(respPtr);}void onRecvMessage(const muduo::net::TcpConnectionPtr &connPtr,const PushMessageResonsePtr &respPtr, muduo::Timestamp time){//LOG(DEBUG) << "收到消息, body: " << respPtr->msg().saved_info().body() << endl;std::string channeId = respPtr->channel_id();auto channelPtr = _channelManager.getChannel(channeId);// 把处理消息的任务交给线程池来做_threadPoolPtr->push(std::bind(&ns_channel::Channel::consumeMessage, channelPtr.get(),respPtr->qname(), respPtr->msg()));}};
}

版权声明:

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

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