当前位置:   article > 正文

基于socket编程实现TCP和UDP的通信流程_socket通信和tcp通信

socket通信和tcp通信

socket编程的常用函数,可以参考以下这篇博客

socket编程-----常用socket编程函数icon-default.png?t=N7T8https://blog.csdn.net/ZZZCY2003/article/details/138071210

关于TCP的三次挥手、四次挥手过程和UDP的报文分析可以参考以下两篇博客

计算机网络--运输层icon-default.png?t=N7T8https://blog.csdn.net/ZZZCY2003/article/details/138007052

TCP协议数据传输过程及报文分析icon-default.png?t=N7T8https://blog.csdn.net/ZZZCY2003/article/details/138035752


目录

一、socket

1.1、socket概述

1.2、基本概念

1.3、关键步骤

1.4、在Linux系统上执行socket过程

二、基于socket编程的TCP通信过程

2.1、基于socket编程的TCP通信流程

2.1.1、服务端(service.c)

2.1.1.1、创建socket

2.1.1.2、绑定socket到端口

2.1.1.3、开始监听连接

2.1.1.4、接受客户端连接

2.1.1.5、读取客户端发送的数据

2.1.1.6、发送响应给客户端

2.1.1.7、关闭套接字

2.1.2、客户端(client.c) 

2.1.2.1、创建socket

2.1.2.2、设置服务器地址结构并来连接服务器

2.1.2.3、发送数据给服务器

2.1.2.4、读取服务器的响应

2.1.2.5、关闭套接字

2.2、基于socket编程的TCP通信流程图

2.3、基于socket编程的TCP示例

2.3.1、服务端(service.c)

2.3.2、客户端(client.c)

三、基于socket编程的UDP通信过程

3.1、基于socket编程的UDP通信流程

3.1.1、服务端

3.1.1.1、创建socket

3.1.1.2、设置服务器地址结构

3.1.1.3、绑定socket

3.1.1.4、打印监听信息

3.1.1.5、接收数据

3.1.1.6、关闭socket

3.1.2、客户端

3.1.2.1、创建socket

3.1.2.2、设置服务器地址结构

3.1.2.3、发送数据

3.1.2.4、打印发送信息

3.1.2.5、关闭socket

3.2、基于socket编程的UDP通信流程图

3.3、基于socket编程的UDP通信示例

3.3.1、服务端(service.c)

3.3.2、客户端(client.c)


一、socket

1.1、socket概述

        Socket编程是计算机网络通信中的一种重要技术,它允许不同的计算机程序通过网络进行通信和数据交换。在Socket编程中,一个程序(通常称为客户端)通过创建Socket对象并与另一个程序(通常称为服务器)建立连接,从而进行通信。

1.2、基本概念

  1. IP地址和端口号:IP地址用于标识网络上的设备,而端口号用于标识设备上的特定服务或应用程序。Socket通过组合IP地址和端口号来唯一标识一个网络上的通信端点。

  2. 客户端与服务器:在Socket编程中,通常有一个服务器程序等待客户端的连接请求。一旦连接建立,客户端和服务器就可以互相发送和接收数据。

  3. 流式套接字(SOCK_STREAM):通常用于TCP协议,提供可靠的、面向连接的通信服务。数据按照字节流的形式发送和接收。

  4. 数据报套接字(SOCK_DGRAM):通常用于UDP协议,提供无连接的通信服务。数据以数据报的形式发送和接收,不保证数据的顺序和可靠性。

1.3、关键步骤

  1. 创建Socket:使用socket函数创建一个新的Socket对象。

  2. 绑定(Bind):对于服务器来说,需要将Socket绑定到一个特定的IP地址和端口号上,以便客户端能够连接。

  3. 监听(Listen):服务器Socket使用listen函数开始监听来自客户端的连接请求。

  4. 接受连接(Accept):服务器使用accept函数接受客户端的连接请求,并返回一个新的Socket对象用于与客户端通信。

  5. 连接(Connect):对于客户端来说,使用connect函数向服务器发起连接请求。

  6. 发送和接收数据:使用send或sendto函数发送数据,使用recv或recvfrom函数接收数据。

  7. 关闭Socket:使用close函数关闭Socket连接,释放资源。

1.4、在Linux系统上执行socket过程

假设已经写好服务端代码(service.c)和客户端代码(client.c)(注意:示例用的是C语言代码,且代码已附在文章最下面)

