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

【Linux】socket网络编程之TCP

在这里插入图片描述

个人主页~


socket网络编程之TCP

  • 一、TCP实现回显服务器
    • 1、服务端
      • (一)TcpServer.hpp
      • (二)main.cpp
    • 2、客户端
      • TcpClient.cpp
  • 二、服务器Start函数
    • 1、多进程版
    • 2、多线程版

一、TCP实现回显服务器

1、服务端

(一)TcpServer.hpp

#include <iostream>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <cstring>
#include <pthread.h>
#include <sys/wait.h>const std::string defaultip = "0.0.0.0";
const int defaultfd = -1;//枚举错误类型
enum
{UsageError = 1,SocketError,BindError,ListenError,
};//封装客户端连接相关信息
class ThreadData
{
public:ThreadData(int fd, const std::string &ip, const uint16_t &p) : sockfd(fd), clientip(ip), clientport(p){}public:int sockfd;std::string clientip;uint16_t clientport;
};class TcpServer
{
public:TcpServer(const uint16_t &port, const std::string &ip = defaultip) : listensock_(defaultfd), port_(port), ip_(ip){}void InitServer(){	//IPv4协议,TCP套接字listensock_ = socket(AF_INET, SOCK_STREAM, 0);if (listensock_ < 0){exit(SocketError);}//local存储本地服务器地址信息并初始化为0struct sockaddr_in local;memset(&local, 0, sizeof(local));//IPv4协议,端口号,IP地址local.sin_family = AF_INET;local.sin_port = htons(port_);inet_aton(ip_.c_str(), &(local.sin_addr));//调用bind将套接字listensock_ 绑定到本地地址localif (bind(listensock_, (struct sockaddr *)&local, sizeof(local)) < 0){exit(BindError);}//将套接字设置为监听状态,最多允许五个客户端连接请求排队等待处理if (listen(listensock_, 5) < 0){exit(ListenError);}}void Start(){while (true){struct sockaddr_in client;socklen_t len = sizeof(client);//新套接字sockfd用于与发起连接请求的客户端进行数据传输//原来的listensock_继续监听int sockfd = accept(listensock_, (struct sockaddr *)&client, &len);if (sockfd < 0){continue;}//将客户端端口号转换为主机字节序uint16_t clientport = ntohs(client.sin_port);//将客户端ip转换为点分十进制字符串char clientip[32];inet_ntop(AF_INET, &(client.sin_addr), clientip, sizeof(clientip));//多进程,从这里开始的下面这段代码,可以用多线程以及线程池替代,在后面说说多线程//这里很巧妙的设计,我们在后边与多线程一起解释pid_t id = fork();if (id == 0){// 子进程关闭监听close(listensock_);//子进程创建“孙子”进程if (fork() > 0)exit(0);Service(sockfd, clientip, clientport); close(sockfd);exit(0);}close(sockfd);// 父进程等待pid_t rid = waitpid(id, nullptr, 0);(void)rid;}}//处理发送来的内容,将数据整合打印到屏幕上void Service(int sockfd, const std::string &clientip, const uint16_t &clientport){char buffer[4096];while (true){ssize_t n = read(sockfd, buffer, sizeof(buffer));if (n > 0){buffer[n] = 0;std::cout << "tcpclient say# " << buffer << std::endl;std::string echo_string = "tcpserver echo# ";echo_string += buffer;write(sockfd, echo_string.c_str(), echo_string.size());}else{break;}}}~TcpServer() {}private:int listensock_;//监听套接字描述符uint16_t port_;//端口号std::string ip_;//ip地址
};

(二)main.cpp

#include "TcpServer.hpp"
#include <iostream>
#include <memory>void Usage(std::string proc)
{std::cout << "\n\rUsage: " << proc << " port[1024+]\n" << std::endl;
}int main(int argc, char *argv[])
{if(argc != 2){Usage(argv[0]);exit(UsageError);}uint16_t port = std::stoi(argv[1]);//智能指针维护服务器std::unique_ptr<TcpServer> tcp_svr(new TcpServer(port));tcp_svr->InitServer();tcp_svr->Start();return 0;
}

2、客户端

TcpClient.cpp

#include <iostream>
#include <cstring>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>void Usage(const std::string &proc)
{std::cout << "\n\rUsage: " << proc << " serverip serverport\n"<< std::endl;
}int main(int argc, char *argv[])
{if (argc != 3){Usage(argv[0]);exit(1);}std::string serverip = argv[1];uint16_t serverport = std::stoi(argv[2]);//创建TCP套接字描述符int sockfd = socket(AF_INET, SOCK_STREAM, 0);if (sockfd < 0){std::cerr << "socket error" << std::endl;return 1;}//还是老套路,初始化服务器地址结构体结构体struct sockaddr_in server;memset(&server, 0, sizeof(server));server.sin_family = AF_INET;server.sin_port = htons(serverport);inet_pton(AF_INET, serverip.c_str(), &(server.sin_addr));//连接服务器int n = connect(sockfd, (struct sockaddr*)&server, sizeof(server));if(n < 0){std::cerr << "connet error" << std::endl;return 2;}//循环输出打印std::string message;while (true){std::cout << "Please Enter# ";std::getline(std::cin, message);write(sockfd, message.c_str(), message.size());char inbuffer[4096];n = read(sockfd, inbuffer, sizeof(inbuffer));if (n > 0){inbuffer[n] = 0;std::cout << inbuffer << std::endl;}  }close(sockfd);return 0;
}

在这里插入图片描述

二、服务器Start函数

1、多进程版

//......pid_t id = fork();if (id == 0){// 子进程关闭监听close(listensock_);//子进程创建“孙子”进程if (fork() > 0)exit(0);Service(sockfd, clientip, clientport); close(sockfd);exit(0);}//父进程关闭sockfd描述符close(sockfd);// 父进程等待pid_t rid = waitpid(id, nullptr, 0);(void)rid;
//......

创建子进程后子进程关闭监听描述符,再创建一个“孙子”进程,然后子进程退出,此时孙子进程成为孤儿进程,被系统领养,再进行其他的工作,工作完成后关闭描述符,退出时由系统回收
父进程关闭新创建的描述符,然后父进程进入进程等待,这个进程等待的时间很短甚至没有,因为子进程在创建完“孙子”进程后就退出了,父进程就可以回收掉子进程继续下一轮的循环

整个过程不会担心父进程由于阻塞等待而造成的一系列问题,也不用修改为非阻塞轮询来消耗资源,被领养的孙子进程有系统回收资源,也不用担心它资源泄露
在这里插入图片描述

2、多线程版

//......
//声明
class TcpServer;class ThreadData
{
public:ThreadData(int fd, const std::string &ip, const uint16_t &p, TcpServer *t) : sockfd(fd), clientip(ip), clientport(p), tsvr_(t){}public:int sockfd;				//套接字描述符std::string clientip;	//ip地址uint16_t clientport;	//端口号TcpServer *tsvr_;		//指向TcpServer的指针
};
//......
//TcpServer结构体内static void *Routine(void *args){	//将线程分离,结束后自动释放所占资源pthread_detach(pthread_self());//调用Service函数ThreadData *td = static_cast<ThreadData *>(args);td->tsvr->Service(td->sockfd, td->clientip, td->clientport);delete td;return nullptr;}
//......

一样的效果
在这里插入图片描述


今日分享就到这了~

在这里插入图片描述

http://www.xdnf.cn/news/400303.html

相关文章:

  • mac M2下虚拟机CentOS 8 安装上安装 Berkeley DB
  • tar -zxvf jdk-8u212-linux-x64.tar.gz -C /opt/module/这个代码的解释
  • 2.3 定积分
  • [Spring]-组件注入
  • 【C++重载操作符与转换】构造函数和复制控制
  • 嵌入式培训之数据结构学习(一)数据结构的基础概念、线性表
  • 【AXI总线专题】AXI-FULL-Master
  • 具身-机器人-分层框架-大脑模块-RoboBrain1.0 RoboOS
  • AI大模型学习二十、利用Dify+deepseekR1 使用知识库搭建初中英语学习智能客服机器人
  • IBM BAW(原BPM升级版)使用教程第十二讲
  • ACL访问控制列表:access-list 10 permit 192.168.10.1
  • Matlab 模糊pid的液压舵机伺服系统
  • 重构金融数智化产业版图:中电金信“链主”之道
  • 【AI提示词】波特五力模型专家
  • Jenkins:库博静态工具CI/CD 的卓越之选
  • Pytorch常用统计和矩阵运算
  • 榜单按行显示
  • IP代理技术原理深度解析:从基础架构到应用实践
  • Open CASCADE学习|由大量Edge构建闭合Wire:有序与无序处理的完整解析
  • Matlab 基于GUI的汽车巡航模糊pid控制
  • STM32 变量存储
  • 深入浅出 iOS 对象模型:isa 指针 与 Swift Metadata
  • 生成对抗网络(GAN)深度解析:理论、技术与应用全景
  • SEGGER Embedded Studio 如何将其中的一个c文件 打包成静态库?
  • 最大m子段和
  • WebGL图形编程实战【6】:性能优化 × 调试工具与技巧精讲
  • (done) 补充:xv6 的一个用户程序 init 是怎么启动的 ?它如何启动第一个 bash ?
  • 模块化PCB设计中联排半孔的应用
  • 接口出现 请求参数格式错误 的解决方法
  • 使用 Navicat 将 Excel 导入数据库