0

0

如何在C++中实现TCP服务器_网络编程实例解析

穿越時空

穿越時空

发布时间:2025-07-06 10:46:11

|

438人浏览过

|

来源于php中文网

原创

c++++ tcp服务器处理多个并发连接的方法包括:1. 多线程,为每个客户端连接创建一个新线程;2. 多进程,为每个客户端连接创建一个新进程;3. i/o多路复用(如select、poll、epoll),使用单个或少量线程管理多个连接;4. 线程池,使用固定大小的线程池处理连接,限制资源消耗。这些方法各有优劣,适用于不同场景,并发量大时推荐使用epoll或线程池方案。

如何在C++中实现TCP服务器_网络编程实例解析

在C++中实现TCP服务器,需要理解Socket编程的基本概念,并利用操作系统提供的网络API。简单来说,就是创建Socket,绑定地址,监听连接,接受连接,然后进行数据收发。

如何在C++中实现TCP服务器_网络编程实例解析

解决方案

如何在C++中实现TCP服务器_网络编程实例解析

以下是一个简单的C++ TCP服务器的实现示例,它接受客户端连接并回显客户端发送的数据:

立即学习C++免费学习笔记(深入)”;

#include 
#include 
#include 
#include 
#include 
#include 
#include 

const int BUFFER_SIZE = 1024;
const int PORT = 8080;

int main() {
    // 1. 创建socket
    int server_fd = socket(AF_INET, SOCK_STREAM, 0);
    if (server_fd == -1) {
        perror("socket failed");
        return 1;
    }

    // 2. 设置socket地址
    sockaddr_in address;
    address.sin_family = AF_INET;
    address.sin_addr.s_addr = INADDR_ANY;
    address.sin_port = htons(PORT);

    // 3. 绑定socket到指定地址和端口
    if (bind(server_fd, (sockaddr*)&address, sizeof(address)) < 0) {
        perror("bind failed");
        close(server_fd);
        return 1;
    }

    // 4. 开始监听连接
    if (listen(server_fd, 3) < 0) { // backlog设置为3,表示最多允许3个等待连接的客户端
        perror("listen failed");
        close(server_fd);
        return 1;
    }

    std::cout << "Server listening on port " << PORT << std::endl;

    // 5. 接受连接
    sockaddr_in client_address;
    socklen_t client_address_len = sizeof(client_address);
    int new_socket = accept(server_fd, (sockaddr*)&client_address, &client_address_len);
    if (new_socket < 0) {
        perror("accept failed");
        close(server_fd);
        return 1;
    }

    char buffer[BUFFER_SIZE] = {0};
    // 6. 接收和发送数据
    while (true) {
        ssize_t bytes_received = recv(new_socket, buffer, BUFFER_SIZE - 1, 0);
        if (bytes_received > 0) {
            buffer[bytes_received] = '\0'; // 确保字符串以null结尾
            std::cout << "Received: " << buffer << std::endl;

            // 回显数据
            send(new_socket, buffer, bytes_received, 0);
            std::cout << "Sent: " << buffer << std::endl;
        } else if (bytes_received == 0) {
            std::cout << "Client disconnected." << std::endl;
            break;
        } else {
            perror("recv failed");
            break;
        }
    }

    // 7. 关闭socket
    close(new_socket);
    close(server_fd);

    return 0;
}

C++ TCP服务器如何处理多个并发连接?

如何在C++中实现TCP服务器_网络编程实例解析

单线程的TCP服务器只能一次处理一个连接。要处理多个并发连接,可以使用以下几种方法:

  1. 多线程: 为每个客户端连接创建一个新线程。这是一种简单直接的方法,但当连接数量非常大时,线程创建和管理的开销会变得很高。
#include 

void* handle_connection(void* socket_desc) {
    int new_socket = *(int*)socket_desc;
    // ... 处理客户端连接 ...
    close(new_socket);
    pthread_exit(NULL);
}

