【C++网络编程】(一)Linux平台下TCP客户/服务端程序

ops/2024/10/11 0:11:46/

文章目录

  • Linux平台下TCP客户/服务端程序
    • 服务端
    • 客户端
    • 相关头文件介绍

Linux平台下TCP客户/服务端程序


图片来源:https://subingwen.cn/linux/socket/

下面实现一个Linux平台下TCP客户/服务端程序:客户端向服务器发送:“你好,服务器…递增数字”,然后服务器发送响应消息:“你好,客户端”。
在这里插入图片描述

服务端

server.cpp

#include <iostream>    
#include <cstdlib>      // std::exit
#include <cstring>      // memset sprintf strlen
#include <arpa/inet.h>  // inet_ntop, htons, ntohs, INADDR_ANY, INET_ADDRSTRLEN 
#include <unistd.h>     // close
// #include <sys/socket.h> // sockaddr_in,  socket(),  bind(), listen(), accept(), send(), recv(),SOCK_STREAM,AF_INET
/*
<arpa/inet.h>包含了<netinet/in.h>,而<netinet/in.h>包含了 <sys/socket.h>。
所以实际使用时,只需要#include <arpa/inet.h>,不需要#include <sys/socket.h> 
*/int main()
{// 1. 创建监听的套接字int lfd = socket(AF_INET, SOCK_STREAM, 0);  // 创建一个TCP套接字if (lfd == -1){perror("socket");  // 错误处理std::exit(EXIT_FAILURE);}// 2. 将socket()返回值和本地的IP端口绑定到一起sockaddr_in addr;  // 用于存储地址信息addr.sin_family = AF_INET; // 地址族,IPv4addr.sin_port = htons(10000);   // 大端端口转换//addr.sin_addr.s_addr = INADDR_ANY;  // 绑定到任意IP地址inet_pton(AF_INET, "172.31.108.107", &addr.sin_addr.s_addr); // 指定IP地址int ret = bind(lfd, (sockaddr*)&addr, sizeof(addr)); // 绑定套接字到地址if (ret == -1){perror("bind");  // 错误处理std::exit(EXIT_FAILURE);}// 3. 设置监听ret = listen(lfd, 128);  // 开始监听if (ret == -1){perror("listen");  // 错误处理std::exit(EXIT_FAILURE);}// 4. 阻塞等待并接受客户端连接sockaddr_in cliaddr; // 用于存储客户端地址信息socklen_t clilen = sizeof(cliaddr);  // 客户端地址结构的大小int cfd = accept(lfd, (sockaddr*)&cliaddr, &clilen); // 接受客户端连接if (cfd == -1){perror("accept");  // 错误处理std::exit(EXIT_FAILURE);}// 打印客户端的地址信息char ip[INET_ADDRSTRLEN] = {0};  // 存储客户端IP地址std::cout << "客户端的IP地址: " << inet_ntop(AF_INET, &cliaddr.sin_addr.s_addr, ip, sizeof(ip))  // 将IP地址转换为字符串<< ", 端口: " << ntohs(cliaddr.sin_port) << std::endl;  // 端口号转换// 5. 和客户端通信while (true){// 接收数据char buf[1024];  // 接收缓冲区memset(buf, 0, sizeof(buf));  // 清零缓冲区int len = recv(cfd, buf, sizeof(buf), 0);  // 从客户端读取数据if (len > 0){std::cout << "客户端: " << buf << std::endl;  // 打印客户端发送的消息sprintf(buf, "你好, 客户端\n");  // 格式化字符串send(cfd, buf, strlen(buf), 0);  // 回应客户端}else if (len == 0){std::cout << "客户端断开了连接..." << std::endl;  // 客户端断开连接break;}else{perror("recv");  // 错误处理break;}}close(cfd);  // 关闭与客户端的连接close(lfd);  // 关闭监听套接字return 0;
}

编译与运行

g++ server.cpp -o server
./server

客户端

client.cpp

