TCP与UDP协议的Socket编程详解(附详细示例)

 更新时间:2025年08月23日 09:11:58   作者:bkspiderx  
scoket编程即套接字编程,是网络编程的基础,它允许两台或者多台计算机进行网络通信,这篇文章主要介绍了TCP与UDP协议的Socket编程的相关资料,需要的朋友可以参考下

前言

在网络通信中,Socket是实现不同主机间进程通信的核心接口。TCP和UDP作为传输层的两大核心协议,其Socket编程方式因协议特性差异而有所不同。本文将详细介绍两种协议的Socket编程流程、实现示例及核心差异。

一、TCP协议的Socket编程

TCP(传输控制协议)是面向连接的可靠传输协议,其Socket编程需遵循"建立连接-数据传输-释放连接"的流程,类似现实中的"打电话"场景。

1. 核心流程

服务器端流程:

  1. 创建Socket:生成用于通信的文件描述符,指定协议族和套接字类型。
  2. 绑定地址:将Socket与特定IP地址和端口绑定,确定通信端点。
  3. 监听连接:进入被动监听状态,等待客户端连接请求。
  4. 接受连接:从等待队列中取出客户端连接,创建新的Socket用于数据传输。
  5. 收发数据:通过新Socket与客户端进行双向数据交互。
  6. 关闭连接:通信结束后,关闭Socket释放资源。

客户端流程:

  1. 创建Socket:生成与服务器通信的Socket。
  2. 发起连接:向服务器的IP和端口发送连接请求。
  3. 收发数据:连接建立后,与服务器进行数据交互。
  4. 关闭连接:通信结束后,关闭Socket。

2. TCP Socket编程示例

服务器端代码(C语言):

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>

#define PORT 8080
#define BUFFER_SIZE 1024

int main() {
    int server_fd, new_socket;
    struct sockaddr_in address;
    int opt = 1;
    int addrlen = sizeof(address);
    char buffer[BUFFER_SIZE] = {0};
    const char *response = "Hello from TCP server";

    // 1. 创建TCP套接字(SOCK_STREAM表示流式套接字)
    if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) == 0) {
        perror("socket failed");
        exit(EXIT_FAILURE);
    }

    // 设置套接字选项,允许端口复用
    if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR | SO_REUSEPORT, &opt, sizeof(opt))) {
        perror("setsockopt");
        exit(EXIT_FAILURE);
    }
    
    // 配置地址信息(IPv4、任意本地IP、端口8080)
    address.sin_family = AF_INET;
    address.sin_addr.s_addr = INADDR_ANY;
    address.sin_port = htons(PORT);

    // 2. 绑定套接字到指定端口
    if (bind(server_fd, (struct sockaddr *)&address, sizeof(address)) < 0) {
        perror("bind failed");
        exit(EXIT_FAILURE);
    }

    // 3. 监听连接请求(最大等待队列长度为5)
    if (listen(server_fd, 5) < 0) {
        perror("listen");
        exit(EXIT_FAILURE);
    }
    
    printf("TCP Server listening on port %d...\n", PORT);

    // 4. 接受客户端连接(阻塞等待)
    if ((new_socket = accept(server_fd, (struct sockaddr *)&address, (socklen_t*)&addrlen)) < 0) {
        perror("accept");
        exit(EXIT_FAILURE);
    }

    // 5. 读取客户端数据
    ssize_t valread = read(new_socket, buffer, BUFFER_SIZE);
    printf("Received from client: %s\n", buffer);

    // 向客户端发送响应
    send(new_socket, response, strlen(response), 0);
    printf("Response sent to client\n");

    // 6. 关闭连接
    close(new_socket);
    close(server_fd);
    return 0;
}

客户端代码(C语言):

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

#define PORT 8080
#define BUFFER_SIZE 1024

