Linux下套接字TCP实现网络通信

Linux下套接字TCP实现网络通信

文章目录

  • Linux下套接字TCP实现网络通信
  • 1.引言
  • 2.具体实现
    • 2.1接口介绍
      • 1.socket()
      • 2.bind()
      • 3.listen()
      • 4.accept()
      • 5.connect()
    • 2.2 服务器端server.hpp
    • 2.3服务端server.cc
    • 2.4客户端client.cc

1.引言

套接字(Socket)是计算机网络中实现网络通信的一种编程接口。它提供了应用程序与网络通信之间的一座桥梁,因为它允许应用程序通过网络发送和接收相应的数据以实现不同主机之间的通信。
在这里插入图片描述

通常套接字由以下两部分组成:

1.网络IP和端口号:IP用来标识主机,而端口号可以标识到单台主机的唯一进程。

2.通信协议:套接字通过规定通信协议来制定数据传输和发送的规则。常见的有TCP和UDP等协议。

TCP是一种面向连接的协议,提供可靠的、有序的、基于字节流的数据传输。

UDP是一种无连接的协议,提供不可靠的、无序的、基于数据报的数据传输。

​ 今天我们来学习TCP实现网络通信。TCP由于能提供可靠、基于字节流的数据传输,使用率与使用场景也比UDP多很多。

我们来看看能实现出什么样的结果(聊天室模拟两个用户随机通信):

若不开启服务端就只开启客户端的话,那么就会像打游戏的某些情况连不上:
在这里插入图片描述

当服务端和客户端都开启后就可以正常通信了:

在这里插入图片描述

这里我们还是通过客户端给服务器端发送消息,通过TCP链接实现通信。

那么事不宜迟,我们马上开始分享实现过程吧!

2.具体实现

2.1接口介绍

1.socket()

​ socket函数是用于创建套接字的函数,创建成功返回文件描述符fd,失败返回-1

int socket(int domain, int type, int protocol);

​ 参数说明:

  • domain

    :指定套接字的地址族(Address Family)

    今天我们选择:

    • AF_INET:IPv4 地址族
  • type

    :指定套接字的类型(Socket Type)

    今天我们选择:

    • SOCK_STREAM:有连接的字节流套接字,用于TCP协议
  • protocol

    :可选参数,指定具体的传输协议。常用的有:

    ​ 今天我们选择:

    • 0:自动选择合适的协议

2.bind()

​ 在Linux下,bind() 函数用于将一个套接字(socket)与特定的IP地址和端口号进行绑定

*int bind(int sockfd, const struct sockaddr addr,socklen_t addrlen);

参数说明:

  • sockfd:要进行绑定的套接字的文件描述符。
  • addr:指向一个 struct sockaddr 结构体的指针,其中包含要绑定的IP地址和端口号信息。
  • addrlenaddr 结构体的长度。

在绑定bind的第二个参数中,我们也需要用到库中定义好的sockaddr_in结构体来初始化!

具体结构体struct sockaddr_in说明:

结构体中有三个值也需要初始化指定一下:

  1. sin_family:表示地址族(Address Family),一般为 AF_INET

  2. sin_port:表示端口号。它是一个 16 位的整数,使用网络字节序(大端字节序)表示。在使用时,通常需要使用 htons() 函数将主机字节序转换为网络字节序。

  3. sin_addr:表示 IPv4 地址。它是一个 struct in_addr 类型的结构体,用于存储 32 位的 IPv4 地址。

    一般服务端用INADDR_ANY,让udp_server在启动时候可以绑定任何ip.

    ​ 客户端用inet_addr函数将字符串转化成32位无符号整数


3.listen()

listen()函数:将套接字设置为监听状态,等待连接请求。

参数:

  • sockfd:套接字的文件描述符。这里我们选择前面socket创建好的返回值.
  • backlog:指定等待连接队列的最大长度。一般不会太大,我们这里写32即可。

4.accept()