1、开启两个终端,并对服务端代码和客户端代码进行编译。在两个终端中分别执行以下各一条代码,作为服务端和客户端。

  1. gcc service.c -o service
  2. gcc client.c -o client

2、运行服务端,使服务端处于等待连接状态。等待客户端的请求连接。

./service

3、在另一个终端运行客户端,向服务端发出请求连接的请求

./client

4、最后,执行完成!

二、基于socket编程的TCP通信过程

2.1、基于socket编程的TCP通信流程

2.1.1、服务端(service.c)

2.1.1.1、创建socket
  1. //创建socket文件描述符
  2. if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) == 0) {
  3. perror("socket failed");
  4. exit(EXIT_FAILURE);
  5. }
2.1.1.2、绑定socket到端口
  1. address.sin_family = AF_INET;
  2. address.sin_addr.s_addr = inet_addr("127.0.0.1");
  3. address.sin_port = htons(PORT);
  4. //绑定socket到端口
  5. if (bind(server_fd, (struct sockaddr *)&address, sizeof(address)) < 0) {
  6. perror("bind failed");
  7. exit(EXIT_FAILURE);
  8. }
2.1.1.3、开始监听连接
  1. //开始监听连接
  2. if (listen(server_fd, 3) < 0) {
  3. perror("listen");
  4. exit(EXIT_FAILURE);
  5. }
2.1.1.4、接受客户端连接
  1. //接受客户端连接
  2. if ((new_socket = accept(server_fd, (struct sockaddr *)&address, (socklen_t*)&addrlen)) < 0) {
  3. perror("accept");
  4. exit(EXIT_FAILURE);
  5. }
2.1.1.5、读取客户端发送的数据
  1. //读取客户端发送的数据
  2. int valread = read(new_socket, buffer, BUFFER_SIZE);
  3. printf("%s\n", buffer);
2.1.1.6、发送响应给客户端
  1. //发送响应给客户端
  2. send(new_socket, hello, strlen(hello), 0);
  3. printf("Hello message sent\n");
2.1.1.7、关闭套接字
  1. //关闭套接字
  2. close(server_fd);
  3. close(new_socket);

2.1.2、客户端(client.c) 

2.1.2.1、创建socket
  1. //创建socket文件描述符
  2. if ((sock = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
  3. perror("Socket creation error");
  4. exit(EXIT_FAILURE);
  5. }
2.1.2.2、设置服务器地址结构并来连接服务器
  1. //设置服务器地址结构
  2. memset(&serv_addr, 0, sizeof(serv_addr));
  3. serv_addr.sin_family = AF_INET;
  4. serv_addr.sin_port = htons(PORT);
  5. //连接到服务器
  6. if (connect(sock, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) < 0) {
  7. perror("Connection failed");
  8. exit(EXIT_FAILURE);
  9. }
  10. printf("Connected to server\n");
2.1.2.3、发送数据给服务器
  1. //发送数据给服务器
  2. if (send(sock, hello, strlen(hello), 0) < 0) {
  3. perror("Send failed");
  4. exit(EXIT_FAILURE);
  5. }
  6. printf("Hello message sent\n");
2.1.2.4、读取服务器的响应
  1. //读取服务器的响应
  2. while ((valread = read(sock, buffer, BUFFER_SIZE - 1)) > 0) {
  3. buffer[valread] = '\0';
  4. printf("Received from server: %s\n", buffer);
  5. }
  6. if (valread < 0) {
  7. perror("Read failed");
  8. }
2.1.2.5、关闭套接字
  1. //关闭socket
  2. close(sock);

2.2、基于socket编程的TCP通信流程图

2.3、基于socket编程的TCP示例

2.3.1、服务端(service.c)

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <unistd.h>
  5. #include <sys/types.h>
  6. #include <sys/socket.h>
  7. #include <arpa/inet.h>
  8. #include <netinet/in.h>
  9. #define PORT 8080
  10. #define BUFFER_SIZE 1024
  11. int main() {
  12. int server_fd, new_socket;
  13. struct sockaddr_in address;
  14. int opt = 1;
  15. int addrlen = sizeof(address);
  16. char buffer[BUFFER_SIZE] = {0};
  17. const char *hello = "Hello from server";
  18. //创建socket文件描述符
  19. if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) == 0) {
  20. perror("socket failed");
  21. exit(EXIT_FAILURE);
  22. }
  23. //设置socket选项,允许地址重用
  24. if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR | SO_REUSEPORT, &opt, sizeof(opt))) {
  25. perror("setsockopt");
  26. exit(EXIT_FAILURE);
  27. }
  28. address.sin_family = AF_INET;
  29. address.sin_addr.s_addr = inet_addr("127.0.0.1");
  30. address.sin_port = htons(PORT);
  31. //绑定socket到端口
  32. if (bind(server_fd, (struct sockaddr *)&address, sizeof(address)) < 0) {
  33. perror("bind failed");
  34. exit(EXIT_FAILURE);
  35. }
  36. //开始监听连接
  37. if (listen(server_fd, 3) < 0) {
  38. perror("listen");
  39. exit(EXIT_FAILURE);
  40. }
  41. //接受客户端连接
  42. if ((new_socket = accept(server_fd, (struct sockaddr *)&address, (socklen_t*)&addrlen)) < 0) {
  43. perror("accept");
  44. exit(EXIT_FAILURE);
  45. }
  46. //读取客户端发送的数据
  47. int valread = read(new_socket, buffer, BUFFER_SIZE);
  48. printf("%s\n", buffer);
  49. //发送响应给客户端
  50. send(new_socket, hello, strlen(hello), 0);
  51. printf("Hello message sent\n");
  52. //关闭套接字
  53. close(server_fd);
  54. close(new_socket);
  55. return 0;
  56. }

