当前位置:   article > 正文

仿Muduo库实现高并发服务器——socket网络通信模块

仿Muduo库实现高并发服务器——socket网络通信模块

本项目就是基于TCP网络通信搭建的。

TCP:

客户端:socket(),connect().

服务端:socket(),bind(),listen(),accept().

下面代码就是对原生API网络套接字的封装。需要熟悉原生API网络套接字接口。

下面这段代码,没什么好讲的,就不再讲了。下面代码会涉及文件描述符的相关设置,我会专出一篇文章,进行讲解。

  1. #define MAX_LISTEN 1024
  2. class Socket {
  3. private:
  4. int _sockfd;
  5. public:
  6. Socket():_sockfd(-1) {}
  7. Socket(int fd): _sockfd(fd) {}
  8. ~Socket() { Close(); }
  9. int Fd() { return _sockfd; }
  10. //创建套接字
  11. bool Create() {
  12. // int socket(int domain, int type, int protocol)
  13. _sockfd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
  14. if (_sockfd < 0) {
  15. ERR_LOG("CREATE SOCKET FAILED!!");
  16. return false;
  17. }
  18. return true;
  19. }
  20. //绑定地址信息
  21. bool Bind(const std::string &ip, uint16_t port) {
  22. struct sockaddr_in addr;
  23. addr.sin_family = AF_INET;
  24. addr.sin_port = htons(port);
  25. addr.sin_addr.s_addr = inet_addr(ip.c_str());
  26. socklen_t len = sizeof(struct sockaddr_in);
  27. // int bind(int sockfd, struct sockaddr*addr, socklen_t len);
  28. int ret = bind(_sockfd, (struct sockaddr*)&addr, len);
  29. if (ret < 0) {
  30. ERR_LOG("BIND ADDRESS FAILED!");
  31. return false;
  32. }
  33. return true;
  34. }
  35. //开始监听
  36. bool Listen(int backlog = MAX_LISTEN) {
  37. // int listen(int backlog)
  38. int ret = listen(_sockfd, backlog);
  39. if (ret < 0) {
  40. ERR_LOG("SOCKET LISTEN FAILED!");
  41. return false;
  42. }
  43. return true;
  44. }
  45. //向服务器发起连接
  46. bool Connect(const std::string &ip, uint16_t port) {
  47. struct sockaddr_in addr;
  48. addr.sin_family = AF_INET;
  49. addr.sin_port = htons(port);
  50. addr.sin_addr.s_addr = inet_addr(ip.c_str());
  51. socklen_t len = sizeof(struct sockaddr_in);
  52. // int connect(int sockfd, struct sockaddr*addr, socklen_t len);
  53. int ret = connect(_sockfd, (struct sockaddr*)&addr, len);
  54. if (ret < 0) {
  55. ERR_LOG("CONNECT SERVER FAILED!");
  56. return false;
  57. }
  58. return true;
  59. }
  60. //获取新连接
  61. int Accept() {
  62. // int accept(int sockfd, struct sockaddr *addr, socklen_t *len);
  63. int newfd = accept(_sockfd, NULL, NULL);//不需要获取客户端信息 就可以将他们置NULL
  64. if (newfd < 0) {
  65. ERR_LOG("SOCKET ACCEPT FAILED!");
  66. return -1;
  67. }
  68. return newfd;
  69. }
  70. //接收数据
  71. ssize_t Recv(void *buf, size_t len, int flag = 0) {
  72. // ssize_t recv(int sockfd, void *buf, size_t len, int flag);
  73. ssize_t ret = recv(_sockfd, buf, len, flag);
  74. if (ret <= 0) {
  75. //EAGAIN 当前socket的接收缓冲区中没有数据了,在非阻塞的情况下才会有这个错误
  76. //EINTR 表示当前socket的阻塞等待,被信号打断了,
  77. if (errno == EAGAIN || errno == EINTR) {
  78. return 0;//表示这次接收没有接收到数据
  79. }
  80. ERR_LOG("SOCKET RECV FAILED!!");
  81. return -1;
  82. }
  83. return ret; //实际接收的数据长度
  84. }
  85. ssize_t NonBlockRecv(void *buf, size_t len) {
  86. return Recv(buf, len, MSG_DONTWAIT); // MSG_DONTWAIT 表示当前接收为非阻塞。
  87. }
  88. //发送数据
  89. ssize_t Send(const void *buf, size_t len, int flag = 0) {
  90. // ssize_t send(int sockfd, void *data, size_t len, int flag);
  91. ssize_t ret = send(_sockfd, buf, len, flag);
  92. if (ret < 0) {
  93. if (errno == EAGAIN || errno == EINTR) {
  94. return 0;
  95. }
  96. ERR_LOG("SOCKET SEND FAILED!!");
  97. return -1;
  98. }
  99. return ret;//实际发送的数据长度
  100. }
  101. ssize_t NonBlockSend(void *buf, size_t len) {
  102. if (len == 0) return 0;
  103. return Send(buf, len, MSG_DONTWAIT); // MSG_DONTWAIT 表示当前发送为非阻塞。
  104. }
  105. //关闭套接字
  106. void Close() {
  107. if (_sockfd != -1) {
  108. close(_sockfd);
  109. _sockfd = -1;
  110. }
  111. }
  112. //创建一个服务端连接
  113. bool CreateServer(uint16_t port, const std::string &ip = "0.0.0.0", bool block_flag = false) {
  114. //1. 创建套接字,2. 绑定地址,3. 开始监听,4. 设置非阻塞, 5. 启动地址重用
  115. if (Create() == false) return false;
  116. if (block_flag) NonBlock();
  117. if (Bind(ip, port) == false) return false;
  118. if (Listen() == false) return false;
  119. ReuseAddress();
  120. return true;
  121. }
  122. //创建一个客户端连接
  123. bool CreateClient(uint16_t port, const std::string &ip) {
  124. //1. 创建套接字,2.指向连接服务器
  125. if (Create() == false) return false;
  126. if (Connect(ip, port) == false) return false;
  127. return true;
  128. }
  129. //设置套接字选项---开启地址端口重用
  130. void ReuseAddress() {
  131. // int setsockopt(int fd, int leve, int optname, void *val, int vallen)
  132. int val = 1;
  133. setsockopt(_sockfd, SOL_SOCKET, SO_REUSEADDR, (void*)&val, sizeof(int));
  134. val = 1;
  135. setsockopt(_sockfd, SOL_SOCKET, SO_REUSEPORT, (void*)&val, sizeof(int));
  136. }
  137. //设置套接字阻塞属性-- 设置为非阻塞
  138. void NonBlock() {
  139. //int fcntl(int fd, int cmd, ... /* arg */ );
  140. int flag = fcntl(_sockfd, F_GETFL, 0);
  141. fcntl(_sockfd, F_SETFL, flag | O_NONBLOCK);
  142. }
  143. };

本文内容由网友自发贡献,转载请注明出处:【wpsshop博客】
推荐阅读
相关标签
  

闽ICP备14008679号