当前位置: 首页 > article >正文

网络编程 - - TCP套接字通信及编程实现

概述

TCP(Transmission Control Protocol,传输控制协议)是一种面向连接的、可靠的传输层协议。在网络编程中,TCP常用于实现客户端和服务器之间的可靠数据传输。本文将基于C语言实现TCP服务端和客户端建立通信的过程。

三次握手

在 TCP 连接建立之前,客户端和服务器之间需要进行三次握手来同步双方的序列号,并确认双方都准备好进行数据传输

  • 第一次握手:客户端向服务器发送一个 SYN(同步序列编号)报文段,表示请求建立连接。客户端进入 SYN_SENT 状态
  • 第二次握手:服务器收到 SYN 报文段后,回复一个 SYN-ACK(同步序列编号 + 确认)报文段,表示同意建立连接。服务器进入 SYN_RCVD 状态
  • 第三次握手:客户端收到 SYN-ACK 报文段后,回复一个 ACK(确认)报文段,表示确认收到服务器的响应。客户端和服务器都进入 ESTABLISHED 状态,连接正式建立

在这里插入图片描述

图片来源:https://img-blog.csdnimg.cn/39bb4f4da21a4513b9506ecdf6a40cf3.png

四次挥手

通信结束时,客户端或服务器可以发起断开连接的请求。断开连接的过程称为四次挥手,以确保双方都能正确关闭连接并释放资源

  • 第一次挥手:主动关闭方(通常是客户端)发送一个 FIN(终止)报文段,表示不再发送数据。主动关闭方进入 FIN_WAIT_1 状态
  • 第二次挥手:被动关闭方(通常是服务器)收到 FIN 报文段后,回复一个 ACK 报文段,表示确认收到 FIN。被动关闭方进入 CLOSE_WAIT 状态,而主动关闭方进入 FIN_WAIT_2 状态
  • 第三次挥手:被动关闭方在处理完所有未完成的数据后,发送一个 FIN 报文段,表示自己也不再发送数据。被动关闭方进入 LAST_ACK 状态
  • 第四次挥手:主动关闭方收到 FIN 报文段后,回复一个 ACK 报文段,表示确认收到 FIN。主动关闭方进入 TIME_WAIT 状态,等待一段时间(通常为2倍的最大报文段生命周期,即2MSL),以确保被动关闭方收到了最后的 ACK。之后,主动关闭方进入 CLOSED 状态,连接完全关闭

2MSL:MSL 的默认值是 30 秒,基于经验选择的一个保守估计,用来确保大多数网络环境下的数据包都能被接收或者超时。

大部分操作系统都允许用户调整 MSL 的值,从而改变 TIME_WAIT 状态的持续时间

在这里插入图片描述

图片来源:https://i-blog.csdnimg.cn/blog_migrate/843f121dd50cd8458daf1fa834bc1f36.png

TCP保证可靠传输方式

  • 序列号:每个TCP报文段都有一个序列号,表示该报文段中的第一个字节在整个数据流中的位置。接收方可以根据序列号重新排序接收到的报文段,确保数据按顺序传递
  • 确认应答:接收方在收到报文段后,会发送一个确认应答,告诉发送方哪些数据已经成功接收。发送方根据确认应答判断是否需要重传丢失或损坏的报文段
  • 超时重传:如果发送方在一定时间内没有收到确认应答,它会认为报文段可能丢失或延迟,并重新发送该报文段。TCP使用动态调整的超时机制来优化重传策略
  • 流量控制:TCP使用滑动窗口机制来控制发送方的发送速率,确保接收方不会被过多的数据淹没。接收方会在确认应答中告知发送方当前可用的接收窗口大小,发送方根据这个信息调整自己的发送速率
  • 拥塞控制:TCP通过多种算法(如慢启动、拥塞避免、快速重传和快速恢复)来动态调整发送方的发送速率,避免网络拥塞。这些算法旨在在网络负载较高时减小发送速率,在网络条件改善时逐渐增加发送速率

TCP通信实现流程

涉及到的库方法

  • 创建套接字(Socket)

    #include <sys/socket.h>
    int socket(int domain, int type, int protocol);
    
  • 绑定(Bind)套接字到指定的IP地址和端口

    #include <sys/socket.h>
    int bind(int socket, const struct sockaddr *address, socklen_t address_len);
    
  • 监听(Listen)客户端的连接请求

    #include <sys/socket.h>
    int listen(int socket, int backlog)
    
  • 接受(Accept)客户端的连接

    #include <sys/socket.h>
    int accept(int socket, struct sockaddr *restrict address, socklen_t *restrict address_len);
    
  • 连接、发送和接收数据(Send/Recv)

    #include <sys/socket.h>
    int connect(int socket, const struct sockaddr *address, socklen_t address_len);
    ssize_t send(int socket, const void *buffer, size_t length, int flags);
    ssize_t recv(int socket, void *buffer, size_t length, int flags);
    
  • 关闭连接(Close)

    #include <unistd.h>
    int close(int fildes);
    