2.3.2、客户端(client.c)

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <unistd.h>
  5. #include <sys/types.h>
  6. #include <sys/socket.h>
  7. #include <netinet/in.h>
  8. #include <netdb.h>
  9. #define PORT 8080
  10. #define BUFFER_SIZE 1024
  11. int main() {
  12. int sock = 0;
  13. struct sockaddr_in serv_addr;
  14. char *hello = "Hello from client";
  15. char buffer[BUFFER_SIZE] = {0};
  16. int valread;
  17. //创建socket文件描述符
  18. if ((sock = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
  19. perror("Socket creation error");
  20. exit(EXIT_FAILURE);
  21. }
  22. //设置服务器地址结构
  23. memset(&serv_addr, 0, sizeof(serv_addr));
  24. serv_addr.sin_family = AF_INET;
  25. serv_addr.sin_port = htons(PORT);
  26. //连接到服务器
  27. if (connect(sock, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) < 0) {
  28. perror("Connection failed");
  29. exit(EXIT_FAILURE);
  30. }
  31. printf("Connected to server\n");
  32. //发送数据给服务器
  33. if (send(sock, hello, strlen(hello), 0) < 0) {
  34. perror("Send failed");
  35. exit(EXIT_FAILURE);
  36. }
  37. printf("Hello message sent\n");
  38. //读取服务器的响应
  39. while ((valread = read(sock, buffer, BUFFER_SIZE - 1)) > 0) {
  40. buffer[valread] = '\0';
  41. printf("Received from server: %s\n", buffer);
  42. }
  43. if (valread < 0) {
  44. perror("Read failed");
  45. }
  46. //关闭socket
  47. close(sock);
  48. return 0;
  49. }

三、基于socket编程的UDP通信过程

3.1、基于socket编程的UDP通信流程

3.1.1、服务端

3.1.1.1、创建socket
  1. //创建socket
  2. if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
  3. perror("socket creation failed");
  4. exit(EXIT_FAILURE);
  5. }
3.1.1.2、设置服务器地址结构
  1. //设置服务器地址结构
  2. memset(&server_addr, 0, sizeof(server_addr));
  3. server_addr.sin_family = AF_INET;
  4. server_addr.sin_addr.s_addr = inet_addr("127.0.0.1");//具体的ip地址
  5. server_addr.sin_port = htons(SERVER_PORT);
