套接字实现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-常用类》专题的最后一节了。本节内容主要讲包装类,内容包括但不限于包装类的诞生&…...
XML Group端口详解
在XML数据映射过程中,经常需要对数据进行分组聚合操作。例如,当处理包含多个物料明细的XML文件时,可能需要将相同物料号的明细归为一组,或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码,增加了开…...
rknn优化教程(二)
文章目录 1. 前述2. 三方库的封装2.1 xrepo中的库2.2 xrepo之外的库2.2.1 opencv2.2.2 rknnrt2.2.3 spdlog 3. rknn_engine库 1. 前述 OK,开始写第二篇的内容了。这篇博客主要能写一下: 如何给一些三方库按照xmake方式进行封装,供调用如何按…...
.Net框架,除了EF还有很多很多......
文章目录 1. 引言2. Dapper2.1 概述与设计原理2.2 核心功能与代码示例基本查询多映射查询存储过程调用 2.3 性能优化原理2.4 适用场景 3. NHibernate3.1 概述与架构设计3.2 映射配置示例Fluent映射XML映射 3.3 查询示例HQL查询Criteria APILINQ提供程序 3.4 高级特性3.5 适用场…...
理解 MCP 工作流:使用 Ollama 和 LangChain 构建本地 MCP 客户端
🌟 什么是 MCP? 模型控制协议 (MCP) 是一种创新的协议,旨在无缝连接 AI 模型与应用程序。 MCP 是一个开源协议,它标准化了我们的 LLM 应用程序连接所需工具和数据源并与之协作的方式。 可以把它想象成你的 AI 模型 和想要使用它…...
MODBUS TCP转CANopen 技术赋能高效协同作业
在现代工业自动化领域,MODBUS TCP和CANopen两种通讯协议因其稳定性和高效性被广泛应用于各种设备和系统中。而随着科技的不断进步,这两种通讯协议也正在被逐步融合,形成了一种新型的通讯方式——开疆智能MODBUS TCP转CANopen网关KJ-TCPC-CANP…...
Module Federation 和 Native Federation 的比较
前言 Module Federation 是 Webpack 5 引入的微前端架构方案,允许不同独立构建的应用在运行时动态共享模块。 Native Federation 是 Angular 官方基于 Module Federation 理念实现的专为 Angular 优化的微前端方案。 概念解析 Module Federation (模块联邦) Modul…...
高防服务器能够抵御哪些网络攻击呢?
高防服务器作为一种有着高度防御能力的服务器,可以帮助网站应对分布式拒绝服务攻击,有效识别和清理一些恶意的网络流量,为用户提供安全且稳定的网络环境,那么,高防服务器一般都可以抵御哪些网络攻击呢?下面…...
自然语言处理——循环神经网络
自然语言处理——循环神经网络 循环神经网络应用到基于机器学习的自然语言处理任务序列到类别同步的序列到序列模式异步的序列到序列模式 参数学习和长程依赖问题基于门控的循环神经网络门控循环单元(GRU)长短期记忆神经网络(LSTM)…...
css3笔记 (1) 自用
outline: none 用于移除元素获得焦点时默认的轮廓线 broder:0 用于移除边框 font-size:0 用于设置字体不显示 list-style: none 消除<li> 标签默认样式 margin: xx auto 版心居中 width:100% 通栏 vertical-align 作用于行内元素 / 表格单元格ÿ…...
安宝特方案丨船舶智造的“AR+AI+作业标准化管理解决方案”(装配)
船舶制造装配管理现状:装配工作依赖人工经验,装配工人凭借长期实践积累的操作技巧完成零部件组装。企业通常制定了装配作业指导书,但在实际执行中,工人对指导书的理解和遵循程度参差不齐。 船舶装配过程中的挑战与需求 挑战 (1…...