代码实现

服务器代码(Linux)

tcp_server.h

#ifndef TCP_SERVER_H
#define TCP_SERVER_H

#include <pthread.h>
#include <netinet/in.h>
#include "cJson.h"
#include "common_base.h"

// 定义常量
#define PORT 18888
#define BUFFER_SIZE 1024
#define TCP_IP "127.0.0.1"

// 外部函数声明
extern void getSerialNoStr(char *buf);

/**
 * 启动服务器主循环,等待客户端连接
 */
void start_serve_tcp(void *arg);

/**
 * 解析接收到的消息
*/
void parse_message(char *data, size_t data_size);

#endif // TCP_SERVER_H

tcp_server.c

#include "tcp_server.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <pthread.h>

/**
 * 启动TCP服务器,等待客户端连接并处理接收到的数据。
 *
 * @param arg 传递给线程的参数,通常为NULL。
 */
void start_serve_tcp(void *arg) 
{
    int server_fd, client_fd;
    struct sockaddr_in server_addr, client_addr;
    socklen_t client_addr_len = sizeof(client_addr);
    char buffer[BUFFER_SIZE] = {0};

    // 创建 Socket
    if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
    {
        printf("Socket 创建失败");
        return;
    }

    // 配置服务器地址
    memset_s(&server_addr, sizeof(server_addr), 0, sizeof(server_addr));
    server_addr.sin_family = AF_INET;

    // 配置端口号和IP
    server_addr.sin_port = htons(PORT);
    server_addr.sin_addr.s_addr = inet_addr(TCP_IP);
    
   // 绑定 Socket
    if (bind(server_fd, (struct sockaddr *)&server_addr, sizeof(server_addr)) == -1) 
    {
        printf("绑定失败\n");
        close(server_fd);
        return;
    }

    // 开始监听
    if (listen(server_fd, 5) == -1)
    {
        printf("监听失败\n");
        close(server_fd);
        return;
    }

    printf("服务器已启动,监听地址:%s:%d\n", inet_ntoa(server_addr.sin_addr), ntohs(server_addr.sin_port));

    // 等待客户端连接
    while (1)
    {
        printf("等待客户端连接...\n");

        if ((client_fd = accept(server_fd, (struct sockaddr *)&client_addr, &client_addr_len)) == -1) 
        {
            printf("接受客户端连接失败");
            continue;
        }

        printf("客户端已连接:%s:%d\n",
               inet_ntoa(client_addr.sin_addr), ntohs(client_addr.sin_port));

        // 接收数据
        int received_size;
        printf("receive before client_fd: %d\n", client_fd);
        while ((received_size = recv(client_fd, buffer, BUFFER_SIZE, 0)) > 0) 
        {
            printf("收到数据,大小: %d 字节\n", received_size);
            parse_message(buffer, received_size);

            printf("jsonString: %s\n", jsonString);
            sleep(1);
            
            if(jsonString)
            {              
                printf("receive after client_fd: %d\n", client_fd);
                // 发送 JSON 数据到客户端
                ssize_t sent_size = send(client_fd, jsonString, strlen(jsonString), 0);
                if (sent_size == -1)
                {
                    printf("发送数据失败\n");
                } 
                else 
                {
                    printf("成功发送 %zd 字节到客户端\n", sent_size);
                }

                free(jsonString);
                jsonString = NULL;
            }
            else 
            {
                printf("生成 JSON 数据失败\n");
            }
        }

        if (received_size == 0)
        {
            printf("客户端已断开连接\n");
        } 
        else if (received_size == -1)
        {
            printf("接收数据失败");
        }

        close(client_fd);
    }

    close(server_fd);
}

void parse_message(char *data, size_t data_size) 
{
    cJSON *rootMsg = NULL;  
	cJSON *serialNo = NULL; // 序列号
	cJSON *netCmd = NULL;   // 操作行为
    MANUAL_TRIG_PARAM manualTrParam;    // 手动触发抓拍接口传参结构体

    // 确保消息头部完整性(消息类型:4字节,数据长度:4字节)
    if (data_size < 8)
    {
        printf("数据长度不足,无法解析\n");
        return;
    }

     // 解析消息类型和数据长度
    int message_type = ntohl(*(int *)data);
    int data_length = ntohl(*(int *)(data + 4));

    printf("消息类型: %d\n", message_type);
    printf("数据长度: %d\n", data_length);

    // 检查数据长度是否匹配
    if (data_size < 8 + data_length) 
    {
        printf("数据长度与实际内容不匹配\n");
        return;
    }
	
     // 解析消息内容
    char *message_content = (char *)malloc(data_length + 1);
    if (!message_content) 
    {
        printf("内存分配失败\n");
        return;
    }
    memcpy(message_content, data + 8, data_length);
    message_content[data_length] = '\0'; // 确保字符串以 \0 结尾

    printf("消息内容: %s\n", message_content);

    rootMsg = cJSON_Parse(message_content);
    if (NULL == rootMsg)
	{
		printf("rootMsg is not json tpye\n");
        free(message_content);
        return;
	}
    printf("解析成功\n");

    serialNo = cJSON_GetObjectItem(rootMsg, "serialNo");
    netCmd = cJSON_GetObjectItem(rootMsg, "netCmd");

    if (serialNo && netCmd)
    {
        printf("serialNo: %s\n", serialNo->valuestring);
        printf("netCmd: %s\n", netCmd->valuestring);
    } 
    else
    {
        printf("缺少必要的 JSON 字段\n");
        cJSON_Delete(rootMsg);
        free(message_content);
        return;
    }

    char serial[64] = {0};
    getSerialNoStr(serial);

    printf("getSerialNoStr: %s\n", serial);
    printf("serialNo: %s\n", serialNo->valuestring);
    
    if(strncmp(serial, serialNo->valuestring, 9) == 0)
    {
        // TODO:业务处理
    }
    else
    {
        printf("serialNo is not equal\n");
    }

    cJSON_Delete(rootMsg);
    free(message_content);
}


