当前位置: 首页 > article >正文

【Linux网络】TCP_Socket

目录

TCP协议(传输控制协议)

listen状态

accept和connect

TCP_echo_server

(1)创建套接字

(2)绑定

(3)设置listen状态

(4)loop

(5)客户端

多线程远程执行命令


TCP协议(传输控制协议)

传输层协议
有连接
可靠传输(前提网络要联通,复杂,维护性要更强)
面向字节流

tcp和udp在创建套接字和绑定部分是一样的;

不同的是,tcp在服务端绑定后需要将套接字的状态设置为listen状态,除此之外,在通信前要获取新连接;在客户端也要connect;

因为tcp是面向连接的,tcp需要未来不断地能够做到获取连接;

listen状态

accept和connect


  

 


 在TCP(传输控制协议)的服务端中获取新连接是至关重要的,原因如下:

  •  建立连接

TCP是面向连接的协议。客户端发起连接请求(通过 connect 函数),服务端需要接受这些请求(通过 accept 函数)来建立连接。这个连接是双向的通信链路,允许数据在客户端和服务端之间可靠地传输。例如,当你访问一个网站时,你的浏览器(客户端)向Web服务器(服务端)发起TCP连接请求,服务器接受这个请求后,两者之间就建立了一条可靠的通信链路,用于传输网页数据。

  •  并发处理

服务端通常需要同时处理多个客户端的请求。通过获取新连接,服务端可以为每个客户端创建一个独立的连接,从而实现并发处理。例如,一个在线游戏服务器可能需要同时处理成千上万的玩家连接。每个玩家的客户端都会向服务器发起TCP连接请求,服务器接受这些请求并创建对应的连接,这样就能同时为多个玩家提供服务。

  • 数据交互

获取新连接是数据传输的起点。一旦连接建立,客户端和服务端就可以通过读写套接字( read 和 write 函数,或者 recv 和 send 函数)来交换数据。例如,在一个文件传输服务中,客户端请求下载文件,服务端接受连接后,就可以通过已建立的连接将文件数据发送给客户端。
 

TCP_echo_server

下面我们来手动实现一下tcp_echo_server;按照下面的流程来实现:

(1)创建套接字

// 1、创建套接字
_listensockfd = ::socket(AF_INET, SOCK_STREAM, 0);
if (_listensockfd < 0)
{
      LOG(FATAL, "socket error\n");
      exit(SOCKET_ERROR);
}
LOG(DEBUG, "socket create sucess\n");