accept() 函数:接受客户端的连接请求,创建一个新的套接字用于与客户端进行通信。

参数:

  • sockfd:套接字的文件描述符。这里我们选择前面socket创建好的返回值.
  • addr:指向客户端地址的结构体指针。创建一个sockaddr的结构体强转一下(struct sockaddr*)即可。
  • addrlen:客户端地址结构体的字节大小。创建一个socklen_t类型的值用来计算结构体大小。

5.connect()

connect() 函数:发起与远程主机建立TCP连接的请求。

参数:

  • sockfd:套接字的文件描述符。这里我们选择前面socket创建好的返回值.
  • addr:指向远程主机地址的结构体指针。创建一个sockaddr的结构体强转一下(struct sockaddr*)即可。
  • addrlen:远程主机地址结构体的字节大小。创建一个socklen_t类型的值用来计算结构体大小。

2.2 服务器端server.hpp

在整个服务器端server.hpp中,我们需要创建tcpServer类,并在类中建立这些成员:监听套接字、端口号等.

在类中我们还需要初始化服务器InitServer()和启动服务器Start()两个接口。

服务器端具体实现思路是:我们创建套接字socket()后开始绑定bind(),之后监听listen(),监听成功后我们获取链接accept()即可。

思路简单,但是实现还需要很多事完成:

static const uint16_t defaultport = 8081;
static const int backlog = 32;
using func_t = std::function<std::string(const std::string&)>;

class tcpServer
{
    public:
    tcpServer(func_t func,uint16_t port = defaultport)
    :_func(func)
    ,_port(port)
    ,_quit(true)
     {}
    ~tcpServer() {}
    void InitServer()
    {
        //1.创建套接字
        _listensock = socket(AF_INET,SOCK_STREAM,0);
        if(_listensock < 0)
        {
            std::cerr << "create socket error" << std::endl;
            exit(-1);
        }

        //2.绑定
        struct sockaddr_in local;
        memset(&local,0,sizeof(local)); //清空结构体
        local.sin_family = AF_INET;
        local.sin_port = htons(_port);    //主机转网络
        local.sin_addr.s_addr = htonl(INADDR_ANY);
        if(bind(_listensock,(struct sockaddr*)&local,sizeof(local)) < 0)
        {
            std::cerr << "bind error" << std::endl;
            exit(-2);
        }

        //3.监听(tcp)
        if(listen(_listensock,backlog) < 0)
        {
            std::cerr <<" listen error" << std::endl;
            exit(-3);
        }


    }

    void Start()
    {
        _quit = false; //运行时设置位运行状态,即不退出的状态
        while(!_quit) //服务器死循环
        {
            struct sockaddr_in client;
            socklen_t len = sizeof(client);
            //4.获取链接accept
            int sock = accept(_listensock,(struct sockaddr*)&client,&len);
            if(sock < 0) {
                std::cerr <<"accept error " <<std::endl;
                continue;} //揽客的sock失败后继续即可

            //5.获取链接成功
            std::cout<< "获取链接成功" << sock << " from " << _listensock <<  std::endl;
            service(sock);
        }
    }

    void service(int sock) //服务
    {
        char buffer[1024];
        while(true)
        {
            ssize_t s = read(sock,buffer,sizeof(buffer)-1);
            if(s > 0) //代表成功读取
            {
                buffer[s] = 0;
                std::string res = _func(buffer); //回调显示
                std::cout<< res <<std::endl;

                write(sock,res.c_str(),res.size());

            }
            else if(s == 0) //代表读到文件结尾 在网络中就相当于对方关闭链接
            {
                close(sock);
                std::cout << "quit" <<std::endl;
                break;
                
            }
            else //文件读取失败
            {
                close(sock);
                std::cerr << " read error" <<std::endl;
                break;
            }


        }
    }
    private:
    uint16_t _port; //端口号
    int _listensock; //监听套接字
    bool _quit; //代表服务器没有运行的状态
    func_t _func; //回调包装器,为了后面输出后回显
};