int main() {
    int sock = 0;
    struct sockaddr_in serv_addr;
    char buffer[BUFFER_SIZE] = {0};
    const char *message = "Hello from TCP client";

    // 1. 创建TCP套接字
    if ((sock = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
        printf("\nSocket creation error\n");
        return -1;
    }

    // 配置服务器地址信息
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_port = htons(PORT);

    // 转换IP地址(字符串转二进制)
    if (inet_pton(AF_INET, "127.0.0.1", &serv_addr.sin_addr) <= 0) {
        printf("\nInvalid address/ Address not supported\n");
        return -1;
    }

    // 2. 连接服务器(三次握手在此过程完成)
    if (connect(sock, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) < 0) {
        printf("\nConnection failed\n");
        return -1;
    }

    // 3. 向服务器发送数据
    send(sock, message, strlen(message), 0);
    printf("Message sent to server\n");

    // 接收服务器响应
    ssize_t valread = read(sock, buffer, BUFFER_SIZE);
    printf("Received from server: %s\n", buffer);

    // 4. 关闭连接
    close(sock);
    return 0;
}

二、UDP协议的Socket编程

UDP(用户数据报协议)是无连接的不可靠传输协议,其Socket编程无需建立连接,直接发送数据报,类似现实中的"寄明信片"场景。

1. 核心流程

服务器端流程:

  1. 创建Socket:生成数据报套接字。
  2. 绑定地址:将Socket与特定IP和端口绑定。
  3. 收发数据:通过recvfrom()接收客户端数据(含客户端地址),通过sendto()向客户端发送响应。
  4. 关闭Socket:通信结束后释放资源。

客户端流程:

  1. 创建Socket:生成数据报套接字。
  2. 收发数据:通过sendto()向服务器发送数据(指定服务器地址),通过recvfrom()接收服务器响应。
  3. 关闭Socket:通信结束后释放资源。

2. UDP Socket编程示例

服务器端代码(C语言):

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>

#define PORT 8080
#define BUFFER_SIZE 1024

int main() {
    int sockfd;
    char buffer[BUFFER_SIZE];
    struct sockaddr_in servaddr, cliaddr;
    const char *response = "Hello from UDP server";

    // 1. 创建UDP套接字(SOCK_DGRAM表示数据报套接字)
    if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
        perror("socket creation failed");
        exit(EXIT_FAILURE);
    }

    memset(&servaddr, 0, sizeof(servaddr));
    memset(&cliaddr, 0, sizeof(cliaddr));

    // 配置服务器地址信息
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = INADDR_ANY;
    servaddr.sin_port = htons(PORT);

    // 2. 绑定套接字到指定端口
    if (bind(sockfd, (const struct sockaddr *)&servaddr, sizeof(servaddr)) < 0) {
        perror("bind failed");
        exit(EXIT_FAILURE);
    }

    printf("UDP Server listening on port %d...\n", PORT);

    socklen_t len = sizeof(cliaddr);

    // 3. 接收客户端数据报(阻塞等待,同时获取客户端地址)
    ssize_t n = recvfrom(sockfd, (char *)buffer, BUFFER_SIZE, MSG_WAITALL, 
                        (struct sockaddr *)&cliaddr, &len);
    buffer[n] = '\0';
    printf("Received from client: %s\n", buffer);

    // 向客户端发送响应(需指定客户端地址)
    sendto(sockfd, (const char *)response, strlen(response), 
           MSG_CONFIRM, (const struct sockaddr *)&cliaddr, len);
    printf("Response sent to client\n");

    // 4. 关闭套接字
    close(sockfd);
    return 0;
}

客户端代码(C语言):

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

#define PORT 8080
#define BUFFER_SIZE 1024

int main() {
    int sockfd;
    char buffer[BUFFER_SIZE];
    struct sockaddr_in servaddr;
    const char *message = "Hello from UDP client";

    // 1. 创建UDP套接字
    if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
        perror("socket creation failed");
        exit(EXIT_FAILURE);
    }

    memset(&servaddr, 0, sizeof(servaddr));

    // 配置服务器地址信息
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(PORT);
    servaddr.sin_addr.s_addr = inet_addr("127.0.0.1");

    socklen_t len = sizeof(servaddr);

    // 2. 向服务器发送数据报(需指定服务器地址)
    sendto(sockfd, (const char *)message, strlen(message), 
           MSG_CONFIRM, (const struct sockaddr *)&servaddr, sizeof(servaddr));
    printf("Message sent to server\n");

    // 接收服务器响应
    ssize_t n = recvfrom(sockfd, (char *)buffer, BUFFER_SIZE, 
                        MSG_WAITALL, (struct sockaddr *)&servaddr, &len);
    buffer[n] = '\0';
    printf("Received from server: %s\n", buffer);

    // 3. 关闭套接字
    close(sockfd);
    return 0;
}

