CppGuide社区 CppGuide社区
首页
  • 🔥最新谷歌C++风格指南(含C++17/20)
  • 🔥C++17详解
  • 🔥C++20完全指南
  • 🔥C++23快速入门
  • C++语言面试问题集锦
  • 🔥交易系统开发岗位求职与面试指南 (opens new window)
  • 第1章 高频C++11重难点知识解析
  • 第2章 Linux GDB高级调试指南
  • 第3章 C++多线程编程从入门到进阶
  • 第4章 C++网络编程重难点解析
  • 第5章 网络通信故障排查常用命令
  • 第6章 高性能网络通信协议设计精要
  • 第7章 高性能服务结构设计
  • 第8章 Redis网络通信模块源码分析
  • 第9章 后端服务重要模块设计探索
  • 🚀 全部章节.pdf 下载 (opens new window)
  • 源码分析系列

    • leveldb源码分析
    • libevent源码分析
    • Memcached源码分析
    • TeamTalk源码分析
    • 优质源码分享 (opens new window)
    • 🔥远程控制软件gh0st源码分析
  • 从零手写C++项目系列

    • 🔥C++游戏编程入门(零基础学C++)
    • 🔥使用C++17从零开发一个调试器 (opens new window)
    • 🔥使用C++20从零构建一个完整的低延迟交易系统 (opens new window)
    • 🔥使用C++从零写一个C语言编译器 (opens new window)
    • 🔥从零用C语言写一个Redis
  • 🔥Windows 10系统编程
  • 🔥Linux 5.x内核开发与调试 完全指南 (opens new window)
  • TCP源码实现超详细注释版.pdf (opens new window)
  • Go语言特性

    • Go系统接口编程
    • 高效Go并发编程
    • Go性能调优
    • Go项目架构设计
  • Go项目实战

    • 🔥使用Go从零开发一个数据库
    • 🔥使用Go从零开发一个编译器 (opens new window)
    • 🔥使用Go从零开发一个解释器 (opens new window)
    • 🔥使用Go从零开发一个解释器 (opens new window)
    • 🔥用Go从零写一个编排器(类Kubernetes) (opens new window)
Rust编程指南
  • SQL零基础指南
  • MySQL开发与调试指南
GitHub (opens new window)
首页
  • 🔥最新谷歌C++风格指南(含C++17/20)
  • 🔥C++17详解
  • 🔥C++20完全指南
  • 🔥C++23快速入门
  • C++语言面试问题集锦
  • 🔥交易系统开发岗位求职与面试指南 (opens new window)
  • 第1章 高频C++11重难点知识解析
  • 第2章 Linux GDB高级调试指南
  • 第3章 C++多线程编程从入门到进阶
  • 第4章 C++网络编程重难点解析
  • 第5章 网络通信故障排查常用命令
  • 第6章 高性能网络通信协议设计精要
  • 第7章 高性能服务结构设计
  • 第8章 Redis网络通信模块源码分析
  • 第9章 后端服务重要模块设计探索
  • 🚀 全部章节.pdf 下载 (opens new window)
  • 源码分析系列

    • leveldb源码分析
    • libevent源码分析
    • Memcached源码分析
    • TeamTalk源码分析
    • 优质源码分享 (opens new window)
    • 🔥远程控制软件gh0st源码分析
  • 从零手写C++项目系列

    • 🔥C++游戏编程入门(零基础学C++)
    • 🔥使用C++17从零开发一个调试器 (opens new window)
    • 🔥使用C++20从零构建一个完整的低延迟交易系统 (opens new window)
    • 🔥使用C++从零写一个C语言编译器 (opens new window)
    • 🔥从零用C语言写一个Redis
  • 🔥Windows 10系统编程
  • 🔥Linux 5.x内核开发与调试 完全指南 (opens new window)
  • TCP源码实现超详细注释版.pdf (opens new window)
  • Go语言特性

    • Go系统接口编程
    • 高效Go并发编程
    • Go性能调优
    • Go项目架构设计
  • Go项目实战

    • 🔥使用Go从零开发一个数据库
    • 🔥使用Go从零开发一个编译器 (opens new window)
    • 🔥使用Go从零开发一个解释器 (opens new window)
    • 🔥使用Go从零开发一个解释器 (opens new window)
    • 🔥用Go从零写一个编排器(类Kubernetes) (opens new window)