2.3服务端server.cc

在服务端的主文件中,我们直接包含上面的头文件。

我们期望的用法是:./tcp_server port,代表运行可执行文件后面需要带一个参数:端口号

所以我们能够从用户中输入的port,通过main函数中的**char* argv[]**参数列表中获取到。并传给tcpSercer类中初始化与启动服务器即可。

#include "server.hpp"
#include<memory>
using namespace std;


static void usage(string proc) //使用手册,代表运行可执行文件后面需要带一个参数:端口号
{
    std::cout << "Usage:\n\t" << proc << "port\n" <<std::endl;


}

std::string echo(const std::string& message)//输出回显
{
    return message;
}

//期望用法:./tcp_server port
int main(int argc,char* argv[])
{
    if(argc != 2) //输入的不是两个参数,说明你不会用。输出使用手册
    {
        usage(argv[0]);
        exit(-1);
    }
   uint16_t port = atoi(argv[1]); //强转成能够使用的类型
   unique_ptr<tcpServer> ts(new tcpServer(echo,port));//采用智能指针创建释放资源
   ts->InitServer();
   ts->Start();
    return 0;
}

2.4客户端client.cc

在客户端中我们conncet尝试链接到服务器端,这里需要做一个重连反馈:正在尝试重连…

我们期望运行格式:./client serverip serverport,代表运行可执行文件后需要两个参数:IP和端口

我们从用户输入的两个参数中传给main,并通过main参数char* argv[]参数列表获取到,之后获取到直接转化即可。


static void usage(string proc)
{
    std::cout << "Usage:\n\t" << proc << "serverip serverport\n" <<std::endl;


}
//期望使用:./client serverip serverport 
int main(int argc,char* argv[])
{
    if(argc != 3)
    {
        usage(argv[0]);
        exit(-2);
    }
    string serverip = argv[1]; //获取到参数
    uint16_t port = atoi(argv[2]);

    //1.创捷套接字

    int sock = socket(AF_INET,SOCK_STREAM,0);
    if(sock < 0)
    {
        std::cerr << " socket error" <<std::endl;
        exit(-1);
    }

    //2.客户端需要链接服务器 --connect
    struct sockaddr_in server; //
    memset(&server,0,sizeof(server)); //清空结构体
    server.sin_family = AF_INET;//初始化结构体
    server.sin_port = htons(port);
    //server.sin_addr.s_addr = inet_addr(serverip.c_str()); //客户端
    inet_aton(serverip.c_str(),&(server.sin_addr));

    int cnt = 5;
    while(connect(sock,(struct sockaddr*)&server,sizeof(server)) != 0) //如果绑定失败
    {
        sleep(1);
        std::cout<<"正在尝试重连... 重连次数:" <<cnt-- <<std::endl;
        if(cnt <= 0) break;
    }    
    if(cnt <= 0)
    {
        cerr<< "服务器连接失败"<<endl;
        exit(-1);
    }
    //3.连接成功
    while(true) //连接成功后从客户端直接输入发送数据
    {
        string line;
        char buffer[1024];
        cout<<"Enter>> "; 
        getline(cin,line);
        write(sock,line.c_str(),line.size()); //给缓冲区写数据
        ssize_t s = read(sock,buffer,sizeof(buffer) -1);
        if(s > 0)//正常写
        {
            buffer[s] = 0;
            cout<< " server rcho >>>" <<buffer <<endl;
        }
        else if(s == 0) //写结束
        {
            cerr << "server quit" <<endl;
            break;
        }
        else{ //异常
            cerr<< " read error " <<endl;
            break;
        }
    }
    close(sock);//关闭套接字,管不管都可以

    return 0;
}

最后运行之后就能获得我们之前通信的结果了:

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

ATA-2161高压放大器的电子实验案例(案例合集)