(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 = INADDR_ANY;

int n = ::bind(_listensockfd, (struct sockaddr *)&local, sizeof(local));

(3)设置listen状态

// 3、设置套接字为listen状态
// 因为tcp是面向连接的,tcp需要未来不断地能够做到获取连接
int m = listen(_listensockfd, MAXCOUNT);
if (m < 0)
{
     LOG(FATAL, "listen error\n");
}
LOG(INFO, "listen sucess\n");

(4)loop

(1)获取新连接

// 获取新连接
struct sockaddr_in client;
socklen_t len = sizeof(client);
int sockfd = ::accept(_listensockfd, (struct sockaddr *)&client, &len);
if (sockfd < 0)
{
      LOG(WARNING, "accept error\n");
      continue;
}
InetAddr addr(client);
LOG(INFO, "get a new link,client info :%s\n", addr.Addrstr().c_str());

(2)

因为下面的几个版本都需要用的Service函数,我先提前放到这里;

    void Service(int sockfd, InetAddr addr)
    {
        while (true)
        {
            char inbuffer[2048];
            ssize_t n = ::read(sockfd, inbuffer, sizeof(inbuffer) - 1); // 读消息,-1是因为它是系统调用不会做字符串处理
            if (n > 0)
            {
                inbuffer[n] = 0;
                string echo_string = "[server echo]# ";
                echo_string += inbuffer;
                n = ::write(sockfd, echo_string.c_str(), echo_string.size()); // 写消息
            }
            else if (n == 0) // 读到文件结尾,也就是通信结束
            {
                LOG(INFO, "client %s quit\n", addr.Addrstr().c_str());
                break;
            }
            else
            {
                LOG(ERROR, "read server error:%s\n", addr.Addrstr().c_str());
                break;
            }
        }
    }

1、version 0 版本

Service(sockfd, addr);

2、version 1 ----多进程版本

 pid_t id =fork();
if(id==0)
{
::close(_listensockfd);
if(fork()>0)exit(0);//子进程退出
Service(sockfd, addr);//孙进程会造成僵尸进程
exit(0);
}
//father
::close(sockfd);
int n=waitpid(id,nullptr,0);//1、忽略;2、父子孙
if(n>0)
{
LOG(INFO,"wait child sucess\n");
}

3、version 2 ----多线程版本

pthread_t tid;
ThreadDate *td = new ThreadDate(sockfd, this, addr);
pthread_create(&tid, nullptr, Execute, td); // 新线程分离

4、version 3 ---线程池版本

task_t aa=std::bind(&TcpServer::Service,this,sockfd,addr);
ThreadPool<task_t>::GetInstance()->Equeue(aa);

(5)客户端

以上我们就完成了tcp_echo_server,也就是用tcp实现的一个简单的通信;

多线程远程执行命令

这里我们用version 2 多线程版本来实现远程执行命令;

为了使代码更加美观,我们在version2版本的基础上,将Service函数进行封装;将它封装到command.hpp的HandlerCommand函数里面;

这样我们调用回调函数,然后再主函数中进行绑定即可;

using command_server_t =std::function<void(int sockfd, InetAddr addr)>;
void HandlerCommand(int sockfd, InetAddr addr)
{
    while (true)
    {
        char inbuffer[2048];
        ssize_t n = ::recv(sockfd, inbuffer, sizeof(inbuffer) - 1,0); // 读消息,-1是因为它是系统调用不会做字符串处理             
        if (n > 0)
        {
            inbuffer[n] = 0;
            string result = Excute(inbuffer);
            n = ::send(sockfd, result.c_str(), result.size(),0); // 写消息
        }
        else if (n == 0) // 读到文件结尾,也就是通信结束
        {
            LOG(INFO, "client %s quit\n", addr.Addrstr().c_str());
            break;
        }
        else
        {
            LOG(ERROR, "read server error:%s\n", addr.Addrstr().c_str());
            break;
        }
    }
}

完成上面的准备工作,怎么执行命令?

在之前我写过手写shell的文章,可以直接用里面的代码,文章用的是exec*函数来执行命令;但是这里我们用一个更加简单的函数:popen直接执行即可;

string Excute(const string &cmdstr)
{
    FILE *fp=popen(cmdstr.c_str(),"r");
    string result;
    if(fp)
    {
        char line[1024];
        while(fgets(line,sizeof(line),fp))//读文件
        {
            result+=line;
        }
        return result;
    }
    else
    {
        return "execute error";
    }
}

完整代码:

tcp_echo_server:

登录 - Gitee.com

远程执行命令:

登录 - Gitee.com


http://www.kler.cn/a/380645.html

相关文章:

  • 2022 NOIP 题解
  • 鸿蒙HarmonyOS开发:给应用添加基础类型通知和进度条类型通知(API 12)
  • MySQL之JDBC入门详解
  • go 聊天系统项目-1
  • 网络编程入门
  • 深度学习基础—语言模型和序列生成
  • vue框架简介
  • Android 解决飞行模式下功耗高,起伏波动大的问题
  • JeecgBoot入门
  • 如何使用springboot+redis开发一个简洁的分布式锁?
  • windows XP,ReactOS系统3.4 共享映射区(Section)---2
  • 视频批量裁剪工具
  • Jupyter notebook 添加目录插件
  • 【VScode】中文版ChatGPT编程工具-CodeMoss!教程+示例+快捷键
  • 移动混合开发面试题及参考答案
  • 企业人力资源招聘面试新工具:AI智能面试系统
  • 大数据与智能算法助力金融市场分析:正大的技术创新探索
  • qt QMenuBar详解
  • windwos安装多版本Maven(图文详细版)
  • stm32使用串口的轮询模式,实现数据的收发
  • 【C++】C++的单例模式
  • 后台管理系统的通用权限解决方案(十)如何自定义SpringMVC的参数解析器
  • springboot+shiro 权限管理
  • 【前端基础】盒子模型
  • 华为HarmonyOS打造开放、合规的广告生态 - 开屏广告
  • 【双指针】【数之和】 LeetCode 633.平方数之和