三、TCP与UDP Socket编程的核心差异

对比维度TCP Socket编程UDP Socket编程
套接字类型使用SOCK_STREAM(流式套接字)使用SOCK_DGRAM(数据报套接字)
连接处理connect()(客户端)和accept()(服务器)建立连接无需建立连接,直接传输数据
数据传输函数使用read()/write()send()/recv()使用sendto()/recvfrom()(需指定地址)
地址关联连接建立后自动关联地址,无需重复指定每次传输都需显式指定目标地址
可靠性保障协议层保证可靠传输,编程无需处理丢失重传需手动实现重传、排序等可靠性机制
服务器并发需为每个客户端创建新Socket(或用多线程)单Socket即可处理多客户端(通过地址区分)

四、实践建议

  1. 协议选择:对可靠性要求高的场景(如文件传输、登录认证)用TCP;对实时性要求高的场景(如视频通话、游戏)用UDP。
  2. 错误处理:实际编程中需强化错误处理(如连接超时、数据截断等)。
  3. 性能优化:TCP可通过调整缓冲区大小优化性能;UDP需控制数据报大小(避免IP分片)。
  4. 安全性:敏感场景需在Socket通信基础上添加加密(如SSL/TLS)。

通过掌握两种协议的Socket编程差异,可根据实际需求选择合适的通信方式,构建高效、可靠的网络应用。

到此这篇关于TCP与UDP协议的Socket编程的文章就介绍到这了,更多相关TCP与UDP协议的Socket编程内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C++实现屏幕截图(全屏截图)

    C++实现屏幕截图(全屏截图)

    屏幕截图已经成为了所有IM即时通讯软件的必备模块,也是日常办公中使用最频繁的功能之一。今天我们从C++开发的角度,来看看屏幕截图的主要功能点是如何实现的,感兴趣的可以了解一下
    2021-11-11
  • C++17 使用 std::string_view避免字符串拷贝优化程序性能

    C++17 使用 std::string_view避免字符串拷贝优化程序性能

    这篇文章主要介绍了C++17 使用 std::string_view避免字符串拷贝优化程序性能,帮助大家提高程序运行速度,感兴趣的朋友可以了解下
    2020-10-10
  • C语言数组详细介绍

    C语言数组详细介绍

    大家好,本篇文章主要讲的是C语言数组详细介绍,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2022-01-01
  • C++ Primer 标准库vector示例详解

    C++ Primer 标准库vector示例详解

    该文章主要介绍了C++标准库中的vector类型,包括其定义、初始化、成员函数以及常见操作,文章详细解释了如何使用vector来存储和操作对象集合,并提供了代码示例来说明vector的使用方法,感兴趣的朋友一起看看吧
    2025-03-03
  • 基于OpenCv的运动物体检测算法

    基于OpenCv的运动物体检测算法

    这篇文章主要为大家详细介绍了基于OpenCv的运动物体检测算法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-01-01
  • Qt使用HTTP上传json格式数据

    Qt使用HTTP上传json格式数据

    这篇文章主要为大家详细介绍了Qt如何使用HTTP上传json格式数据,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下
    2024-12-12
  • Qt视频播放器的实现示例

    Qt视频播放器的实现示例

    本文主要介绍了Qt视频播放器的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-08-08
  • C语言光标旋转与倒计时功能实现示例详解

    C语言光标旋转与倒计时功能实现示例详解

    这篇文章主要为大家介绍了C语言实现光标旋转与倒计时功能的示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步早日升职加薪
    2021-11-11
  • C语言每日练习之进制转换

    C语言每日练习之进制转换

    这篇文章主要介绍了C语言进制转换,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-11-11
  • C语言SQLite3事务和锁的操作实例

    C语言SQLite3事务和锁的操作实例

    这篇文章主要介绍了C语言SQLite3事务和锁的操作,结合完整实例形式分析了C语言针对SQLite3数据库的事务与锁相关操作技巧,需要的朋友可以参考下
    2017-07-07

最新评论