ATA-2161是一款理想的可放大交直流信号的单通道高压放大器。最大差分输出1600Vp-p(800Vp)高压&#xff0c;可以驱动高压型负载。凭借其优异的指标参数受到不少电子工程师的喜欢&#xff0c;其在电子实验中的应用也非常频繁&#xff0c;下面为大家整理出ATA-2161高压放大器的应用…

Android全面屏下,默认不会全屏显示,屏幕底部会留黑问题

前些天发现了一个蛮有意思的人工智能学习网站,8个字形容一下"通俗易懂&#xff0c;风趣幽默"&#xff0c;感觉非常有意思,忍不住分享一下给大家。 &#x1f449;点击跳转到教程 公司以前的老项目&#xff0c;便出现了这种情况&#xff0c;网上搜索了各种资料&#xf…

LVS集群 (四十四)

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 目录 前言 一、集群概述 1. 负载均衡技术类型 2. 负载均衡实现方式 二、LVS结构 三、LVS工作模式 四、LVS负载均衡算法 1. 静态负载均衡 2. 动态负载均衡 五、ipvsadm命令详…

docker高级(redis集群三主三从)

1. 新建6个docker容器redis实例 docker run -d --name redis-node-1 --net host --privilegedtrue -v /redis/share/redis-node-1:/data redis:6.0.8 --cluster-enabled yes --appendonly yes --port 6381docker run -d --name redis-node-2 --net host --privilegedtrue -v /…

Matlab图像处理-平移运算

几何运算 几何运算又称为几何变换&#xff0c;是将一幅图像中的坐标映射到另外一幅图像中的新坐标位置&#xff0c;它不改变图像的像素值&#xff0c;只是改变像素所在的几何位置&#xff0c;使原始图像按照需要产生位置、形状和大小的变化。 图像几何运算的一般定义为&#…

STM32使用PID调速

STM32使用PID调速 PID原理 PID算法是一种闭环控制系统中常用的算法&#xff0c;它结合了比例&#xff08;P&#xff09;、积分&#xff08;I&#xff09;和微分&#xff08;D&#xff09;三个环节&#xff0c;以实现对系统的控制。它的目的是使 控制系统的输出值尽可能接近预…

JVM——内存模型

1.java内存模型 1.1 原子性 1.2 问题分析 这里与局部变量自增不同&#xff0c;局部变量调用iinc是在局部变量表槽位上进行自增。 静态变量是在操作数栈自增。 这里的主内存和工作内存时再JMM里的说法。 因为操作系统是时间片切换的多个线程轮流使用CPU. 1.3解决方法 JMM中…

从项目中突显技能:在面试中讲述你的编程故事

&#x1f337;&#x1f341; 博主猫头虎 带您 Go to New World.✨&#x1f341; &#x1f984; 博客首页——猫头虎的博客&#x1f390; &#x1f433;《面试题大全专栏》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33a; &a…

hadoop 学习:mapreduce 入门案例一:WordCount 统计一个文本中单词的个数

一 需求 这个案例的需求很简单 现在这里有一个文本wordcount.txt&#xff0c;内容如下 现要求你使用 mapreduce 框架统计每个单词的出现个数 这样一个案例虽然简单但可以让新学习大数据的同学熟悉 mapreduce 框架 二 准备工作 &#xff08;1&#xff09;创建一个 maven 工…

24个非常实用的Python小技巧

嗨喽&#xff0c;大家好呀~这里是爱看美女的茜茜呐 1.唯一性 以下方法可以检查给定列表是否有重复的地方&#xff0c;可用set&#xff08;&#xff09;的属性将其从列表中删除。 x [1,1,2,2,3,2,3,4,5,6] y [1,2,3,4,5] len(x) len(set(x)) # False len(y) len(set(y)) # Tr…

基于LOF算法的异常值检测