Rust编程指南
  • SQL零基础指南
  • MySQL开发与调试指南
GitHub (opens new window)
  • 第1章高频C++11重难点知识解析

  • 第2章Linux GDB高级调试指南

  • 第3章C++多线程编程从入门到进阶

  • 第4章C++网络编程重难点解析

  • 第5章网络通信故障排查常用命令

  • 第6章高性能网络通信协议设计精要

  • 第7章高性能服务结构设计

    • 7.1 网络通信组件的效率问题
    • 7.2 最原始的服务器结构
    • 7.3 一个连接一个线程模型
    • 7.4 Reactor 模式
    • 7.5 one thread one loop 思想
    • 7.6 收数据与发数据的正确姿势
    • 7.7 发送/接收缓冲区设计要点
    • 7.8 网络库的分层设计
      • 7.9 后端服务中的定时器设计
      • 7.10 业务数据处理一定要单独开线程吗
      • 7.11 侵入式程序结构与非侵入式程序结构
      • 7.12 带有网络通信模块的服务器的经典结构
    • 第8章Redis 网络通信模块源码分析

    • 第9章后端服务重要模块设计探索

    • C++后端开发进阶
    • 第7章高性能服务结构设计
    zhangxf
    2023-04-05
    目录

    7.8 网络库的分层设计

    计算机界有一句经典名言:“计算机科学领域的任何问题都可以通过增加一个间接的中间层来解决”(Any problem in computer science can be solved by anther layer of indirection.)。这句话几乎概括了计算机系统软件体系结构的设计要点,整个体系结构从上到下都是按照严格的层次结构设计的,不仅是计算机系统软件整个体系是这样的,体系里面的每个组件比如 OS 本身,很多应用程序、软件系统甚至很多硬件结构都是按照这种层次的结构组织和设计的。

    # 7.8.1 网络库设计中的各个层

    在常见的网络通信库中,根据功能也可以分成很多层,根据离业务的远近从上到下依次是:

    • Session 层

      该层处于最上层,在设计上不属于网络框架本身的部分,其作用是记录各种业务状态数据和处理各种业务逻辑。业务逻辑处理完毕后,如果需要进行网络通信,则依赖 Connection 层进行数据收发。

      例如一个 session 类可能有如下接口和成员数据:

      class ChatSession
      {
      public:
          ChatSession(const std::shared_ptr<TcpConnection>& conn, int sessionid);
          virtual ~ChatSession();
          
          int32_t GetSessionId()
          {
              return m_id;
          }
      
          int32_t GetUserId()
          {
              return m_userinfo.userid;
          }
      
          std::string GetUsername()
          {
              return m_userinfo.username;
          }
      
          std::string GetNickname()
          {
              return m_userinfo.nickname;
          }
      
          std::string GetPassword()
          {
              return m_userinfo.password;
          }
      
          int32_t GetClientType()
          {
              return m_userinfo.clienttype;
          }
      
          int32_t GetUserStatus()
          {
              return m_userinfo.status;
          }
      
          int32_t GetUserClientType()
          {
              return m_userinfo.clienttype;
          }
      
          void SendUserStatusChangeMsg(int32_t userid, int type, int status = 0);
      
      private:
      	//各个业务逻辑处理方法
          bool Process(const std::shared_ptr<TcpConnection>& conn, const char* inbuf, size_t buflength);
          
          void OnHeartbeatResponse(const std::shared_ptr<TcpConnection>& conn);
          void OnRegisterResponse(const std::string& data, const std::shared_ptr<TcpConnection>& conn);
          void OnLoginResponse(const std::string& data, const std::shared_ptr<TcpConnection>& conn);
          void OnGetFriendListResponse(const std::shared_ptr<TcpConnection>& conn);
          void OnFindUserResponse(const std::string& data, const std::shared_ptr<TcpConnection>& conn);
          void OnChangeUserStatusResponse(const std::string& data, const std::shared_ptr<TcpConnection>& conn);
          void OnOperateFriendResponse(const std::string& data, const std::shared_ptr<TcpConnection>& conn);
          
          std::shared_ptr<TcpConnection> GetConnectionPtr()
          {
              if (tmpConn_.expired())
                  return NULL;
      
              return tmpConn_.lock();
          }
      	//调用下层Connection层发送数据的方法
          void Send(int32_t cmd, int32_t seq, const std::string& data);
          void Send(int32_t cmd, int32_t seq, const char* data, int32_t dataLength);
          void Send(const std::string& p);
          void Send(const char* p, int32_t length);
      
      private:
          int32_t           m_id;                 //session id
          OnlineUserInfo    m_userinfo;
          int32_t           m_seq;                //当前Session数据包序列号
          bool              m_isLogin;            //当前Session对应的用户是否已经登录
          
          //引用下层Connection层的成员变量
          std::weak_ptr<TcpConnection>    tmpConn_;
      };
      
      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      31
      32
      33
      34
      35
      36
      37
      38
      39
      40
      41
      42
      43
      44
      45
      46
      47
      48
      49
      50
      51
      52
      53
      54
      55
      56
      57
      58
      59
      60
      61
      62
      63
      64
      65
      66
      67
      68
      69
      70
      71
      72
      73
      74
      75
      76
      77
      78
      79
      80
      81
      82

      上述代码中除了业务状态数据和业务接口以外,还有一个 Send 系列的函数,这个函数依赖 Connection 对象进行数据收发。但是需要注意的是 Session 对象并不拥有 Connection 对象,也就是说 Session 对象不控制 Connection 对象的生命周期,这是因为虽然 Session 对象的主动销毁(如收到客户端不合理的数据,关闭 Session 对象)会引起 Connection 对象的销毁,但 Connection 对象本身也可能因为网络出错等原因被销毁,进而引起 Session 对象被销毁。因此,上述类接口描述中,ChatSession 类使用了一个弱指针(weak_ptr)来引用 TCPConnection 对象。这是需要注意的地方。

    • Connection 层

      该层是网络框架设计中最上面的一层(技术层的最上层),每一路客户端连接对应一个 Connection 对象。一般用于记录该路连接的各种状态,常见的状态信息有,如连接状态、数据收发缓冲区信息、数据流量记录状态、本端和对端地址和端口号信息等,同时也提供对各种网络事件的处理接口,这些接口或被本层自己使用,或被 Session 层使用。Connection 持有一个 Channel 对象,且掌管着 Channel 对象的生命周期。

      一个 Connection 对象可能提供的接口和记录的数据状态如下:

      class TcpConnection
      {
      	public:		
      		TcpConnection(EventLoop* loop,
      			            const string& name,
      			            int sockfd,
      			            const InetAddress& localAddr,
      			            const InetAddress& peerAddr);
      		~TcpConnection();
      
      		const InetAddress& localAddress() const { return localAddr_; }
      		const InetAddress& peerAddress() const { return peerAddr_; }
      		bool connected() const { return state_ == kConnected; }
      
      		
      		void send(const void* message, int len);
      		void send(const string& message);		
      		void send(Buffer* message);  // this one will swap data
      		void shutdown(); // NOT thread safe, no simultaneous calling
      		
      		void forceClose();
      
      		void setConnectionCallback(const ConnectionCallback& cb)
      		{
      			connectionCallback_ = cb;
      		}
      
      		void setMessageCallback(const MessageCallback& cb)
      		{
      			messageCallback_ = cb;
      		}
      		
      		void setCloseCallback(const CloseCallback& cb)
      		{
      			closeCallback_ = cb;
      		}
      		
      		void setErrorCallback(const ErrorCallback& cb)
      		{
      			errorCallback_ = cb;
      		}
      
      		Buffer* inputBuffer()
      		{
      			return &inputBuffer_;
      		}
      
      		Buffer* outputBuffer()
      		{
      			return &outputBuffer_;
      		}
      
      	private:
      		enum StateE { kDisconnected, kConnecting, kConnected, kDisconnecting };
      		void handleRead(Timestamp receiveTime);
      		void handleWrite();
      		void handleClose();
      		void handleError();
      		void sendInLoop(const string& message);
      		void sendInLoop(const void* message, size_t len);
      		void shutdownInLoop();
      		void forceCloseInLoop();
      		void setState(StateE s) { state_ = s; }
      
          private:
      		//连接状态信息
      		StateE                      state_;
      		//引用Channel对象
      		std::shared_ptr<Channel>    channel_;
      		//本端的地址信息
      		const InetAddress           localAddr_;
      		//对端的地址信息
      		const InetAddress           peerAddr_;
      		
      		ConnectionCallback          connectionCallback_;
      		MessageCallback             messageCallback_;
      		CloseCallback               closeCallback_;
      		ErrorCallback				errorCallback_;	
      		
      		//接收缓冲区
      		Buffer                      inputBuffer_;
      		//发送缓冲区
      		Buffer                      outputBuffer_; 
      		//流量统计类
      		CFlowStatistics				flowStatistics;
      };
      
      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      31
      32
      33
      34
      35
      36
      37
      38
      39
      40
      41
      42
      43
      44
      45
      46
      47
      48
      49
      50
      51
      52
      53
      54
      55
      56
      57
      58
      59
      60
      61
      62
      63
      64
      65
      66
      67
      68
      69
      70
      71
      72
      73
      74
      75
      76
      77
      78
      79
      80
      81
      82
      83
      84
      85
      86
    • Channel 层

      Channel 层一般持有一个 socket(Linux 下也叫 fd),是实际进行数据收发的地方,因而一个 Channel 对象会记录当前需要监听的各种网络事件(读写和出错事件)状态,同时提供对这些事件的状态的判断和增删改的接口。在部分网络库的实现中,Channel 对象管理着 socket 对象的生命周期,而另外一些库的实现则由 Connection 对象来管理 socket 的生命周期。如果实现是前者,则 Channel 对象也提供对 socket 进行创建和关闭的接口。由于 TCP 收发数据是全双工的(收发走独立的通道,互不影响),收发逻辑一般不会有什么依赖关系,但收发操作一般会在同一个线程中进行操作,这样的目的是为了防止收或发的过程中,改变了 socket 的状态,对另外一个操作产生影响。例如在一个线程中收数据时出错,关闭了连接,另外一个线程正在发送数据,这该情何以堪呢。

      一个 Channel 对象提供的函数接口和状态数据如下所示:

      class Channel
      {
      public:
      	Channel(EventLoop* loop, int fd);
      	~Channel();
      
      	void handleEvent(Timestamp receiveTime);
      	
      	int fd() const;
      	int events() const;
      	void set_revents(int revt);
      	void add_revents(int revt);
      	void remove_events();
      	bool isNoneEvent() const;
      
      	bool enableReading();
      	bool disableReading();
      	bool enableWriting();
      	bool disableWriting();
      	bool disableAll();
      
      	bool isWriting() const { return events_ & kWriteEvent; }
      
      private:		
      	const int                   fd_;
      	//当前需要检测的事件
      	int                         events_;
      	//处理后的事件
      	int                         revents_; 		
      };
      
      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
    • socket 层

      严格意义上说,并不存在所谓的 socket 层,这一层只是对常用的 socket 函数进行了一层封装,例如封装实现跨平台,方便上层(Channel 层或 Connection 层)使用。很多网络库没有这一层。例如下面就是对常用的 socket 函数的功能做了一层简单的封装:

      namespace sockets
      {		
              SOCKET createOrDie();
              SOCKET createNonblockingOrDie();
      
              void setNonBlockAndCloseOnExec(SOCKET sockfd);
      
              void setReuseAddr(SOCKET sockfd, bool on);
              void setReusePort(SOCKET sockfd, bool on);
      
      		int  connect(SOCKET sockfd, const struct sockaddr_in& addr);
      		void bindOrDie(SOCKET sockfd, const struct sockaddr_in& addr);
      		void listenOrDie(SOCKET sockfd);
      		int  accept(SOCKET sockfd, struct sockaddr_in* addr);
              int32_t read(SOCKET sockfd, void *buf, int32_t count);
      #ifndef WIN32
      		ssize_t readv(SOCKET sockfd, const struct iovec *iov, int iovcnt);
      #endif
      		int32_t write(SOCKET sockfd, const void *buf, int32_t count);
      		void close(SOCKET sockfd);
      		void shutdownWrite(SOCKET sockfd);
      
      		void toIpPort(char* buf, size_t size, const struct sockaddr_in& addr);
      		void toIp(char* buf, size_t size, const struct sockaddr_in& addr);
      		void fromIpPort(const char* ip, uint16_t port, struct sockaddr_in* addr);
      
      		int getSocketError(SOCKET sockfd);
      
      		const struct sockaddr* sockaddr_cast(const struct sockaddr_in* addr);
      		struct sockaddr* sockaddr_cast(struct sockaddr_in* addr);
      		const struct sockaddr_in* sockaddr_in_cast(const struct sockaddr* addr);
      		struct sockaddr_in* sockaddr_in_cast(struct sockaddr* addr);
      
      		struct sockaddr_in getLocalAddr(SOCKET sockfd);
      		struct sockaddr_in getPeerAddr(SOCKET sockfd);
      }
      
      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      31
      32
      33
      34
      35
      36

      在实际实践中,有的服务设计网络通信模块时会将 Connection 对象与 Channel 对象合并成一个对象,这取决于当前业务需要记录的技术上的数据的多少和技术上处理这些数据的复杂性高低。所以在某些服务代码中只看到 Connection 对象或者 Channel 对象请不要觉得奇怪。

      另外,对于服务器端程序,抛开业务本身,在技术层面上,我们需要管理许多的 Connection 对象,一般会使用一个叫 Server 对象(如 TcpServer)来集中管理,这是网络库本身需要处理好的部分。例如一个 TcpServer 对象可能提供的函数接口和状态数据如下:

      class TcpServer
      {
      public:
      	typedef std::function<void(EventLoop*)> ThreadInitCallback;
      	enum Option
      	{
      		kNoReusePort,
      		kReusePort,
      	};
      
      	TcpServer(EventLoop* loop,
      			  const InetAddress& listenAddr,
      			  const std::string& nameArg,
      			  Option option = kReusePort);      
      	~TcpServer();  
      
      	void addConnection(int sockfd, const InetAddress& peerAddr);		
      	void removeConnection(const TcpConnection& conn);
      
      	typedef std::map<string, TcpConnectionPtr> ConnectionMap;
      
      private:
      	int                                             nextConnId_;
      	ConnectionMap                                   connections_;
      };
      
      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25

    对于客户端程序,同样也可以设计出一个 TCPClient 对象管理各个 Connector(连接器对象)。

    对于 Session 对象来说,虽然与 Connection 对象一一对应,但在业务层(网络通信框架之外)需要有专门的类去管理这些 Session 对象的生命周期,我们一般把这个专门的类称之为 SessionManager 或者 SessionFactory。

    # 7.8.2 Session 进一步分层

    不同的服务,其业务可能千差万别,实际开发中我们可能根据业务场景,将 Session 层进一步拆分成多个层,每一层专注于其自己的业务逻辑。例如对于即时聊天服务器,我们可以将 Session 划分为两层:ChatSession、CompressionSession 和 TcpSession,ChatSession 专注于聊天业务本身的处理,CompressSession 负责数据的解压缩,TcpSession 用于将数据加工成网络层需要的格式或者将网络层送上来的数据还原成业务需要的格式(如数据装包和解包)。示意图如下:

    # 7.8.3 连接信息与 EventLoop/Thread 的对应关系

    综合各层对象,一个 socket(fd)只会对应一个channel 对象、一个 Connection 对象以及一个 Session 对象,这一组对象构成了一路连接信息(技术上加业务上的)。结合我们前面介绍了 one thread one loop 思想,每一路连接信息只能属于一个 loop,也就是只会属于某一个线程;但是,反过来,一个 loop 或者一个线程可以同时拥有多个连接信息。这就保证了我们只会在同一个线程里面去是处理特定的 socket 的收发事件。

    上次更新: 2025/04/01, 20:53:14
    7.7 发送/接收缓冲区设计要点
    7.9 后端服务中的定时器设计

    ← 7.7 发送/接收缓冲区设计要点 7.9 后端服务中的定时器设计→

    最近更新
    01
    第二章 关键字static及其不同用法
    03-27
    02
    第一章 auto与类型推导
    03-27
    03
    C++语言面试问题集锦 目录与说明
    03-27
    更多文章>
    Copyright © 2024-2025 沪ICP备2023015129号 张小方 版权所有
    • 跟随系统
    • 浅色模式
    • 深色模式
    • 阅读模式