int main() {
    // ... 创建socket,绑定,监听 ...

    while (true) {
        int new_socket = accept(server_fd, (sockaddr*)&client_address, &client_address_len);
        if (new_socket < 0) {
            perror("accept failed");
            continue;
        }

        pthread_t thread_id;
        int *new_sock_ptr = new int(new_socket); // 动态分配内存,避免线程间竞争
        if( pthread_create( &thread_id , NULL ,  handle_connection , (void*) new_sock_ptr) < 0)
        {
            perror("could not create thread");
            close(new_socket);
            delete new_sock_ptr;
            continue;
        }
        pthread_detach(thread_id); // 分离线程,避免内存泄漏
    }

    // ... 关闭socket ...
    return 0;
}
  1. 多进程: 为每个客户端连接创建一个新进程。与多线程类似,但进程间的资源隔离更好。然而,进程创建的开销比线程更大。

  2. I/O多路复用 (select, poll, epoll): 使用单个线程或少量线程来管理多个连接。这种方法通过监听多个socket上的事件,然后在事件发生时进行处理,从而避免了为每个连接创建线程或进程的开销。epoll在处理大量并发连接时通常比selectpoll更有效率。

#include 

int main() {
    // ... 创建socket,绑定,监听 ...

    int epoll_fd = epoll_create1(0);
    if (epoll_fd == -1) {
        perror("epoll_create1");
        return 1;
    }

    epoll_event event;
    event.data.fd = server_fd;
    event.events = EPOLLIN; // 监听读事件

    if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, server_fd, &event) == -1) {
        perror("epoll_ctl: server_fd");
        return 1;
    }

    epoll_event events[10]; // 存储就绪的事件

    while (true) {
        int nfds = epoll_wait(epoll_fd, events, 10, -1); // -1 表示阻塞等待
        if (nfds == -1) {
            perror("epoll_wait");
            return 1;
        }

        for (int i = 0; i < nfds; ++i) {
            if (events[i].data.fd == server_fd) {
                // 新连接
                int new_socket = accept(server_fd, (sockaddr*)&client_address, &client_address_len);
                if (new_socket < 0) {
                    perror("accept failed");
                    continue;
                }

                // 将新socket添加到epoll监听
                event.data.fd = new_socket;
                event.events = EPOLLIN;
                if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, new_socket, &event) == -1) {
                    perror("epoll_ctl: new_socket");
                    close(new_socket);
                    continue;
                }
            } else {
                // 现有连接上的数据
                int client_socket = events[i].data.fd;
                char buffer[BUFFER_SIZE] = {0};
                ssize_t bytes_received = recv(client_socket, buffer, BUFFER_SIZE - 1, 0);
                if (bytes_received > 0) {
                    buffer[bytes_received] = '\0';
                    std::cout << "Received from " << client_socket << ": " << buffer << std::endl;
                    send(client_socket, buffer, bytes_received, 0);
                } else if (bytes_received == 0) {
                    std::cout << "Client " << client_socket << " disconnected." << std::endl;
                    epoll_ctl(epoll_fd, EPOLL_CTL_DEL, client_socket, NULL); // 从epoll中移除
                    close(client_socket);
                } else {
                    perror("recv failed");
                    epoll_ctl(epoll_fd, EPOLL_CTL_DEL, client_socket, NULL);
                    close(client_socket);
                }
            }
        }
    }

    close(epoll_fd);
    close(server_fd);
    return 0;
}
  1. 线程池: 使用一个固定大小的线程池来处理客户端连接。当有新连接到达时,将其分配给线程池中的一个空闲线程。这种方法可以限制线程数量,避免过度消耗系统资源。

如何处理TCP连接中的粘包和拆包问题?

TCP是面向流的协议,数据在传输过程中可能会出现粘包(多个数据包合并成一个)或拆包(一个数据包被分成多个)的问题。为了解决这个问题,需要在应用层进行处理。

