套接字实现TCP

news2024/11/15 19:56:52

套接字

套接字的意义就是客户端与服务器进行双向通信的端点,如果有不理解点上面套接字三字更近距离了解套接字。
网络套接字与客户连接的特定网络有关的服务端口号,这个端口号允许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);//初始化全部置为0
        local.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));//空间初始化为0
    server.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
    //建立网络连接必须使用connect
    if(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());//写入sock
        if(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;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/389967.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【数据结构与算法】数据结构有哪些?算法有哪些?

1. 算法与数据结构总览图 2.常用的数据结构 2.1.数组&#xff08;Array&#xff09; 数组是一种聚合数据类型&#xff0c;它是将具有相同类型的若干变量有序地组织在一起的集合。数组可以说是最基本的数据结构&#xff0c;在各种编程语言中都有对应。一个数组可以分解为多个数…

k8s篇之Pod 干预与 PDB

文章目录自愿干预和非自愿干预PDBPDB 示例分离集群所有者和应用程序所有者角色如何在集群上执行中断操作自愿干预和非自愿干预 Pod 不会消失&#xff0c;除非有人&#xff08;用户或控制器&#xff09;将其销毁&#xff0c;或者出现了不可避免的硬件或软件系统错误。 我们把这…

Vue+ECharts实现可视化大屏

由于项目需要一个数据大屏页面&#xff0c;所以今天学习了vue结合echarts的图标绘制 首先需要安装ECharts npm install echarts --save因为只是在数据大屏页面绘制图表&#xff0c;所以我们无需把它设置为全局变量。 可以直接在该页面引入echarts&#xff0c;就可以在数据大…

『MyBatis技术内幕』源码调试前提

准备源代码包 下载源代码 3.4.6 版本 https://github.com/mybatis/mybatis-3/releases?page2 通过 idea 导入然后回自动下载所有依赖&#xff0c;根据 3.4.6 版本的 pom.xml 找到依赖的 mybatis-parent 版本 <parent><groupId>org.mybatis</groupId><ar…

《计算机网络:自顶向下方法》学习笔记——第一章:计算机网络和因特网

计网 第一章 计算机网络和因特网 1.1 什么是因特网 回答这个问题有两种方式 其一&#xff0c;我们能够描述因特网的具体构成&#xff0c;即构成因特网的基本硬件和软件组件&#xff1b;其二&#xff0c;我们能够根据为分布式应用提供服务的联网基础设施来描述因特网。 1.1.…

加油站ai视觉识别系统 yolov7

加油站ai视觉识别系统通过yolov7网络模型深度学习&#xff0c;加油站ai视觉识别算法对现场画面中人员打电话抽烟等违规行为&#xff0c;还有现场出现明火烟雾等危险状态。除此之外&#xff0c;模型算法还可以对卸油时灭火器未正确摆放、人员离岗不在现场、卸油过程静电释放时间…

20230304学习笔记

1、Mybatis #{}和${}的区别是什么 a、#{}是预编辑处理、是占位符&#xff0c;${}是字符串拼接符。 b、#{}替换为&#xff1f;号&#xff0c;用PreparedStatement来赋值&#xff0c;${}直接替换变量的值&#xff0c;用Statement赋值。 c、#{}在DBMS中、自动加入单引号&#…

XSS-labs靶场1-13关解法答案

目录 XSS-labs克隆/下载地址: 第一关 解法 第二关 解法 第三关 解法 第四关 解法 第五关 解法 第六关 解法 第七关 解法 第八关 解法 第九关 解法 第十关 解法 第十一关 解法 第十二关 解法 第十三关 解法 从XSS payload 中关于浏览器解码的一些总结 XSS-labs克隆/下载地…

javaDoc生成方式

javaDoc生成方式 命令行生成 在cmd控制台窗口上找到需要生成文件的路径&#xff0c;然后执行命令。 # javadoc -encoding UTF-8 -charset UTF-8 文件名 javadoc -encoding UTF-8 -charset UTF-8 Doc.java这样就生成完成了。 测试Doc.java文件 package com.jiang.base;/***…

grid了解

结构 <div class"grid"><div>1</div><div>2</div><div>3</div><div>4</div><div>5</div><div>6</div><div>7</div><div>8</div><div>9</div>&l…

css中重难点整理

一、vertical-align 在学习vertical-align的时候&#xff0c;可能会很困惑。即使网上有一大推文章讲veitical-align,感觉看完好像懂了&#xff0c;等自己布局的时候用到vertical-align的时候好像对它又很陌生。这就是我在布局的时候遇到的问题。 本来vertical-align就很不好理…

LeetCode分类刷题-----贪心算法

贪心算法贪心455.分发饼干376.摆动序列53.最大子序和122.买卖股票的最佳时机||55.跳跃游戏45.跳跃游戏||1005.K次取反后最大化的数组和134.加油站135.分发糖果860.柠檬水找零406.根据身高重建队列452.用最少数量的箭引爆气球![在这里插入图片描述](https://img-blog.csdnimg.cn…

[算法和数据结构]--回溯算法之DFS初识

回溯算法——DFSDFS介绍(Depth First Search)DFS经典题目1. 员工的重要性2. 图像渲染3.被围绕的区域4.岛屿数量5. 电话号码的字母组合6.数字组合7. 活字印刷8. N皇后DFS介绍(Depth First Search) 回溯法&#xff08;back tracking&#xff09;&#xff08;探索与回溯法&#x…

嵌入式和Python(二):python初识及其基本使用规则

目录 一&#xff0c;python基本特点 二&#xff0c;python使用说明 ● 两种编程方式 ① 交互式编程 ② 脚本式编程 ● python中文编码 ● python行和缩进 ● python引号 ● python空行 ● python等待用户输入 ① 没有转换变量类型 ② 转换变量类型 ● python变…

[Pytorch]DataSet和DataLoader逐句详解

将自己的数据集引入Pytorch是搭建属于自己的神经网络的重要一步&#xff0c;这里我设计了一个简单的实验&#xff0c;结合这个实验代码&#xff0c;我将逐句教会大家如何将数据引入DataLoader。 这里以目标检测为例&#xff0c;一个batch中包含图片文件、先验框的框体坐标、目标…

【C语言进阶:指针的进阶】你真分得清sizeof和strlen?

本章重点内容&#xff1a; 字符指针指针数组数组指针数组传参和指针传参函数指针函数指针数组指向函数指针数组的指针回调函数指针和数组面试题的解析这篇博客 FLASH 将带大家一起来练习一些容易让人凌乱的题目&#xff0c;通过这些题目来进一步加深和巩固对数组&#xff0c;指…

基于注解@Transactional事务基本用法

关于概念性的放在最下面,熟读几遍 在使用时候也没多关注,总是加个Transactional 初识下 一般查询 Transactional(propagation Propagation.SUPPORTS) 增删改 Transactional(propagation Propagation.REQUIRED) 当然不能这么马虎 Spring中关于事务的传播 一个个测试,事…

计算机网络第八版——第一章课后题答案(超详细)

第一章 该答案为博主在网络上整理&#xff0c;排版不易&#xff0c;希望大家多多点赞支持。后续将会持续更新&#xff08;可以给博主点个关注~ 【1-01】计算机网络可以向用户提供哪些服务&#xff1f; 解答&#xff1a;这道题没有现成的标准答案&#xff0c;因为可以从不同的…

操作系统——15.FCFS、SJF、HRRN调度算法

这节我们来看一下进程调度的FCFS、SJF、HRRN调度算法 目录 1.概述 2.先来先服务算法&#xff08;FCFS&#xff0c;First Come First Serve&#xff09; 3.短作业优先算法&#xff08;SJF&#xff0c;Shortest Job First&#xff09; 4.高响应比优先算法&#xff08;HRRN&…

Jackson CVE-2018-5968 反序列化漏洞

0x00 前言 同CVE-2017-15095一样&#xff0c;是CVE-2017-7525黑名单绕过的漏洞&#xff0c;主要还是看一下绕过的调用链利用方式。 可以先看&#xff1a; Jackson 反序列化漏洞原理 或者直接看总结也可以&#xff1a; Jackson总结 影响版本&#xff1a;至2.8.11和2.9.x至…