• 项目:UDP聊天室



    UDP

    UDP(User Datagram Protocol)是一种无连接、不可靠、面向数据报的传输协议。与TCP相比,UDP更加轻量级,不提供像TCP那样的可靠性和流控制机制,但具备较低的通信延迟和较少的开销。

    UDP具有以下几个特点:

    1. 无连接性:UDP在通信之前不需要进行握手或建立连接,可以直接向目标主机发送数据报。这使得UDP的开销较低,适用于实时数据传输或需要快速响应的应用场景。

    2. 不可靠性:UDP不保证数据报的可靠性传输,发送端一旦发送数据包就不会去确认是否到达目标主机。这意味着UDP数据包可能会在传输过程中丢失、重复、乱序,接收方需要自行处理这些问题。

    3. 面向数据报:UDP将应用层交给它的数据封装成单个数据报发送,每个数据报都是独立的。接收方以数据报为单位进行处理,不会像TCP一样存在拆包和粘包的问题。

    4. 快速:由于UDP没有连接的建立和断开过程,且不需要进行可靠性的保证,因此UDP的通信延迟较低。这使得UDP适用于需要实时性和高性能的应用,例如音视频传输、实时游戏等。

    UDP常见的应用场景包括以下几个方面:

    1. 实时数据传输:UDP适合用于实时数据传输,如音视频流、实时视频会议等。由于UDP的低延迟和快速性能,可以保证数据的及时到达,并避免了TCP的可靠性机制带来的可能的延迟。

    2. DNS(Domain Name System):UDP常用于DNS查询,客户端通过UDP向DNS服务器发送域名解析请求,DNS服务器返回响应信息。

    3. 移动应用:UDP适合用于移动应用,如移动终端上的实时定位、实时数据传输等场景。由于UDP不需要进行连接的建立和断开,更加适应移动网络环境的不稳定性。

    需要注意的是,由于UDP不提供可靠性保证,需要在应用层面上考虑数据的可靠性和完整性,例如通过应用层协议、数据校验和重传机制等手段来确保数据的正确传输。

    要求

    利用UDP协议,实现一套聊天室软件。服务器端记录客户端的地址,客户端发送消息后,服务器群发给各个客户端软件

    问题思考:

    l 客户端会不会知道其它客户端地址?
    UDP客户端不会直接互连,所以不会获知其它客户端地址,所有客户端地址存储在服务器端。

    l 有几种消息类型?
    登录:服务器存储新的客户端的地址。把某个客户端登录的消息发给其它客户端。
    聊天:服务器只需要把某个客户端的聊天消息转发给所有其它客户端。 退出:服务器删除退出客户端的地址,并把退出消息发送给其它客户端。

    l 服务器如何存储客户端的地址?
    数据结构使用链表最为简单

    流程图

    服务器

    server.png

    客户端

    client.png

    思路:

    1.首先要实现客户端和服务器端的全双工通信,因此要用到多进程,多线程或IO多路复用,IO多路复用包括select,poll,epoll。其中epoll效率最高,支持百万级别的并发。

    三种IO多路复用特点以及流程如下

    2023-08-29T15:02:22.png

    2.服务器采用有头单向链表,存储每个客户端的IP地址,链表节点结构体如下。

    1. /* 链表结点结构体 */
    2. typedef struct node_t
    3. {
    4. struct sockaddr_in addr; //IP地址结构体
    5. struct node_t *next; //指针域
    6. }node_t;

    3.建立传送消息的结构体,结构体分为三部分,分别是消息类型,昵称和消息正文。服务器端根据判断消息类型进行不同的操作。消息结构体如下。
    / 消息对应的结构体(同一个协议) /

    1. typedef struct msg_t
    2. {
    3. int type; //L登录 M聊天 Q退出
    4. char name[32]; //用户名
    5. char text[128]; //消息正文
    6. } MSG_t;

    4.客户端运行时,首先发送链接类型信息,进行链接。链接后发送信息,根据信息内容判断是否退出。

    实现

    此项目可以使用多进程,多线程,select,poll,epoll实现,此次我会用 多线程和epoll两种方式实现。其余方法相差不大。

    多进程实现

    head.h

    1. #ifndef __HEAD_H__
    2. #define __HEAD_H__
    3. #include
    4. #include
    5. #include
    6. #include
    7. #include
    8. #include
    9. #include
    10. #include
    11. #include
    12. #include
    13. #include
    14. #include
    15. #include
    16. /* 链表结点结构体 */
    17. typedef struct node_t
    18. {
    19. struct sockaddr_in addr; //IP地址结构体
    20. struct node_t *next; //指针域
    21. }node_t;
    22. /* 消息对应的结构体(同一个协议) */
    23. typedef struct msg_t
    24. {
    25. int type; //L登录 M聊天 Q退出
    26. char name[32]; //用户名
    27. char text[128]; //消息正文
    28. } MSG_t;
    29. enum type_t
    30. {
    31. login, //枚举后面要用逗号隔开 ---0
    32. message, // ---1
    33. quit, // ---2
    34. };
    35. #endif

    server.c

    1. #include "head.h"
    2. struct sockaddr_in saddr, caddr;
    3. //创建链表
    4. node_t *CreateList()
    5. {
    6. node_t *p = (node_t *)malloc(sizeof(node_t));
    7. if (NULL == p)
    8. {
    9. perror("CreateList is err");
    10. return NULL;
    11. }
    12. p->next = NULL;
    13. return p;
    14. }
    15. //客户端链接
    16. void server_link(int socked, node_t *p, MSG_t *msg)
    17. {
    18. sprintf(msg->text, "%s连接", msg->name);
    19. printf("%s\n", msg->text);
    20. while (p->next)
    21. {
    22. p = p->next;
    23. sendto(socked, msg, sizeof(*msg), 0, (struct sockaddr *)(&(p->addr)), sizeof(p->addr));
    24. }
    25. node_t *new = (node_t *)malloc(sizeof(node_t));
    26. new->next = NULL;
    27. new->addr=caddr;
    28. p->next = new;
    29. }
    30. //客户端发送正文消息
    31. void server_message(int socked, node_t *p, MSG_t *msg)
    32. {
    33. while (p->next)
    34. {
    35. p = p->next;
    36. if ((memcmp(&(p->addr), &caddr, sizeof(caddr))) == 0)
    37. continue;
    38. sendto(socked, msg, sizeof(*msg), 0, (struct sockaddr *)(&(p->addr)), sizeof(p->addr));
    39. }
    40. }
    41. //客户端退出
    42. void server_quit(int socked, node_t *p, MSG_t *msg)
    43. {
    44. node_t *pdel = NULL;
    45. sprintf(msg->text, "%s退出", msg->name);
    46. printf("%s\n", msg->text);
    47. while (p->next)
    48. {
    49. if ((memcmp(&(p->next->addr), &caddr, sizeof(caddr))) == 0)
    50. {
    51. pdel = p->next;
    52. p->next = pdel->next;
    53. free(pdel);
    54. pdel = NULL;
    55. }
    56. else
    57. {
    58. p = p->next;
    59. sendto(socked, msg, sizeof(*msg), 0, (struct sockaddr *)(&(p->addr)), sizeof(p->addr));
    60. }
    61. }
    62. }
    63. //6.服务器端发送消息,所有客户端接收
    64. void *mypthread(void *socked)
    65. {
    66. MSG_t msg;
    67. msg.type = message;
    68. strcpy(msg.name, "服务器");
    69. while (1)
    70. {
    71. fgets(msg.text, sizeof(msg.text), stdin);
    72. if (msg.text[strlen(msg.text) - 1] == '\n')
    73. msg.text[strlen(msg.text) - 1] = '\0';
    74. sendto(*(int *)(socked), &msg, sizeof(msg), 0, (struct sockaddr *)(&saddr), sizeof(saddr));
    75. }
    76. pthread_exit(NULL);
    77. }
    78. int main(int argc, char const *argv[])
    79. {
    80. //1.建立socket套接字
    81. int socked = socket(AF_INET, SOCK_DGRAM, 0);
    82. if (socked < 0)
    83. {
    84. perror("socket is err");
    85. return -1;
    86. }
    87. //2.bind服务器端口号和ip地址
    88. saddr.sin_family = AF_INET;
    89. saddr.sin_port = htons(atoi(argv[1]));
    90. saddr.sin_addr.s_addr = inet_addr("0.0.0.0");
    91. int len = sizeof(caddr);
    92. if ((bind(socked, (struct sockaddr *)(&saddr), sizeof(saddr))) < 0)
    93. {
    94. perror("bind is err");
    95. return -1;
    96. }
    97. //3.建立多进程,主进程服务器接收,一个服务器发送
    98. pthread_t pid;
    99. pthread_create(&pid, NULL, mypthread, &socked);
    100. pthread_detach(pid);
    101. //4.主线程不断接收客户端发送的消息
    102. node_t *p = CreateList();
    103. MSG_t msg;
    104. while (1)
    105. {
    106. if ((recvfrom(socked, &msg, sizeof(msg), 0, (struct sockaddr *)(&caddr), &len)) < 0)
    107. {
    108. perror("recvfrom is err");
    109. return -1;
    110. }
    111. //5.根据消息类型进行分类讨论
    112. switch (msg.type)
    113. {
    114. case login:
    115. server_link(socked, p, &msg);
    116. break;
    117. case message:
    118. server_message(socked, p, &msg);
    119. break;
    120. case quit:
    121. server_quit(socked, p, &msg);
    122. break;
    123. default:
    124. break;
    125. }
    126. }
    127. close(socked);
    128. return 0;
    129. }

    client.c

    1. #include "head.h"
    2. struct sockaddr_in saddr, caddr;
    3. int len;
    4. //5.子线程接收客户端数据
    5. void *mypthread(void *socked)
    6. {
    7. MSG_t msg;
    8. while (1)
    9. {
    10. int temp = recvfrom(*(int *)(socked), &msg, sizeof(msg), 0, (struct sockaddr *)(&caddr), &len);
    11. if (temp < 0)
    12. {
    13. perror("recvfrom err");
    14. exit(-1);
    15. }
    16. else
    17. {
    18. printf("\t\t\t\t%s(%s)\n",msg.text,msg.name);
    19. }
    20. }
    21. }
    22. int main(int argc, char const *argv[])
    23. {
    24. //1.建立socket套接字
    25. int socked = socket(AF_INET, SOCK_DGRAM, 0);
    26. if (socked < 0)
    27. {
    28. perror("socket is err");
    29. return -1;
    30. }
    31. //2.绑定服务器ip和端口号
    32. saddr.sin_port = htons(atoi(argv[2]));
    33. saddr.sin_addr.s_addr = inet_addr(argv[1]);
    34. saddr.sin_family = AF_INET;
    35. len = sizeof(caddr);
    36. //3.连接服务器
    37. MSG_t msg;
    38. msg.type = login;
    39. printf("请输入名称:\n");
    40. fgets(msg.name, sizeof(msg.name), stdin);
    41. if (msg.name[strlen(msg.name) - 1] == '\n')
    42. msg.name[strlen(msg.name) - 1] = '\0';
    43. sendto(socked, &msg, sizeof(msg), 0, (struct sockaddr *)(&saddr), sizeof(saddr));
    44. //4.创建子进程接收服务器数据,主进程发送数据
    45. pthread_t pid;
    46. pthread_create(&pid, NULL, mypthread, &socked);
    47. pthread_detach(pid);
    48. //5.主线程发送客户端数据
    49. while (1)
    50. {
    51. fgets(msg.text, sizeof(msg.text), stdin);
    52. if (msg.text[strlen(msg.text) - 1] == '\n')
    53. msg.text[strlen(msg.text) - 1] = '\0';
    54. if (strcmp(msg.text, "quit") == 0)
    55. msg.type = quit;
    56. else
    57. msg.type = message;
    58. sendto(socked, &msg, sizeof(msg), 0, (struct sockaddr *)(&saddr), sizeof(saddr));
    59. }
    60. close(socked);
    61. return 0;
    62. }

    Makefile

    1. all:
    2. gcc client.c -o client -lpthread
    3. gcc server.c -o server -lpthread
    4. .PHONY:clean
    5. clean:
    6. rm *.o

    epoll实现

    head.h

    1. #ifndef __HEAD_H__
    2. #define __HEAD_H__
    3. #include
    4. #include
    5. #include
    6. #include
    7. #include
    8. #include
    9. #include
    10. #include
    11. #include
    12. #include
    13. #include
    14. #include
    15. #include
    16. #include
    17. /* 链表结点结构体 */
    18. typedef struct node_t
    19. {
    20. struct sockaddr_in addr; //IP地址结构体
    21. struct node_t *next; //指针域
    22. }node_t;
    23. /* 消息对应的结构体(同一个协议) */
    24. typedef struct msg_t
    25. {
    26. int type; //L登录 M聊天 Q退出
    27. char name[32]; //用户名
    28. char text[128]; //消息正文
    29. } MSG_t;
    30. enum type_t
    31. {
    32. login, //枚举后面要用逗号隔开 ---0
    33. message, // ---1
    34. quit, // ---2
    35. };
    36. #endif

    server.c

    1. #include "head.h"
    2. struct sockaddr_in saddr, caddr;
    3. //创建链表
    4. node_t *CreateList()
    5. {
    6. node_t *p = (node_t *)malloc(sizeof(node_t));
    7. if (NULL == p)
    8. {
    9. perror("CreateList is err");
    10. return NULL;
    11. }
    12. p->next = NULL;
    13. return p;
    14. }
    15. //客户端链接
    16. void server_link(int socked, node_t *p, MSG_t *msg)
    17. {
    18. sprintf(msg->text, "%s连接", msg->name);
    19. printf("%s\n", msg->text);
    20. while (p->next)
    21. {
    22. p = p->next;
    23. sendto(socked, msg, sizeof(*msg), 0, (struct sockaddr *)(&(p->addr)), sizeof(p->addr));
    24. }
    25. node_t *new = (node_t *)malloc(sizeof(node_t));
    26. new->next = NULL;
    27. new->addr = caddr;
    28. p->next = new;
    29. }
    30. //客户端发送正文消息
    31. void server_message(int socked, node_t *p, MSG_t *msg)
    32. {
    33. while (p->next)
    34. {
    35. p = p->next;
    36. if ((memcmp(&(p->addr), &caddr, sizeof(caddr))) == 0)
    37. continue;
    38. sendto(socked, msg, sizeof(*msg), 0, (struct sockaddr *)(&(p->addr)), sizeof(p->addr));
    39. }
    40. }
    41. //客户端退出
    42. void server_quit(int socked, node_t *p, MSG_t *msg)
    43. {
    44. node_t *pdel = NULL;
    45. sprintf(msg->text, "%s退出", msg->name);
    46. printf("%s\n", msg->text);
    47. while (p->next)
    48. {
    49. if ((memcmp(&(p->next->addr), &caddr, sizeof(caddr))) == 0)
    50. {
    51. pdel = p->next;
    52. p->next = pdel->next;
    53. free(pdel);
    54. pdel = NULL;
    55. }
    56. else
    57. {
    58. p = p->next;
    59. sendto(socked, msg, sizeof(*msg), 0, (struct sockaddr *)(&(p->addr)), sizeof(p->addr));
    60. }
    61. }
    62. }
    63. int main(int argc, char const *argv[])
    64. {
    65. //1.建立socket套接字
    66. int socked = socket(AF_INET, SOCK_DGRAM, 0);
    67. if (socked < 0)
    68. {
    69. perror("socket is err");
    70. return -1;
    71. }
    72. //2.bind服务器端口号和ip地址
    73. saddr.sin_family = AF_INET;
    74. saddr.sin_port = htons(atoi(argv[1]));
    75. saddr.sin_addr.s_addr = inet_addr("0.0.0.0");
    76. int len = sizeof(caddr);
    77. if ((bind(socked, (struct sockaddr *)(&saddr), sizeof(saddr))) < 0)
    78. {
    79. perror("bind is err");
    80. return -1;
    81. }
    82. //3.主线程不断接收客户端发送的消息
    83. node_t *p = CreateList();
    84. MSG_t msg;
    85. //1).创建红黑树以及链表
    86. //树的跟节点/树的句柄
    87. int epfd = epoll_create(1);
    88. //2).上树
    89. struct epoll_event event;
    90. struct epoll_event events[32];
    91. event.events = EPOLLET | EPOLLIN;
    92. event.data.fd = 0;
    93. epoll_ctl(epfd, EPOLL_CTL_ADD, 0, &event);
    94. event.data.fd = socked;
    95. epoll_ctl(epfd, EPOLL_CTL_ADD, socked, &event);
    96. while (1)
    97. {
    98. //3).阻塞等待文件描述符产生事件
    99. int ret = epoll_wait(epfd, events, 32, -1);
    100. if (ret < 0)
    101. {
    102. perror("epoll err");
    103. return -1;
    104. }
    105. //4).根据文件描述符号,进行处理
    106. for (int i = 0; i < ret; ++i)
    107. {
    108. if (events[i].data.fd == 0)
    109. {
    110. msg.type = message;
    111. strcpy(msg.name, "客户端");
    112. fgets(msg.text, sizeof(msg.text), stdin);
    113. if (msg.text[strlen(msg.text) - 1] == '\n')
    114. msg.text[strlen(msg.text) - 1] = '\0';
    115. sendto(*(int *)(socked), &msg, sizeof(msg), 0, (struct sockaddr *)(&saddr), sizeof(saddr));
    116. }
    117. else if (events[i].data.fd == socked)
    118. {
    119. if ((recvfrom(socked, &msg, sizeof(msg), 0, (struct sockaddr *)(&caddr), &len)) < 0)
    120. {
    121. perror("recvfrom is err");
    122. return -1;
    123. }
    124. //4.根据消息类型进行分类讨论
    125. switch (msg.type)
    126. {
    127. case login:
    128. server_link(socked, p, &msg);
    129. break;
    130. case message:
    131. server_message(socked, p, &msg);
    132. break;
    133. case quit:
    134. server_quit(socked, p, &msg);
    135. break;
    136. default:
    137. break;
    138. }
    139. }
    140. }
    141. }
    142. close(socked);
    143. return 0;
    144. }

    client.c

    1. #include "head.h"
    2. struct sockaddr_in saddr, caddr;
    3. int len;
    4. //5.子线程接收客户端数据
    5. void *mypthread(void *socked)
    6. {
    7. MSG_t msg;
    8. while (1)
    9. {
    10. int temp = recvfrom(*(int *)(socked), &msg, sizeof(msg), 0, (struct sockaddr *)(&caddr), &len);
    11. if (temp < 0)
    12. {
    13. perror("recvfrom err");
    14. exit(-1);
    15. }
    16. else
    17. {
    18. printf("\t\t\t\t%s(%s)\n", msg.text, msg.name);
    19. }
    20. }
    21. }
    22. int main(int argc, char const *argv[])
    23. {
    24. //1.建立socket套接字
    25. int socked = socket(AF_INET, SOCK_DGRAM, 0);
    26. if (socked < 0)
    27. {
    28. perror("socket is err");
    29. return -1;
    30. }
    31. //2.绑定服务器ip和端口号
    32. saddr.sin_port = htons(atoi(argv[2]));
    33. saddr.sin_addr.s_addr = inet_addr(argv[1]);
    34. saddr.sin_family = AF_INET;
    35. len = sizeof(caddr);
    36. //3.连接服务器
    37. MSG_t msg;
    38. msg.type = login;
    39. printf("请输入名称:\n");
    40. fgets(msg.name, sizeof(msg.name), stdin);
    41. if (msg.name[strlen(msg.name) - 1] == '\n')
    42. msg.name[strlen(msg.name) - 1] = '\0';
    43. sendto(socked, &msg, sizeof(msg), 0, (struct sockaddr *)(&saddr), sizeof(saddr));
    44. //4.创建子进程接收服务器数据,主进程发送数据
    45. pthread_t pid;
    46. pthread_create(&pid, NULL, mypthread, &socked);
    47. pthread_detach(pid);
    48. //5.主线程发送客户端数据
    49. while (1)
    50. {
    51. fgets(msg.text, sizeof(msg.text), stdin);
    52. if (msg.text[strlen(msg.text) - 1] == '\n')
    53. msg.text[strlen(msg.text) - 1] = '\0';
    54. if (strcmp(msg.text, "quit") == 0)
    55. {
    56. msg.type = quit;
    57. sendto(socked, &msg, sizeof(msg), 0, (struct sockaddr *)(&saddr), sizeof(saddr));
    58. break;
    59. }
    60. else
    61. msg.type = message;
    62. sendto(socked, &msg, sizeof(msg), 0, (struct sockaddr *)(&saddr), sizeof(saddr));
    63. }
    64. return 0;
    65. }

    Makefile

    1. all:
    2. gcc client.c -o client -lpthread
    3. gcc server.c -o server -lpthread
    4. .PHONY:clean
    5. clean: rm *.o

  • 相关阅读:
    webm格式转换成mp4?
    有刷电机的速度pid-位置pid算法
    SpringBoot统一功能处理
    Vue 父子组件传值有哪些方式
    【C++11】lambda匿名函数和包装器
    EXCEL单元格中提取部分文本
    EN 14342木地板产品—CE认证
    预告|易天光通信将亮相第24届CIOE光博会
    微信小程序游戏开发│石头剪刀布游戏(附源码)
    java字符串类的使用
  • 原文地址:https://blog.csdn.net/m0_73731708/article/details/132893004