• 网络通信框架的完善


    问题

    如何拓展之前开发的通信框架,使其支持 UDP 通信,进而成为一个完善的网络通信框架?

    UDP 通信拓展

    UDP 通信实体概要设计 

    每个 UDP Point 地位对等,可通过 IP 地址和 port 号进行通信

    UDP Point 数据收发单位为:Message 或 Byte

    在接口设计上,与 TcpClient 保持一致 (框架接口一致性)

    目标:封装原生 socket 细节,关注 UDP 通信逻辑

    UDP 通信实体接口设计

    关键代码实现 - 初始化

     

    关键代码实现 - 消息接收

    由于 UDP 是以数据报的方式进行通信 (非数据流方式,报文间有明显边界)

    因此,不能直接通过 MParser_ReadFd(...) 解析出消息

    必须先将报文完整接收到内存中,再进行从内存中解析出消息

    即通过 MParser_ReadMem(...) 间接完成消息解析

    可以通过 MSG_PEEK 来获取缓冲区中的报文长度

     

    关键代码实现 - 消息发送

    关键代码实现 - 通信选项设置

    由于 UdpPoint 封装了 socket 文件描述符的细节,所以需要对外提供改变 socket 属性的接口函数,以便用户使用广播和多播以及其他的功能。

    UDP 通信端设计与实现

    udp_point.h

    1. #ifndef UDP_POINT_H
    2. #define UDP_POINT_H
    3. #include "message.h"
    4. typedef void UdpPoint;
    5. UdpPoint* UdpPoint_New(int port);
    6. UdpPoint* UdpPoint_From(int fd);
    7. int UdpPoint_SendMsg(UdpPoint* point, Message* msg, const char* remote, int port);
    8. int UdpPoint_SendRaw(UdpPoint* point, char* buf, int length, const char* remote, int port);
    9. Message* UdpPoint_RecvMsg(UdpPoint* point, char* remote, int* port);
    10. int UdpPoint_RecvRaw(UdpPoint* point, char* buf, int length, char* remote, int* port);
    11. void UdpPoint_Del(UdpPoint* point);
    12. int UdpPoint_Available(UdpPoint* point);
    13. void UdpPoint_SetData(UdpPoint* point, void* data);
    14. void* UdpPoint_GetData(UdpPoint* point);
    15. int UdpPoint_SetOpt(UdpPoint* point, int level, int optname, const void* optval, unsigned int optlen);
    16. int UdpPoint_GetOpt(UdpPoint* point, int level, int optname, void* optval, unsigned int* optlen);
    17. #endif

    udp_point.c

    1. #include "udp_point.h"
    2. #include "msg_parser.h"
    3. #include
    4. #include
    5. #include
    6. #include
    7. #include
    8. #include
    9. typedef struct udp_point
    10. {
    11. int fd;
    12. MParser* parser;
    13. void* data;
    14. } Point;
    15. char g_temp[4096] = {0};
    16. static void ParseAddr(struct sockaddr_in addr, char* remote, int* port)
    17. {
    18. if(remote)
    19. {
    20. strcpy(remote, inet_ntoa(addr.sin_addr));
    21. }
    22. if(port)
    23. {
    24. *port = ntohs(addr.sin_port);
    25. }
    26. }
    27. UdpPoint* UdpPoint_New(int port)
    28. {
    29. Point* ret = NULL;
    30. int fd = -1;
    31. struct sockaddr_in addr = {0};
    32. int ok = 1;
    33. addr.sin_family = AF_INET;
    34. addr.sin_addr.s_addr = htonl(INADDR_ANY);
    35. addr.sin_port = htons(port);
    36. ok = ok && ((fd = socket(PF_INET, SOCK_DGRAM, 0))!= -1);
    37. ok = ok && (bind(fd, (struct sockaddr*)&addr, sizeof(addr)) != -1);
    38. ok = ok && (ret = UdpPoint_From(fd));
    39. if(!ok)
    40. {
    41. if(fd != -1)
    42. {
    43. close(fd);
    44. }
    45. }
    46. return ret;
    47. }
    48. UdpPoint* UdpPoint_From(int fd)
    49. {
    50. Point* ret = (Point*)malloc(sizeof(Point));
    51. if(ret)
    52. {
    53. ret->fd = fd;
    54. ret->parser = MParser_New();
    55. ret->data = NULL;
    56. }
    57. return ((ret && ret->parser) ? ret : (free(ret), ret = NULL));
    58. }
    59. int UdpPoint_SendMsg(UdpPoint* point, Message* msg, const char* remote, int port)
    60. {
    61. int ret = -1;
    62. int length = 0;
    63. char* buf = (char*)msg;
    64. if(point && msg && remote)
    65. {
    66. length = Message_Size(msg);
    67. Message_H2N(msg);
    68. ret = UdpPoint_SendRaw(point, buf, length, remote, port);
    69. Message_N2H(msg);
    70. }
    71. return ret;
    72. }
    73. int UdpPoint_SendRaw(UdpPoint* point, char* buf, int length, const char* remote, int port)
    74. {
    75. int ret = -1;
    76. Point* c = (Point*)point;
    77. if(c && buf && remote)
    78. {
    79. struct sockaddr_in addr = {0};
    80. addr.sin_family = AF_INET;
    81. addr.sin_addr.s_addr = inet_addr(remote);
    82. addr.sin_port = htons(port);
    83. ret = sendto(c->fd, buf, length, 0, (struct sockaddr*)&addr, sizeof(addr));
    84. }
    85. return ret;
    86. }
    87. Message* UdpPoint_RecvMsg(UdpPoint* point, char* remote, int* port)
    88. {
    89. Message* ret = NULL;
    90. Point* c = (Point*)point;
    91. if(c)
    92. {
    93. int length = 0;
    94. length = recvfrom(c->fd, g_temp, sizeof(g_temp), MSG_PEEK , NULL, NULL);
    95. if(length > 0)
    96. {
    97. char* buf = (char*)malloc(length);
    98. if(buf)
    99. {
    100. length = UdpPoint_RecvRaw(point, buf, length, remote, port);
    101. ret = MParser_ReadMem(c->parser, buf, length);
    102. free(buf);
    103. }
    104. }
    105. }
    106. return ret;
    107. }
    108. int UdpPoint_RecvRaw(UdpPoint* point, char* buf, int length, char* remote, int* port)
    109. {
    110. int ret = -1;
    111. Point* c = (Point*)point;
    112. if(c && buf)
    113. {
    114. struct sockaddr_in addr = {0};
    115. socklen_t addrlen = sizeof(addr);
    116. ret = recvfrom(c->fd, buf, length, 0, (struct sockaddr*)&addr, &addrlen);
    117. if(ret != -1)
    118. {
    119. ParseAddr(addr, remote, port);
    120. }
    121. }
    122. return ret;
    123. }
    124. void UdpPoint_Del(UdpPoint* point)
    125. {
    126. Point* c = (Point*)point;
    127. if(c)
    128. {
    129. close(c->fd);
    130. MParser_Del(c->parser);
    131. free(c);
    132. }
    133. }
    134. int UdpPoint_Available(UdpPoint* point)
    135. {
    136. int ret = -1;
    137. Point* c = (Point*)point;
    138. if(c)
    139. {
    140. ret = recvfrom(c->fd, g_temp, sizeof(g_temp), MSG_PEEK | MSG_DONTWAIT, NULL, NULL);
    141. }
    142. return ret;
    143. }
    144. void UdpPoint_SetData(UdpPoint* point, void* data)
    145. {
    146. Point* c = (Point*)point;
    147. if(c)
    148. {
    149. c->data = data;
    150. }
    151. }
    152. void* UdpPoint_GetData(UdpPoint* point)
    153. {
    154. Point* c = (Point*)point;
    155. return (c ? c->data : NULL);
    156. }
    157. int UdpPoint_SetOpt(UdpPoint* point, int level, int optname, const void* optval, unsigned int optlen)
    158. {
    159. int ret = -1;
    160. Point* c = (Point*)point;
    161. if(c)
    162. {
    163. ret = setsockopt(c->fd, level, optname, optval, optlen);
    164. }
    165. return ret;
    166. }
    167. int UdpPoint_GetOpt(UdpPoint* point, int level, int optname, void* optval, unsigned int* optlen)
    168. {
    169. int ret = -1;
    170. Point* c = (Point*)point;
    171. if(c)
    172. {
    173. ret = getsockopt(c->fd, level, optname, optval, optlen);
    174. }
    175. return ret;
    176. }

    udp_test.c

    1. #include "udp_point.h"
    2. #include
    3. #include
    4. #include
    5. char buf[4096] = {0};
    6. int main(int argc, char* argv[])
    7. {
    8. UdpPoint* point = UdpPoint_New(6666);
    9. Message* msg = NULL;
    10. int i = 0;
    11. char remote[16] = {0};
    12. int port = 0;
    13. printf("point = %p\n", point);
    14. sleep(5);
    15. i = UdpPoint_Available(point);
    16. printf("i = %d\n", i);
    17. // i = UdpPoint_RecvRaw(point, buf, sizeof(buf), remote, &port);
    18. // printf("buf = %s\n", buf);
    19. // printf("remote ip = %s, port = %d\n", remote, port);
    20. // UdpPoint_SendRaw(point, buf, i, remote, port);
    21. msg = UdpPoint_RecvMsg(point, remote, &port);
    22. if(msg != NULL)
    23. {
    24. printf("remote ip = %s, port = %d\n", remote, port);
    25. printf("type = %d\n", msg->type);
    26. printf("cmd = %d\n", msg->cmd);
    27. printf("index = %d\n", msg->index);
    28. printf("total = %d\n", msg->total);
    29. for(int j = 0; j < msg->length; j++)
    30. {
    31. printf("%02X ", msg->payload[j]);
    32. }
    33. printf("\n");
    34. UdpPoint_SendMsg(point, msg, remote, port);
    35. free(msg);
    36. }
    37. UdpPoint_Del(point);
    38. return 0;
    39. }

    每个 UDP 数据报都有一个长度,接收端必须以该长度为最小单元将其所有内容一次性读出,否则数据将被截断。因为 MParser_ReadFd(...) 会先读取缓冲区的12个字节作为头部信息,然后再去读取头部中所记录的字节长度,会读取两次,所以在第一次读取到12个字节数据后,这个数据报的剩余字节数据就读取不到了。所以采用先把数据报的内容读取到我们定义的缓冲区中,然后再去通过我们定义的缓冲区去解析它,以间接的方式来获取 Message。

    测试裸数据的测试结果如下图所示:

    测试 Message 的测试结果如下图所示:

     

  • 相关阅读:
    前端项目--尚医通学习分享
    【面试经典150 | 区间】插入区间
    为什么 eBPF 如此受欢迎?
    利用ansbile部署lamp并部署Discuz(非分布式)
    Python 教程之从头开始构建个人助理,如何在 Python 中构建概念验证个人助理:意图分类、语音到文本和文本到语音(教程含源码)
    机器学习之随机森林
    在get对象属性时发现没有get对应的方法
    ThreadPoolExecutor 基础入门
    python通过socket 搭建极简web服务器
    HCIA笔记-1 网络基础
  • 原文地址:https://blog.csdn.net/qq_52484093/article/details/126335824