3.1.1.3、绑定socket
  1. //绑定socket
  2. if (bind(sockfd, (const struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) {
  3. perror("bind failed");
  4. exit(EXIT_FAILURE);
  5. }
3.1.1.4、打印监听信息
printf("Server is listening on port %d\n", SERVER_PORT);
3.1.1.5、接收数据
  1. //接收数据
  2. n = recvfrom(sockfd, (char *)buffer, BUFFER_SIZE, MSG_WAITALL, (struct sockaddr *)&client_addr, &len);
  3. buffer[n] = '\0';
  4. printf("Received from client: %s\n", buffer);
3.1.1.6、关闭socket
  1. //关闭socket
  2. close(sockfd);

3.1.2、客户端

3.1.2.1、创建socket
  1. //创建socket
  2. if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
  3. perror("socket creation failed");
  4. exit(EXIT_FAILURE);
  5. }
3.1.2.2、设置服务器地址结构
  1. //设置服务器地址结构
  2. memset(&server_addr, 0, sizeof(server_addr));
  3. server_addr.sin_family = AF_INET;
  4. server_addr.sin_port = htons(SERVER_PORT);
  5. if (inet_pton(AF_INET, SERVER_IP, &server_addr.sin_addr) <= 0) {
  6. perror("Invalid address/Address not supported");
  7. exit(EXIT_FAILURE);
  8. }
3.1.2.3、发送数据
  1. //发送数据
  2. len = sendto(sockfd, (const char *)buffer, strlen(buffer), MSG_CONFIRM,(const struct sockaddr *)&server_addr, sizeof(server_addr));
  3. if (len < 0) {
  4. perror("sendto failed");
  5. exit(EXIT_FAILURE);
  6. }
3.1.2.4、打印发送信息

打印一条消息到控制台,表明数据已成功发送到服务器

printf("Message sent to server: %s\n", buffer);
3.1.2.5、关闭socket
  1. //关闭socket
  2. close(sockfd);

3.2、基于socket编程的UDP通信流程图

3.3、基于socket编程的UDP通信示例

3.3.1、服务端(service.c)

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <unistd.h>
  5. #include <sys/types.h>
  6. #include <sys/socket.h>
  7. #include <netinet/in.h>
  8. #include <arpa/inet.h>
  9. #define SERVER_PORT 8888
  10. #define BUFFER_SIZE 1024
  11. int main() {
  12. int sockfd;
  13. struct sockaddr_in server_addr, client_addr;
  14. socklen_t len = sizeof(client_addr);
  15. char buffer[BUFFER_SIZE];
  16. ssize_t n;
  17. //创建socket
  18. if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
  19. perror("socket creation failed");
  20. exit(EXIT_FAILURE);
  21. }
  22. //设置服务器地址结构
  23. memset(&server_addr, 0, sizeof(server_addr));
  24. server_addr.sin_family = AF_INET;
  25. server_addr.sin_addr.s_addr = inet_addr("127.0.0.1");//具体的ip地址
  26. server_addr.sin_port = htons(SERVER_PORT);
  27. //绑定socket
  28. if (bind(sockfd, (const struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) {
  29. perror("bind failed");
  30. exit(EXIT_FAILURE);
  31. }
  32. printf("Server is listening on port %d\n", SERVER_PORT);
  33. //接收数据
  34. n = recvfrom(sockfd, (char *)buffer, BUFFER_SIZE, MSG_WAITALL,
  35. (struct sockaddr *)&client_addr, &len);
  36. buffer[n] = '\0';
  37. printf("Received from client: %s\n", buffer);
  38. //关闭socket
  39. close(sockfd);
  40. return 0;
  41. }

3.3.2、客户端(client.c)

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <unistd.h>
  5. #include <sys/types.h>
  6. #include <sys/socket.h>
  7. #include <netinet/in.h>
  8. #include <arpa/inet.h>
  9. #define SERVER_IP "127.0.0.1"
  10. #define SERVER_PORT 8888
  11. #define BUFFER_SIZE 1024
  12. int main() {
  13. int sockfd;
  14. struct sockaddr_in server_addr;
  15. char buffer[BUFFER_SIZE] = "Hello from UDP client!";
  16. ssize_t len;
  17. //创建socket
  18. if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
  19. perror("socket creation failed");
  20. exit(EXIT_FAILURE);
  21. }
  22. //设置服务器地址结构
  23. memset(&server_addr, 0, sizeof(server_addr));
  24. server_addr.sin_family = AF_INET;
  25. server_addr.sin_port = htons(SERVER_PORT);
  26. if (inet_pton(AF_INET, SERVER_IP, &server_addr.sin_addr) <= 0) {
  27. perror("Invalid address/Address not supported");
  28. exit(EXIT_FAILURE);
  29. }
  30. //发送数据
  31. len = sendto(sockfd, (const char *)buffer, strlen(buffer), MSG_CONFIRM,
  32. (const struct sockaddr *)&server_addr, sizeof(server_addr));
  33. if (len < 0) {
  34. perror("sendto failed");
  35. exit(EXIT_FAILURE);
  36. }
  37. printf("Message sent to server: %s\n", buffer);
  38. //关闭socket
  39. close(sockfd);
  40. return 0;
  41. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/weixin_40725706/article/detail/850738
推荐阅读
相关标签
  

闽ICP备14008679号