当前位置: 首页 > 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;}
//......

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


今日分享就到这了~

在这里插入图片描述

相关文章:

  • DDD领域驱动开发
  • 付费专栏·Python潮流周刊电子书合集(epub、pdf、markdown)下载
  • 木马查杀引擎—关键流程图
  • vue3搭建实战项目笔记四
  • Linux——数据库备份与恢复
  • ZYNQ笔记(二十一): VDMA HDMI 彩条显示
  • 机器学习第六讲:向量/矩阵 → 数据表格的数学表达,如Excel表格转数字阵列
  • 配置Hadoop集群环境-使用脚本命令实现集群文件同步
  • 皇冠CAD(CrownCAD)建模教程:配电开关
  • React Agent:从零开始构建 AI 智能体|React Flow 实战・智能体开发・低代码平台搭建
  • Docker私有仓库实战:官方registry镜像实战应用
  • -MAC桢-
  • 车联网大数据:从数据到场景的闭环实践
  • 配置文件介绍xml、json
  • 嵌入式软件开发常见warning之 warning: implicit declaration of function
  • 【RabbitMQ】应用问题、仲裁队列(Raft算法)和HAProxy负载均衡
  • 面试题 - Kafka、RabbitMQ、RocketMQ如何选型?
  • 运用数组和矩阵对数据进行存取和运算——NumPy模块 之五
  • 【prometheus+Grafana篇】基于Prometheus+Grafana实现windows操作系统的监控与可视化
  • 解决IDEA无法运行git的问题
  • 演员黄晓明、金世佳进入上海戏剧学院2025年博士研究生复试名单
  • A股高开高走:沪指涨0.82%,创指涨2.63%,超4100股收涨
  • 多省市已开展现房销售试点,去年全国现房销售面积占比超30%
  • 伊朗外长称正与美国进行“善意”的会谈
  • 鄂州:锁死中小学教师编制总量,核减小学编制五百名增至初中
  • 重庆大学通报本科生发14篇SCI论文:涉事学生及其父亲被处理