目录 LOF算法简介Sklearn官网LOF算法应用实例1Sklearn官网LOF算法应用实例2基于LOF算法鸢尾花数据集异常值检测读取数据构造数据可视化&#xff0c;画出可疑异常点LOF算法 LOF算法简介 LOF异常检测算法是一种基于密度的异常检测算法&#xff0c;基于密度的异常检测算法主要思想…

Vue项目中app.js过大,导致web初始化加载过慢问题

1、删除多余不需要的库&#xff1a; npm uninstall xxx 如例如moment库文件是很大的可以直接放到index.html文件直接CDN引入 2、修改/config/index.js配置文件&#xff1a;将productionGzip设置为false ​ 3、设置vue-router懒加载 懒加载配置&#xff1a; ​ 非懒加载配置&…

基于PIC单片机篮球计分计时器

一、系统方案 本设计采用PIC单片机作为主控制器&#xff0c;矩阵键盘控制&#xff0c;比分&#xff0c;计时控制&#xff0c;24秒&#xff0c;液晶12864显示。 二、硬件设计 原理图如下&#xff1a; 三、单片机软件设计 1、首先是系统初始化 2、液晶显示程序 /*************…

【JSDocvscode】使用JSDoc、在vscode中开启node调试、使用vscode编写运行Python程序

JSDoc JSDoc是JavaScript的一种注释语法&#xff0c;同时通过JSDoc注释也可以规避js弱类型中不进行代码提示的问题 图形展示JSDoc的效果&#xff1a; 上述没有进行JSDoc&#xff0c;然后我们a点什么 是没有任何提示的 上述就是加上 JSDoc的效果 常用的 vscode 其实内置了 js…

使用apifox前置数据base64编码并添加一个字段

具体前置脚本如下&#xff1a; // pm.request.body.update 处理 body 参数里的变量 let bodyStr pm.request.body.raw; // base64 编码数据 let bodyEncode btoa(bodyStr); console.log(bodyEncode) let newBody {"data": bodyEncode,"sendTime": &qu…

linux 设置与命令基础(二)

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 目录 前言 一、系统基本操作 二、命令类型 三、命令语法 四、命令补齐 五、命令帮助 六、系统基本操作命令 总结 前言 这是本人学习Linux的第二天&#xff0c;今天主…

汽车电子笔记之:基于AUTOSAR的电机控制器架构设计

目录 1、概述 2、AUTOSAR设计 2.1、SWC设计 2.2、PORT设计 2.3、Runnable设计 2.4、电机控制器OS实现 1、概述 电机控制器应用层的软件架构较为复杂,主要包括PMSM(Permanent-MagnetSynchronous Motor)的矢量控制算法。根据PMSM的控制算法,对算法中的软件功能进行分析&…

视频集中存储/云存储平台EasyCVR国标GB28181协议接入的报文交互数据包分析

安防视频监控/视频集中存储/云存储/磁盘阵列EasyCVR平台可拓展性强、视频能力灵活、部署轻快&#xff0c;可支持的主流标准协议有国标GB28181、RTSP/Onvif、RTMP等&#xff0c;以及支持厂家私有协议与SDK接入&#xff0c;包括海康Ehome、海大宇等设备的SDK等。视频汇聚融合管理…

cortex-A7核LED灯实验--STM32MP157

实验目的&#xff1a;实现LED1 / LED2 / LED3三盏灯工作 一&#xff0c;分析电路图 1&#xff0c;思路 分析电路图可知&#xff1a; 网络编号 引脚编号 LED1 PE10 LED2 > PF10 LED3 > PE8 2&#xff0c;工作原理&#xff1a; 写1&#xff1a;LED灯亮&#xf…

前端:html实现页面切换、顶部标签栏,类似于浏览器的顶部标签栏(完整版)

效果 代码 <!DOCTYPE html> <html><head><style>/* 左侧超链接列表 */.link {display: block;padding: 8px;background-color: #f2f2f2;cursor: pointer;}/* 顶部标签栏 */#tabsContainer {width:98%;display: flex;align-items: center;overflow-x: …
最新文章