常见的解决方案包括:

  1. 固定长度: 每个数据包都具有固定的长度。接收方可以按照固定长度读取数据。

  2. 分隔符: 在每个数据包的末尾添加一个特殊的分隔符。接收方通过查找分隔符来分割数据包。例如,可以使用换行符\n作为分隔符。

  3. 长度字段: 在每个数据包的开头添加一个长度字段,指示数据包的长度。接收方首先读取长度字段,然后根据长度读取剩余的数据。

// 使用长度字段解决粘包/拆包示例
#include 
#include 
#include 
#include 

// 封装数据,添加长度字段
std::vector package_data(const std::string& data) {
    int data_length = data.length();
    std::vector package(sizeof(int) + data_length);
    std::memcpy(package.data(), &data_length, sizeof(int)); // 写入长度
    std::memcpy(package.data() + sizeof(int), data.c_str(), data_length); // 写入数据
    return package;
}

// 解包数据,读取长度和数据
std::string unpackage_data(const char* buffer) {
    int data_length;
    std::memcpy(&data_length, buffer, sizeof(int)); // 读取长度
    std::string data(buffer + sizeof(int), data_length); // 读取数据
    return data;
}

int main() {
    std::string message = "Hello, TCP!";
    std::vector packaged_data = package_data(message);

    // 模拟接收到的数据
    char received_buffer[100];
    std::memcpy(received_buffer, packaged_data.data(), packaged_data.size());

    std::string unpacked_message = unpackage_data(received_buffer);
    std::cout << "Unpacked message: " << unpacked_message << std::endl;

    return 0;
}
  1. TLV (Type-Length-Value): 一种更通用的数据包格式,包含类型、长度和值三个字段。

如何处理TCP连接断开的情况?

TCP连接可能会因为各种原因断开,例如客户端主动关闭连接、网络故障、服务器崩溃等。服务器需要能够检测到连接断开,并进行相应的处理,例如释放资源、清理状态等。

可以通过以下方式检测连接断开:

  1. recv返回值: 当客户端关闭连接时,recv函数会返回0。

  2. errno: recvsend函数如果出错,会设置errno变量。某些errno值,例如ECONNRESET(连接被对方重置)或EPIPE(管道破裂),表示连接已断开。

    ImgGood
    ImgGood

    免费在线AI照片编辑器

    下载
  3. 心跳机制: 服务器定期向客户端发送心跳包,如果客户端在一定时间内没有响应,则认为连接已断开。

// 在主循环中,检查recv的返回值
while (true) {
    ssize_t bytes_received = recv(new_socket, buffer, BUFFER_SIZE - 1, 0);
    if (bytes_received > 0) {
        // ... 处理数据 ...
    } else if (bytes_received == 0) {
        std::cout << "Client disconnected." << std::endl;
        break; // 退出循环,关闭连接
    } else {
        if (errno == ECONNRESET || errno == EPIPE) {
            std::cout << "Connection reset by peer." << std::endl;
            break;
        }
        perror("recv failed");
        break;
    }
}

C++ TCP服务器如何实现优雅关闭?

优雅关闭指的是在服务器关闭之前,先停止接受新的连接,并处理完当前正在处理的连接,然后再关闭服务器。这可以避免客户端在连接过程中丢失数据。

实现优雅关闭的步骤:

  1. 停止接受新连接: 调用shutdown(server_fd, SHUT_RD)阻止socket接收新的连接。

  2. 处理现有连接: 等待所有现有连接处理完成。可以使用计数器来跟踪当前正在处理的连接数量,并在每个连接处理完成后递减计数器。

  3. 关闭socket: 调用close(server_fd)关闭socket。