#include <iostream>     // std::cout, std::cerr
#include <cstdlib>     // std::exit
#include <unistd.h>     // close, sleep
#include <cstring>      // memset, strlen
#include <arpa/inet.h>  // socket, connect, inet_pton, htonsint main()
{// 1. 创建通信的套接字int fd = socket(AF_INET, SOCK_STREAM, 0);  // 创建一个TCP套接字if (fd == -1){perror("socket");  // 错误处理std::exit(EXIT_FAILURE);}// 2. 连接服务器sockaddr_in addr;  // 用于存储服务器地址信息addr.sin_family = AF_INET; // 地址族,IPv4addr.sin_port = htons(10000);   // 大端端口转换inet_pton(AF_INET, "172.31.108.107", &addr.sin_addr.s_addr); // 将IP地址转换为网络字节顺序int ret = connect(fd, (sockaddr*)&addr, sizeof(addr)); // 连接到服务器if (ret == -1){perror("connect");  // 错误处理std::exit(EXIT_FAILURE);}// 3. 和服务器端通信int number = 0;while (true){// 发送数据char buf[1024];  // 数据缓冲区sprintf(buf, "你好, 服务器...%d", number++);  // 格式化字符串send(fd, buf, strlen(buf), 0);  // 发送数据// 接收数据memset(buf, 0, sizeof(buf));  // 清空缓冲区int len = recv(fd, buf, sizeof(buf), 0);  // 从服务器读取数据if (len > 0){std::cout << "服务器: " << buf;  // 打印服务器发送的消息}else if (len == 0){std::cout << "服务器断开了连接..." << std::endl;  // 服务器断开连接break;}else{perror("recv");  // 错误处理break;}sleep(1);   // 每隔1秒发送一条数据}close(fd);  // 关闭套接字return 0;
}

编译与运行

g++ client.cpp -o client
./client

相关头文件介绍

  1. <cstdlib> :提供了一些常用的标准库函数,源自 C 的 stdlib.h,这些函数与程序控制、内存分配、随机数生成等功能相关。使用到的函数:

    • std::exit(int status):终止程序执行,status 用来返回退出状态码,0 表示正常退出,非 0 表示异常退出。 无返回值,直接终止程序。
  2. <cstring>:是对 C 语言 string.h 的封装,提供了用于操作 C 风格字符串(以 '\0' 结尾的字符数组)和内存操作的函数。使用到的函数:

    • ptr = memset(void* ptr, int value, size_t num):将指定内存区域的前 num 个字节设置为 value。 返回指向 ptr 的指针,即被修改的内存区域的起始地址。
    • n = sprintf(char* buffer, const char* format, ...):将格式化数据写入 buffer,并返回写入的字符数。返回写入 buffer 中的字符数(不包括终止符 '\0')。
    • len = strlen(const char* str):返回 C 风格字符串 str 的长度(不包括终止符 '\0')。 返回 str 的长度。
  3. <arpa/inet.h>:提供了一些用于网络编程的工具函数,主要用于 IP 地址与主机字节序、网络字节序的转换。使用到的函数和宏:

    • inet_ntop(int af, const void* src, char* dst, socklen_t size):将网络格式(大端序)的二进制 IP 地址转换为可读的点分十进制或冒号分隔的字符串。
    • inet_pton(int af, const char* src, void* dst):将可读的点分十进制或冒号分隔的字符串格式的 IP 地址转换为网络格式(大端序)的二进制格式。
    • netshort = htons(uint16_t hostshort):将主机字节序(小端序)的 16 位数转换为网络字节序(大端序)。 返回转换后的网络字节序的 16 位数。
    • hostshort = ntohs(uint16_t netshort):将网络字节序的 16 位数转换为主机字节序。 返回转换后的主机字节序的 16 位数。
    • INADDR_ANY:用于表示绑定到所有可用的本地接口(IP 地址为 0.0.0.0)。
    • INET_ADDRSTRLEN 是一个常量,表示用于存储 IPv4 地址的字符串格式的最大长度。其值通常为 16,这是因为 IPv4 地址的最坏情况是点分十进制表示的字符串形式,如 “255.255.255.255”,该字符串的长度为 15,加上一个字符串终止符 ‘\0’,总共为 16。
  4. <sys/socket.h>:提供了与套接字编程相关的函数和数据结构,定义了套接字的创建、绑定、监听、接受连接、数据收发等功能。使用到的函数、宏和结构体:

    • sockfd = socket(int domain, int type, int protocol):创建一个套接字,domain 表示协议族(如 IPv4 , IPv6),type 表示套接字类型(如 SOCK_STREAM 表示 TCP),protocol 通常为 0,表示默认协议。 返回新的套接字描述符,失败时返回 -1
    • result = bind(int sockfd, const struct sockaddr* addr, socklen_t addrlen):将套接字绑定到特定地址(IP 和端口)。 返回 0 表示成功,返回 -1 表示出错。
    • result = listen(int sockfd, int backlog):将套接字设置为监听模式,backlog 表示队列中可以等待的最大连接数。 返回 0 表示成功,返回 -1 表示出错。
    • new_sockfd = accept(int sockfd, struct sockaddr* addr, socklen_t* addrlen):接受连接请求,并返回一个新的套接字,失败时返回 -1
    • bytes_sent = send(int sockfd, const void* buf, size_t len, int flags):通过连接的套接字发送数据。 返回成功发送的字节数,失败时返回 -1
    • bytes_received = recv(int sockfd, void* buf, size_t len, int flags):从连接的套接字接收数据。 返回成功接收的字节数,返回 0 表示对方关闭连接,失败时返回 -1
    • AF_INET: 是一个常量,用于指定地址族,表示使用 IPv4 地址。
    • SOCK_STREAM:是一个常量,用于指定套接字类型,表示该套接字将使用 TCP 协议进行流式数据传输。
    • sockaddr_in:专门用于 IPv4 地址的结构体,包含 sin_family(地址族)、sin_port(端口号)、sin_addr(IP 地址)等字段。
      struct sockaddr_in {short int          sin_family;   // 地址族unsigned short int sin_port;     // 端口号 (网络字节序)struct in_addr     sin_addr;     // IP 地址unsigned char      sin_zero[8];  // 填充字段(未使用)
      };
      
  5. <unistd.h>:是 Unix-like 操作系统的头文件,提供了对系统调用的访问接口,包含文件操作、进程管理等低级功能。使用到的函数:

    • result = close(int fd):关闭文件描述符 fd,在网络编程中用于关闭套接字。 返回 0 表示成功,返回 -1 表示出错。
    • sleep(unsigned int seconds) :暂停执行当前线程 seconds 秒。

