套接字实现TCP
套接字
套接字的意义就是客户端与服务器进行双向通信的端点,如果有不理解点上面套接字三字更近距离了解套接字。
网络套接字与客户连接的特定网络有关的服务端口号,这个端口号允许linux进入特定的端口号的连接转到正确的服务器进程。
套接字通信的建立过程:
- 服务器端
1.使用socket创建一个套接字。它是OS分配给服务器进程的类似文件描述符的资源。
2.服务器进程用系统调用bind命名套接字。然后服务器开始等待客户端连接到这个命名套接字。
3.服务器通过系统调用accept来接受客户的连接。accept会创建一个不同于命名套接字的新套接字来与这个特定客户进行通信,而命名套接字则被保留下来继续处理其他客户的连接请求。
4.系统获取客户端基本信息,为客户端提供服务。
- 客户端
1.调用socket创建一个未命名套接字。
2.通过远端主机信息,向服务器发起连接。
实现一个客户端与主机端连接,将客户文本进行统一大小写转换。
实现服务器端ServerTcp:
第一步 创建类名字为ServerTcp,在类里创建私有变量套接字与端口和IP。
private:
//套接字
int listenSock_;
//port接口
uint16_t port_;
//ip
std::string ip_;
第二步创建构造函数与析构函数。
ServerTcp(uint16_t port,const std::string &ip=""):port_(port),ip_(ip),listenSock_(-1){ }
~ServerTcp(){ }第三步 创建套接字socket,并判断是否创建成功listenSock_=socket(PF_INET,SOCK_STREAM,0);if(listenSock_<0){logMessage(FATAL,"socket:%s",strerror(errno));//日志消息打印错误码exit(SOCKET_ERR);//套接字错误}logMessage(DEBUG,"socket:%s,%d",strerror(errno),listenSock_);
第四步 进行绑定
4.1填充服务器信息
struct sockaddr_in local;//用户栈memset(&local,0,sizeof local);//初始化全部置为0local.sin_family=PF_INET;//绑定域local.sin_port=htons(port_);//将主机字节顺序转为网络字节顺序ip_.empty()?(local.sin_addr.s_addr=INADDR_ANY):(inet_aton(ip_.c_str(),&local.sin_addr));
4.2将本地socket信息,写入sock_对应的内核区域if(bind(listenSock_,(const struct sockaddr *)&local,sizeof local)<0){logMessage(FATAL,"bind: %s",strerror(errno));exit(BIND_ERR);//绑定错误}logMessage(DEBUG,"bind: %s,%d",strerror(errno),listenSock_);
第五步 监听socket,tcp是面向连接的if(listen(listenSock_,5)<0)//把一个未连接的套接字转换成一个被动套接字{ //指示内核应该接受指向该套接字的连接请求logMessage(FATAL,"listen: %s",strerror(errno));exit(LISTEN_ERR);}logMessage(DEBUG,"listen:%s,%d",strerror(errno),listenSock_);
第六步 创建loop函数完成对客户的服务
先获取连接,得到客户端基本信息提供你的服务,本次服务为小写转为大写服务。
void loop(){while (true){struct sockaddr_in peer;socklen_t len=sizeof(peer);//4、获取连接,accept的返回值是一个新的socket fd??int serviceSock=accept(listenSock_,(struct sockaddr *)&peer,&len);//accept() 告诉它你有空闲的连接。它将返回一个新的套接字文 件描述符!这样你就有两个套接字了,原来的一个还在侦听你的那个端口, 新的在准备发送 (send())//和接收 ( recv()) 数据。这就是这个过程! if(serviceSock<0){//获取链接失败logMessage(WARINING,"accept: %s[%d]",strerror(errno),serviceSock);continue;//继续链接}//4.1获取客户端基本信息uint16_t peerPort=ntohs(peer.sin_port);//网络字节顺序转主机字节顺序std::string peerIp=inet_ntoa(peer.sin_addr);//返回10位的IP地址logMessage(DEBUG,"accept: %s | %s[%d], sock fd:%d",strerror(errno),peerIp.c_str(),peerPort,serviceSock);//5 提供服务---echo 小写转大写//5.0多线程版本ThreadData *td=new ThreadData(peerPort,peerIp,serviceSock,this);//每个客户连接过来换创建独立的空间,和ip。实现多进程共享服务pthread_t tid;pthread_create(&tid,nullptr,threadRoutine,(void *)td);} }
这里面的create对应的方法。方法里面又存放有小写转大写的代码。
class ThreadData
{
public:uint16_t clientPort_;std::string clientIp_;int sock_;ServerTcp *this_;
public:ThreadData(uint16_t port,std::string ip,int sock,ServerTcp *ts):clientPort_(port),clientIp_(ip),sock_(sock),this_(ts){}
};
static void *threadRoutine(void *args){pthread_detach(pthread_self());//线程分离自动释放资源ThreadData *td=static_cast<ThreadData *>(args);//强制类型转换td->this_->transService(td->sock_,td->clientIp_,td->clientPort_);delete td;return nullptr;}//大小写转换服务//TCP && UDP :支持全双工void transService(int sock,const std::string &clientIp,uint16_t clinentPort){assert(sock>=0);assert(!clientIp.empty());assert(clinentPort>=1024);char inbuffer[BUFFER_SIZE];while (true){ssize_t s=read(sock,inbuffer,sizeof(inbuffer)-1);if(s>0){inbuffer[s]='\0';if(strcasecmp(inbuffer,"quit")==0){logMessage(DEBUG,"client quit -- %s[%d]",clientIp.c_str(),clinentPort);break;}logMessage(DEBUG,"trans before: %s[%d]>>> %s",clientIp.c_str(),clinentPort,inbuffer);//可以进行大小写转化了for(int i=0;i<s;i++){if(isalpha(inbuffer[i]) && islower(inbuffer[i]))//判断是否是字母检查字符是否是小写inbuffer[i] =toupper(inbuffer[i]);}logMessage(DEBUG,"trans after: %s[%d]>>> %s",clientIp.c_str(),clinentPort,inbuffer);write(sock,inbuffer,strlen(inbuffer));}else if(s==0){//s==0表示对方关闭,client退出logMessage(DEBUG,"client quit -- %s[%d]",clientIp.c_str(),clinentPort);break;} else{logMessage(DEBUG,"%s[%d] --read: %s",clientIp.c_str(),clinentPort,strerror(errno));break;}}//client退出,服务到此结束close(sock);logMessage(DEBUG,"server close %d done",sock);}
static void Usage(std::string proc)
{std::cerr<<"Usage:\n\t"<<proc<<" Port ip"<<std::endl;std::cerr<<"Example:\n\t"<<proc<<" 127.0.0.1 8080\n"<<std::endl;
}
//实现的时候创建ServerTcp调用init和loop服务器端书写完毕。
实现客户端clientTcp
volatile bool quit=false;//客户端口
//当输入prog para_1,有一个参数,则由操作系统传来的参数为:
// argc=2,表示除了程序名外还有一个参数。
// argv[0]指向输入的程序路径及名称。
// argv[1]指向参数para_1字符串。// 当输入prog para_1 para_2 有2个参数,则由操作系统传来的参数为:
// argc=3,表示除了程序名外还有2个参数。
// argv[0]指向输入的程序路径及名称。
// argv[1]指向参数para_1字符串。
// argv[2]指向参数para_2字符串。static void Usage(std::string proc)
{std::cerr<<"Usage:\n\t"<<proc<<" serverIp serverPort"<<std::endl;std::cerr<<"Example:\n\t"<<proc<<" 127.0.0.1 8080\n"<<std::endl;
}int main(int argc,char *argv[])
{if(argc!=3){Usage(argv[0]);exit(USAGE_ERR);}std::string serverIp=argv[1];uint16_t serverPort=atoi(argv[2]);
//1、创建套接字int sock=socket(AF_INET,SOCK_STREAM,0);if(sock<0){std::cerr<<"socket: "<<strerror(errno)<<std::endl;exit(SOCKET_ERR);//如果出错退出数据}
//2、向服务器发起链接
//2.1发起前的准备工作就是连接远端主机的基本信息struct sockaddr_in server;//在客户机中,我们首先创建了一个SCTP套接字,//然后创建了一个sockaddr结构,其中包含了将要连接的端点。memset(&server,0,sizeof(server));//空间初始化为0server.sin_family=AF_INET;//域(定义哪种地址族)说白了就是IP协议server.sin_port=htons(serverPort);//端口号(主机字节顺序转换成网络字节顺序)//就是返回IP对应的网络字节序表示的无符号整数.//网络字节序n转化为点分十进制的IP,例如服务器IP地址inet_aton(serverIp.c_str(),&server.sin_addr);//将主机地址转化为二进制数存储在INP结构中
//2.2发起请求,connect会自动帮我们进行bind//建立网络连接必须使用connectif(connect(sock,(const struct sockaddr *)&server,sizeof(server))!=0){std::cerr<<"connect: "<<strerror(errno)<<std::endl;exit(CONN_ERR);//通讯失败(连接服务器失败)}std::cout<<"info : connect success: "<<sock<<std::endl;std::string message;while (!quit){message.clear();//清理std::cout<<"请输入你的消息>>>";std::getline(std::cin,message);if(strcasecmp(message.c_str(),"quit")==0)//将字符全部转换成小写{quit=true;}ssize_t s=write(sock,message.c_str(),message.size());//写入sockif(s>0){message.resize(1024);ssize_t s=read(sock,(char *)(message.c_str()),1024);if(s>0){message[s]=0;}std::cout<<"Server Echo>>> "<<message<<std::endl;}else if(s <= 0){break;}}close(sock);return 0;
}
相关文章:

套接字实现TCP
套接字 套接字的意义就是客户端与服务器进行双向通信的端点,如果有不理解点上面套接字三字更近距离了解套接字。 网络套接字与客户连接的特定网络有关的服务端口号,这个端口号允许linux进入特定的端口号的连接转到正确的服务器进程。 套接字通信的建立过…...

MMSeg绘制模型指定层的Heatmap热力图
文章首发及后续更新:https://mwhls.top/4475.html,无图/无目录/格式错误/更多相关请至首发页查看。 新的更新内容请到mwhls.top查看。 欢迎提出任何疑问及批评,非常感谢! 摘要:绘制模型指定层的热力图 可视化环境安装 …...

关于Paul C.R. - Inductance_ Loop and partial-Wiley (2009)一书的概括
环感抗和部分感抗是两种不同的电路元件,它们通常用于描述不同类型的导体结构中的电流承载能力。 环感抗通常用于描述绕制在磁性芯上的线圈。当电流通过线圈时,它会在磁性芯中产生一个磁场,这个磁场又会对线圈产生一个磁通量。这个磁通量的大…...

基于支持向量机SVM的面部表情分类预测
目录 支持向量机SVM的详细原理 SVM的定义 SVM理论 Libsvm工具箱详解 简介 参数说明 易错及常见问题 SVM应用实例,基于SVM的面部表情分类预测 代码 结果分析 展望 支持向量机SVM的详细原理 SVM的定义 支持向量机(support vector machines, SVM)是一种二分类模型,它的基本…...

java内存模型的理解
java内存模型的理解并发问题产生的源头缓存导致的可见性问题线程切换导致的原子性问题编译优化带来的有序性问题小结Java内存模型: 解决可见性和有序性问题Java内存模型与JVM内存模型的区别volatile关键字Happens-Before规则小结思考题参考并发问题产生的源头 缓存导致的可见性…...

自己写一个简单的IOC
什么是SpringIOC? 答:IOC即控制反转,就是我们不在手动的去new一个对象,而是将创建对象的权力交给Spring去管理,我们想要一个User类型的对象,就只需要定义一个User类型的变量user1,然后让Spring去…...

用Python批量重命名文件
案例 今天,我们来整理文件夹中的文件,给某个文件夹下的所有文件重新命名。要求是给所有文件按照修改时间,按顺序在文件名前面加上编号。比如将文件资料.xlsx重命名为1. 资料.xlsx import osdef Get_modify_time(file):return os.path.getmtime(file) #获取文件修改时间path…...

iis之web服务器搭建、部署(详细)~千锋
目录 Web服务器 部署web服务器 实验一 发布一个静态网站 实验二 一台服务器同时发布多个web站点 网站类型 Web服务器 也叫网页服务或HTTP服务器web服务器使用的协议是HTTPHTTP协议端口号:TCP 80、HTTPS协议端口号:TCP 443Web服务器发布软件&…...
javascript的ajax
学什么Ajax基础JSON跨域XHR对象Ajax进阶Ajax应用Ajax扩展Ajax基础初识 AjaxAjax的基本用法GET请求POST请求JSON初识JSONJSON的3种形式JSON的常用方法跨域初识跨域CORS跨域资源共享JSONPXHR 对象XHR的属性XHR的方法XHR的事件Ajax进阶FormData封装Ajax使用Promise改造封装好的Aja…...
SpringBoot入门 - 开发中还有哪些常用注解
本文主要介绍一些SpringBoot中常用的注解。Spring Boot 常用注解SpringBootApplicationTarget(ElementType.TYPE) Retention(RetentionPolicy.RUNTIME) Documented Inherited Configuration EnableAutoConfiguration ComponentScan public interface SpringBootApplication {/*…...

网络基础(三)
目录 网络层与数据链路层 编辑 网络层 IP协议 IP的构成 TCP和IP各自解决什么问题? 基本概念 协议头格式 协议头分析 如何分离与封装? 如何分用(向上交付)? 16位首部校验和 32位源IP和目的IP 4位版本 8位服务类型(Type Of…...

Go语言函数高级篇
Go语言函数高级篇1.高阶函数函数作为参数函数作为返回值2.匿名函数3.defer4.内置函数1.高阶函数 高阶函数分为函数作为参数和函数作为返回值两部分。 函数作为参数 函数可以作为参数: package mainimport "fmt"func add(x, y int) int {return x y }…...

ubuntu16.04 python代码自启动和可执行文件自启动
1 python代码自启动 参考 https://blog.csdn.net/qq_38288618/article/details/104096606 准备好python文件 test.py import time c1 while 1:time.sleep(1)cc1print(c)运行 sudo chmod 777 test.py python3 test.py准备run.sh 文件 #!/bin/bash gnome-terminal -x bash -…...

应用层协议 HTTP HTTPS
目录 应用层 再谈 "协议" 序列化和反序列化 关于 json库 request序列化 request反序列化 response序列化 response反序列化 PS:命令宏 HTTP协议 认识URL urlencode和urldecode HTTP协议格式 HTTP请求 HTTP响应 请求方法 编辑 HT…...

图神经网络 pytorch GCN torch_geometric KarateClub 数据集
图神经网络 安装Pyg 首先安装torch_geometric需要安装pytorch然后查看一下自己电脑Pytorch的版本 import torch print(torch.__version__) #1.12.0cu113然后进入官网文档网站 链接: https://pytorch-geometric.readthedocs.io/en/latest/install/installation.html 安装自己…...

【博学谷学习记录】超强总结,用心分享丨人工智能 自然语言处理 文本特征处理小结
目录文本特征处理作用常见的文本特征处理方法添加n-gram特征说明提取n-gram文本长度规范说明实现导包问题记录心得文本特征处理作用 文本特征处理包括为语料添加具有普适性的文本特征, 如:n-gram特征 以及对加入特征之后的文本语料进行必要的处理, 如: 长度规范. 这些特征处…...
2023年中职网络安全竞赛解析——隐藏信息探索
隐藏信息探索 需求环境可私信博主,求个三连! 1.访问服务器的FTP服务,下载图片QR,从图片中获取flag,并将flag提交; 2.访问服务器的FTP服务,下载文件document,从文件中获取flag,并将flag提交; 3.访问服务器的FTP服务,下载图片beach,从图片中获取flag,并将flag提交…...
实用操作--迁移到Spring Boot 3 和 Spring 6 需要关注的JAVA新特性
正如你所了解到的,迁移到Spring Boot 3 或 Spring 6也将迁移到Java 17和Jakarta EE 9。 如果你非常重视可观察性和本机可执行文件,将从Spring Boot 和 Spring 的主要发行版中获益最大。 本文针对迁移到Spring Boot 3 或 Spring 6涉及的变化进行了梳理和总结。 1. java17新特…...
等保检测风险处理方案
文章目录等保检测风险处理方案1. Apache HTTP Server "httpOnly" Cookie信息泄露漏洞(CVE-2012-0053))2. 检测到目标web应用表单存在口令猜测攻击3. X-Content-Type-Options响应头缺失4. X-XSS-Protection响应头缺失5. Content-Security-Policy响应头缺失6. Referrer…...

java 包装类 万字详解(通俗易懂)
前言简介和溯源拆装箱String类和基本类型的相互转化String类和包装类型的相互转化八大包装类的常用方法汇总(含代码演示)一、前言 : 本节内容是我们《API-常用类》专题的最后一节了。本节内容主要讲包装类,内容包括但不限于包装类的诞生&…...

多云管理“拦路虎”:深入解析网络互联、身份同步与成本可视化的技术复杂度
一、引言:多云环境的技术复杂性本质 企业采用多云策略已从技术选型升维至生存刚需。当业务系统分散部署在多个云平台时,基础设施的技术债呈现指数级积累。网络连接、身份认证、成本管理这三大核心挑战相互嵌套:跨云网络构建数据…...
《Playwright:微软的自动化测试工具详解》
Playwright 简介:声明内容来自网络,将内容拼接整理出来的文档 Playwright 是微软开发的自动化测试工具,支持 Chrome、Firefox、Safari 等主流浏览器,提供多语言 API(Python、JavaScript、Java、.NET)。它的特点包括&a…...
Frozen-Flask :将 Flask 应用“冻结”为静态文件
Frozen-Flask 是一个用于将 Flask 应用“冻结”为静态文件的 Python 扩展。它的核心用途是:将一个 Flask Web 应用生成成纯静态 HTML 文件,从而可以部署到静态网站托管服务上,如 GitHub Pages、Netlify 或任何支持静态文件的网站服务器。 &am…...

新能源汽车智慧充电桩管理方案:新能源充电桩散热问题及消防安全监管方案
随着新能源汽车的快速普及,充电桩作为核心配套设施,其安全性与可靠性备受关注。然而,在高温、高负荷运行环境下,充电桩的散热问题与消防安全隐患日益凸显,成为制约行业发展的关键瓶颈。 如何通过智慧化管理手段优化散…...

python执行测试用例,allure报乱码且未成功生成报告
allure执行测试用例时显示乱码:‘allure’ �����ڲ����ⲿ���Ҳ���ǿ�&am…...

初学 pytest 记录
安装 pip install pytest用例可以是函数也可以是类中的方法 def test_func():print()class TestAdd: # def __init__(self): 在 pytest 中不可以使用__init__方法 # self.cc 12345 pytest.mark.api def test_str(self):res add(1, 2)assert res 12def test_int(self):r…...

基于 TAPD 进行项目管理
起因 自己写了个小工具,仓库用的Github。之前在用markdown进行需求管理,现在随着功能的增加,感觉有点难以管理了,所以用TAPD这个工具进行需求、Bug管理。 操作流程 注册 TAPD,需要提供一个企业名新建一个项目&#…...
虚拟电厂发展三大趋势:市场化、技术主导、车网互联
市场化:从政策驱动到多元盈利 政策全面赋能 2025年4月,国家发改委、能源局发布《关于加快推进虚拟电厂发展的指导意见》,首次明确虚拟电厂为“独立市场主体”,提出硬性目标:2027年全国调节能力≥2000万千瓦࿰…...
站群服务器的应用场景都有哪些?
站群服务器主要是为了多个网站的托管和管理所设计的,可以通过集中管理和高效资源的分配,来支持多个独立的网站同时运行,让每一个网站都可以分配到独立的IP地址,避免出现IP关联的风险,用户还可以通过控制面板进行管理功…...

[ACTF2020 新生赛]Include 1(php://filter伪协议)
题目 做法 启动靶机,点进去 点进去 查看URL,有 ?fileflag.php说明存在文件包含,原理是php://filter 协议 当它与包含函数结合时,php://filter流会被当作php文件执行。 用php://filter加编码,能让PHP把文件内容…...