客户端代码(Windows)
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <winsock2.h>
#include <ws2tcpip.h>

#pragma comment(lib, "ws2_32.lib")

void send_message(const char *ip, int port, int message_type, const char *content) {
    WSADATA wsa;
    SOCKET sock;
    struct sockaddr_in server_addr;
    char buffer[1024];
    int recv_size;
    char recv_buffer[1024];

    // 初始化 Winsock
    if (WSAStartup(MAKEWORD(2, 2), &wsa) != 0) {
        printf("Winsock 初始化失败,错误代码: %d\n", WSAGetLastError());
        exit(EXIT_FAILURE);
    }

    // 创建 Socket
    if ((sock = socket(AF_INET, SOCK_STREAM, 0)) == INVALID_SOCKET) {
        printf("Socket 创建失败,错误代码: %d\n", WSAGetLastError());
        WSACleanup();
        exit(EXIT_FAILURE);
    }

    // 配置服务器地址
    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(port);
    server_addr.sin_addr.s_addr = inet_addr(ip);

    // 连接服务器
    if (connect(sock, (struct sockaddr *)&server_addr, sizeof(server_addr)) == SOCKET_ERROR) {
        printf("连接服务器失败,错误代码: %d\n", WSAGetLastError());
        closesocket(sock);
        WSACleanup();
        exit(EXIT_FAILURE);
    }
    // 准备消息
    int data_length = strlen(content);
    int net_message_type = htonl(message_type);
    int net_data_length = htonl(data_length);


    memcpy(buffer, &net_message_type, 4);
    memcpy(buffer + 4, &net_data_length, 4);
    memcpy(buffer + 8, content, data_length);

    // 发送消息
    send(sock, buffer, 8 + data_length, 0);

     // 接收服务器返回的数据
    if ((recv_size = recv(sock, recv_buffer, sizeof(recv_buffer) - 1, 0)) == SOCKET_ERROR) {
        printf("接收数据失败,错误代码: %d\n", WSAGetLastError());
    } else {
        recv_buffer[recv_size] = '\0'; // 确保以 null 结尾
        printf("服务器返回: %s\n", recv_buffer);
    }

    closesocket(sock);
    WSACleanup();
}

int main() {
    send_message("127.0.0.1", 18888, 1, "{id: 'FS123456', command: 'NET_CAP'}");
    return 0;
}

http://www.kler.cn/a/509108.html

相关文章:

  • python中的RPA->playwright自动化录制脚本实战案例笔记
  • git命令
  • RV1126+FFMPEG推流项目(6)视频码率及其码率控制方式
  • 菜品管理(day03)
  • JVM直击重点
  • Android面试题
  • 配置web服务端对https进行抓包
  • Python学习指南:从零到进阶的系统流程
  • UllnnovationHub,一个开源的WPF控件库
  • AI 音频工具合集
  • edge浏览器恢复旧版滚动条
  • LLM | 大模型微调学习资源合集个人整理(持续更新)
  • 国产编辑器EverEdit - 列编辑模式
  • 【ROS2 中间件RMW】基于FastDDS共享内存实现ROS2跨进程零拷贝通讯
  • python——句柄
  • 在线json格式化工具
  • Webpack简述
  • 如何在没有root权限的情况下使用R语言
  • 在线图片压缩工具
  • 2024年12月蓝桥杯Scratch12月stema选拔赛真题—小星星
  • 微软确认Win10停更不碍Microsoft 365使用!未来是否更新成谜
  • 复健第二天之[SWPUCTF 2022 新生赛]ez_ez_unserialize
  • leetcode刷题记录(六十一)——73. 矩阵置零
  • C# 反射获取私有静态方法详解
  • 移动端布局 ---- 学习分享
  • AWTK fscript 中的 输入/出流 扩展函数