http://www.ppmy.cn/ops/123721.html

相关文章

Git 工作区、暂存区和仓库

在使用 Git 进行版本控制时&#xff0c;工作区、暂存区和仓库概念的详细解释&#xff1a; 1. 工作区&#xff08;Working Directory&#xff09; 工作区是你在计算机上实际编辑文件的地方。当你克隆一个 Git 仓库或在现有目录中初始化一个 Git 仓库时&#xff0c;这个目录就是…

大载重无人机物资吊运技术培训详解

大载重无人机物资吊运技术培训详解主要涉及理论知识、实操技能、安全规范以及应用领域等多个方面。以下是对这些方面的详细解析&#xff1a; 一、理论知识 1. 无人机基础知识 无人机类型与结构&#xff1a;了解大载重无人机的类型、结构特点及其工作原理&#xff0c;特别是针…

【华为HCIP实战课程七】OSPF邻居关系排错MTU问题,网络工程师

一、MTU MUT默认1500,最大传输单元,一致性检测 [R3-GigabitEthernet0/0/1]mtu 1503//更改R3的MTU为1503 查看R3和SW1之间的OSPF邻居关系正常: 默认华为设备没有开启MTU一致性检测! [R3-GigabitEthernet0/0/1]ospf mtu-enable //手动开启MTU检测 [SW1-Vlanif30]ospf mtu…

微信小程序hbuilderx+uniapp+Android 新农村综合风貌旅游展示平台

目录 项目介绍支持以下技术栈&#xff1a;具体实现截图HBuilderXuniappmysql数据库与主流编程语言java类核心代码部分展示登录的业务流程的顺序是&#xff1a;数据库设计性能分析操作可行性技术可行性系统安全性数据完整性软件测试详细视频演示源码获取方式 项目介绍 小程序端…

Allegro PCB中过孔的整体替换

Cadence Allegro PCB中过孔的整体替换 在PCB设计过程中&#xff0c;之前是使用的小的过孔&#xff0c;后面需要替换成大的过孔&#xff0c;一个一个去替换过孔非常麻烦的&#xff0c;这里&#xff0c;讲解一下如何去整体的替换过孔&#xff0c;具体的操作方法如下所示&#xf…

SpringBoot在线教育系统:从零到一的构建过程

1系统概述 1.1 研究背景 随着计算机技术的发展以及计算机网络的逐渐普及&#xff0c;互联网成为人们查找信息的重要场所&#xff0c;二十一世纪是信息的时代&#xff0c;所以信息的管理显得特别重要。因此&#xff0c;使用计算机来管理微服务在线教育系统的相关信息成为必然。开…

【Blender Python】5.Blender场景中的集合

概述 这里的“集合”是指Blender场景中的集合。你可以在“大纲视图”面板中看到 图标的&#xff0c;就是集合&#xff0c;可以看做是文件夹&#xff0c;用于分类和整理场景中的对象。 获取场景的集合 >>> C.scene bpy.data.scenes[Scene]>>> C.scene.coll…

AcWing 4890:哞操作 ← 字符串

【题目来源】https://www.acwing.com/problem/content/4893/【题目描述】 给定一个字符串&#xff0c;其中的每个字符要么是 M&#xff0c;要么是 O。 你可以通过以下操作将该字符串变为 MOO&#xff1a;改变字符串中的第一个或最后一个字符&#xff08;M 变为 O&#xff0c;O …