C++ TCP服务器的性能优化策略有哪些?

  1. 选择合适的I/O模型: epoll通常比selectpoll更适合处理大量并发连接。

  2. 使用非阻塞I/O: 配合epoll使用非阻塞I/O可以避免线程阻塞在recvsend调用上。

  3. 调整TCP参数: 可以通过setsockopt函数调整TCP参数,例如TCP_NODELAY(禁用Nagle算法)和SO_REUSEADDR(允许端口重用)。

  4. 使用零拷贝技术: 例如sendfile,可以减少数据在内核空间和用户空间之间的拷贝次数。

  5. 使用连接池: 对于需要频繁建立和断开连接的应用,可以使用连接池来重用连接,减少连接建立的开销。

  6. 数据压缩: 对于传输大量数据的应用,可以使用数据压缩来减少网络带宽的占用。

  7. 缓存: 对于频繁访问的数据,可以使用缓存来减少磁盘I/O。

  8. 代码优化: 使用高效的数据结构和算法,减少内存分配和拷贝,避免不必要的系统调用。

如何使用C++实现一个简单的TCP客户端?

TCP客户端的代码与服务器端类似,主要区别在于客户端需要主动连接服务器。

#include 
#include 
#include 
#include 
#include 
#include 
#include 

const int BUFFER_SIZE = 1024;
const int PORT = 8080;
const char* SERVER_IP = "127.0.0.1"; // 服务器IP地址

int main() {
    // 1. 创建socket
    int client_fd = socket(AF_INET, SOCK_STREAM, 0);
    if (client_fd == -1) {
        perror("socket failed");
        return 1;
    }

    // 2. 设置服务器地址
    sockaddr_in server_address;
    server_address.sin_family = AF_INET;
    server_address.sin_port = htons(PORT);
    if (inet_pton(AF_INET, SERVER_IP, &server_address.sin_addr) <= 0) {
        perror("inet_pton failed");
        close(client_fd);
        return 1;
    }

    // 3. 连接服务器
    if (connect(client_fd, (sockaddr*)&server_address, sizeof(server_address)) < 0) {
        perror("connect failed");
        close(client_fd);
        return 1;
    }

    std::cout << "Connected to server." << std::endl;

    // 4. 发送和接收数据
    char buffer[BUFFER_SIZE] = {0};
    std::string message;
    while (true) {
        std::cout << "Enter message: ";
        std::getline(std::cin, message);

        if (message == "exit") {
            break;
        }

        send(client_fd, message.c_str(), message.length(), 0);
        ssize_t bytes_received = recv(client_fd, buffer, BUFFER_SIZE - 1, 0);

        if (bytes_received > 0) {
            buffer[bytes_received] = '\0';
            std::cout << "Received: " << buffer << std::endl;
        } else if (bytes_received == 0) {
            std::cout << "Server disconnected." << std::endl;
            break;
        } else {
            perror("recv failed");
            break;
        }
    }

    // 5. 关闭socket
    close(client_fd);
    return 0;
}

相关专题

更多
treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

534

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

17

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

14

2026.01.06

length函数用法
length函数用法

length函数用于返回指定字符串的字符数或字节数。可以用于计算字符串的长度,以便在查询和处理字符串数据时进行操作和判断。 需要注意的是length函数计算的是字符串的字符数,而不是字节数。对于多字节字符集,一个字符可能由多个字节组成。因此,length函数在计算字符串长度时会将多字节字符作为一个字符来计算。更多关于length函数的用法,大家可以阅读本专题下面的文章。

918

2023.09.19

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

480

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

143

2025.12.24

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

400

2023.08.14

PHP 高并发与性能优化
PHP 高并发与性能优化

本专题聚焦 PHP 在高并发场景下的性能优化与系统调优,内容涵盖 Nginx 与 PHP-FPM 优化、Opcode 缓存、Redis/Memcached 应用、异步任务队列、数据库优化、代码性能分析与瓶颈排查。通过实战案例(如高并发接口优化、缓存系统设计、秒杀活动实现),帮助学习者掌握 构建高性能PHP后端系统的核心能力。

98

2025.10.16

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

8

2026.01.15

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 6.8万人学习

C 教程
C 教程

共75课时 | 4万人学习

C++教程
C++教程

共115课时 | 12.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号