最新资讯

  • 【linux】高级IO,I/O多路转接多路转接之epoll,epoll版本的TCP服务器的实现

【linux】高级IO,I/O多路转接多路转接之epoll,epoll版本的TCP服务器的实现

2026-01-29 07:14:24 栏目:最新资讯 3 阅读

小编个人主页详情<—请点击
小编个人gitee代码仓库<—请点击
linux系统编程专栏<—请点击
linux网络编程专栏<—请点击
倘若命中无此运,孤身亦可登昆仑,送给屏幕面前的读者朋友们和小编自己!


目录

    • 前言
    • 一、前置知识
    • 二、nocopy
    • 三、Epoller
    • 四、Main.cc
    • 五、EpollServer.hpp
    • 六、源代码
      • Epoller.hpp
      • EpollServer.hpp
      • Log.hpp
      • Main.cc
      • makefile
      • nocopy.hpp
      • Socket.hpp
    • 总结


前言

【linux】高级IO,I/O多路转接之epoll,接口和原理讲解,epoll_create,epoll_wait,epoll_ctl——书接上文 详情请点击<——,本文会在上文的基础上进行讲解,所以对上文不了解的读者友友请点击前方的蓝字链接进行学习
本文由小编为大家介绍——【linux】高级IO,I/O多路转接多路转接之epoll,epoll版本的TCP服务器的实现


一、前置知识

  1. 上一篇文章,详情请点击<—— 那么在有了上一篇文章中,关于epoll的三个接口epoll_create,epoll_wait,epoll_ctl的铺垫性讲解,以及epoll原理的讲解后,接下来小编就要实现是epoll版本的TCP服务器,我们需要使用到日志,套接字,链接如下
    关于Log.hpp日志的讲解,详情请点击<——
    关于Socket.hpp套接字的讲解,在第三点的TCP服务器的Socket.hpp进行的讲解,详情请点击<——
  2. 那么如上我们的epoll版本的TCP服务器需要建立以及包含的源文件和头文件概况如上
    (一)Epoller.hpp,由于我们要实现的是epoll版本的TCP服务器,所以就要创建,等待,控制epoll模型,那么使用到epoll的三个接口,那么关于epoll的三个接口以及文件描述符epfd我们期望进行一定的封装,所以我们就要实现一个Epoller类放在Epoller.hpp文件中
    (二)EpollerServer.hpp,用于放置epoll版本的TCP服务器对应的EpollerServer类,包含初始化服务器,启动服务器等
    (三)Log.hpp是日志,其中定义了Log lg,所以我们可以直接使用lg打印日志
    (四)Main.cc是主函数,包含调用服务器的逻辑
    (五)makefile用于编译构建可执行程序
    (六)nocopy.hpp用于方式nocopy类,作为一款服务器,这个服务器只能有一份,所以我们并不期望服务器被拷贝,所以我们让服务器EpollServer继承nocopy即可,同样的Epoller这个类是用于创建,等待,控制epoll模型,那么我们期望创建的epoll模型不可被拷贝,所以我们也让Epoller继承nocopy
    (七)Socket.hpp用于封装套接字的原生接口

二、nocopy

  1. nocopy是处于nocopy.hpp的类,那么对于nocopy这个类要实现的是禁止拷贝这个功能,所以我们只需要将拷贝构造和赋值运算符使用delete关键字禁用,这样凡是公有继承自这个nocopy的类都无法进行拷贝了
#pragma once


class nocopy
{
public:
    nocopy(){};

    nocopy(const nocopy& ) = delete;
    
    const nocopy& operator=(const nocopy& ) = delete;
};
  1. 所以此时我们在main函数中,尝试一下对公有继承自nocopy的类Epoller进行拷贝构造
#include "Epoller.hpp"

class Epoller : public nocopy
{};

int main()
{
    Epoller ep;
    Epoller ep1 = ep;

    return 0;
}

运行结果如下

  1. 所以我们可以让Epoller类公有继承自nocopy,所以这样Epoller创建的epoll模型就无法被拷贝,让epoll模型只为它所对应的epoll提供一次等待多个fd的工作
  2. 同样的,我们也要让EpollerServer类公有继承自nocopy,所以这样epoll版本的TCP服务器只能有一份,不能被拷贝

三、Epoller

  1. Epoller是处于Epoller.hpp的类,Epoller是对epoll的三个系统调用函数,分别是epoll_create,epoll_wait,epoll_ctl以及文件描述符epfd进行封装,那么关于epoll的三个系统调用分别是epoll_create,epoll_wait,epoll_ctl如何使用小编在上一篇中进行讲解,本文和上一篇文章的耦合度很高,所以请读者友友学习完上一篇文章之后再来学习Epoller的封装,上一篇文章,详情请点击<——
  2. 那么接下来我们就要实现类Epoller了,所以这个类是用于创建epoll模型的,对于epoll模型我们不期望被拷贝,所以这里我们不期望Epoller类被拷贝,所以这里我们就让Epoller类公有继承自nocopy即可
  3. 此时我们来看一下类的私有成员变量,那么首先就要有一个_epfd作为我们找到epoll模型的入口,epoll可以一次等待多个fd,所以对于超时时间timeout这里我们初始化为3000微秒,即对应3秒
  4. Epoller类的私有成员变量中还应该有一个静态const的int类型size,对于size的初始值我们随便设置,这个size是用于给epoll_create进行传参的,对于epoll_create的参数在现在这个标准下已经废弃不使用了,所以这里对于size我们可以设置为任意值,那么这里小编就将size设置为128
  5. 那么在Epoller构造函数中我们就可以创建epoll模型了,那么如何创建呢?调用epoll_create即可,epoll_create可以创建epoll模型,并且将epoll模型关联的文件描述符epfd返回,如果失败,那么返回-1,此时我们打印日志,并且将错误描述也使用strerror打印出来
  6. 如果创建epoll模型成功,那么返回的是一个大于等于0的数,此时我们同样打印日志,将这个epfd打印出来即可
  7. 那么在Epoller的析构函数中,如果_epfd大于等于0,那么我们使用close关闭这个_epfd即可,当_epfd被关闭的时候,底层对应的epoll模型就会被释放,即底层的红黑树和就绪队列就会被释放,即其中的节点就都会被释放
#include 
#include 
#include 
#include 
#include "Log.hpp"
#include "nocopy.hpp"


class Epoller : public nocopy
{
    static const int size = 128;

public:
    Epoller()
    {
        _epfd = epoll_create(size);
        if(_epfd == -1)
        {
            lg(Error, "epoll_create error: %s", strerror(errno));
        }
        else
        {
            lg(Info, "epoller_create success, epfd: %d", _epfd);
        }
    }

    


    ~Epoller()
    {
        if(_epfd > 0)
        {
            close(_epfd);
        }
    }

private:
    int _epfd;
    int _timeout{3000};
};

  1. 接下来我们开始封装epoll_wait,epoll_wait可以从epoll模型中的就绪数组中获取就绪节点,那么对于epoll_wait我们将其封装在EpollerWait这个成员函数中即可,那么对于EpollerWait的参数需要一个struct epoll_event类型的数组revents,用于将epoll_wait等待到的fd及其事件带出去,即revents这个数组是一个输出型参数,参数中还有一个num用于给maxevents传参,也是期望上层传入,所以num是一个输入型参数
  2. 所以此时我们就可以调用epoll_wait然后传入文件描述符_epfd,事件数组revents,num最大就绪事件的fd的个数,超时时间_timeout,然后对于返回值n我们不进行处理,将n带出去,由外部进行处理,epoll_wait的返回值n表示本次获取的事件数组revents中就绪的fd的个数
int EpollerWait(struct epoll_event revents[], int num)
{
    int n = epoll_wait(_epfd, revents, num, _timeout);
    
    return n;
}

  1. 接下来我们封装epoll_ctl,epoll_ctl是对epoll模型中的红黑树新增,删除,修改节点中的fd以及event事件的,所以我们将epoll_ctl封装在EpollerUpdate,那么对于EpollerUpdate的参数要包括要对节点进行操作oper,对哪一个文件描述符sock进行操作,对文件描述符sock上的哪一个事件event进行操作
  2. 所以我们观察上图,oper分为三种,分别是EPOLL_CTL_DEL删除节点,EPOLL_CTL_ADD新增节点,EPOLL_CTL_MOD修改节点,那么其实我们思考一下,epoll_ctl本质上是操作的epoll模型中的红黑树,并且红黑树节点的键值key是文件描述符sock,节点value中包含文件描述符sock和event事件
  3. 那么如果我们仅仅想要删除一个节点,所以我们只需要知道这个节点的键值key即可在红黑树中找到这个节点进行删除,并不需要对event事件进行操作,但是如果我们想要新增节点或者修改节点,除了要使用文件描述符sock作为键值新增或者修改找到这个节点之外,还需要对节点中包含的event事件进行新增或者修改的操作
  4. 所以我们由此就可以进行分类了,将EPOLL_CTL_DEL删除节点是一类,将EPOLL_CTL_ADD新增节点和EPOLL_CTL_MOD修改节点归为一类,所以此时我们定义一个返回值n默认为0,如果epoll_ctl对节点操作成功那么返回值为0,如果操作失败那么返回值为-1,所以此时接下来我们就可以根据上面的讲解进行if-else语句的判断了
  5. 如果当前要进行的操作oper是EPOLL_CTL_DEL删除节点,那么我们就调用epoll_ctl依次传入_epfd,要进行的操作oper,删除的文件描述符sock,最后由于我们仅仅是删除节点,所以我们只需要利用文件描述符sock找到红黑树的节点进行删除即可,并不需要对节点中的事件进行新增或者修改操作,所以这里对于事件event我们可以不进行传入,仅仅传入nullptr即可,那么如果返回值为-1,那么我们打印日志即可
  6. 如果当前要进行的操作oper是EPOLL_CTL_ADD新增节点或者EPOLL_CTL_MOD修改节点,那么此时我们就调用epoll_ctl依次传入_epfd,要进行的操作oper,要操作的文件描述符sock,最后由于我们是要新增或者修改节点,所以我们需要利用文件描述符sock新增或者修改红黑树的节点,并且由于是新增或者修改节点,所以对于节点中的event我们也要进行新增或者修改
  7. 所以此时我们还要传入event,但是这里特别注意,我们要传入的event并不是EpollerUpdate参数中uint32_t类型的event,而是上图中struct epoll_event类型的ev,并且这个uint32_t类型的event是struct epoll_event类型的第一个成员变量,那么对于第二个成员变量data,那么我们只需要设置其中的参数文件描述符fd为sock即可
  8. 那么这时候有的读者友友可能会想,epoll_ctl的参数中明明已经有文件描述符sock了,为什么还要在struct epoll_event中的data中也要包含文件描述符sock呢?
  9. 其实epoll_ctl参数中包含的文件描述符sock是作为键值key用于新增或找到红黑树中的节点的,而此时这个struct epoll_event中既包含fd也包含event事件,那么将来event_wait获取的数组中的对象epoll_event,这个epoll_event才可以让我们得知是哪个fd就绪了,fd上的哪个event就绪了
  10. 所以此时我们就初始化完成了struct epoll_event类型的对象ev了,那么我们将ev取地址直接进行传入到epoll_ctl的最后一个参数即可,那么如果epoll_ctl的返回值为-1,说明此时oper操作进行新增或修改节点失败了,所以此时我们打印日志即可,最后在if-else语句外面返回epoll_ctl的返回值n即可
int EpollerUpdate(int oper, int sock, uint32_t event)
{
    int n = 0;
    if(oper == EPOLL_CTL_DEL)
    {
        n = epoll_ctl(_epfd, oper, sock, nullptr);
        if(n == -1)
        {
            lg(Error, "epoll_ctl delete error");
        }
    }
    else
    {
        // EPOLL_CTL_ADD || EPOLL_CTL_MOD
        struct epoll_event ev;
        ev.data.fd = sock;
        ev.events = event;
        
        n = epoll_ctl(_epfd, oper, sock, &ev);
        if(n == -1)
        {
            lg(Error, "epoll_ctl error");
        }
    }

    return n;
}
  1. 所以此时我们在Epoller类中就完成了对epoll的三个系统调用接口epoll_create,epoll_wait,epoll_ctl的封装,并且也将epoll模型对应的文件描述符epfd也封装在了Epoller类的私有成员变量中
  2. 那么从此以后,如果我们想要使用epoll等待多个fd,那么只需要包含"Epoller.hpp"这个头文件,然后实例化Epoller类对象就完成了创建epoll模型,那么进行调用成员变量可以完成将epoll等待的多个fd获取上来,还可以控制节点中对应的fd和event的操作

四、Main.cc

  1. 那么在main函数中包含对服务器的调用逻辑,所以我们使用智能指针unique_ptr管理epoll版本的TCP服务器,那么在new服务器对象的时候,进行传参服务器要绑定的端口号8080即可
  2. 接下来服务器要首先调用Init进行初始化,接下来之后再调用Start将服务器启动起来
#include "EpollServer.hpp"
#include 


int main()
{
    std::unique_ptr<EpollServer> epoll_svr(new EpollServer(8080));

    epoll_svr->Init();
    epoll_svr->Start();

    return 0;
}

五、EpollServer.hpp

  1. 那么对于一款服务器来讲,我们不期望这个服务器可以被拷贝,所以我们让EpollServer公有继承nocopy即可,接下来我们来看EpollServer的私有成员变量,首先作为一款服务器应该有服务器所绑定的端口号port,所以这里的私有成员变量要有端口号_port
  2. 并且我们要实现的服务器是epoll版本的TCP服务器,所以要进行socket编程,那么对于socket相关的接口,我们已经封装在了Socket.hpp中的类Sock中了,所以这里我们要实例化一个Sock的类对象,那么我们期望这个类对象可以自动管理生命周期,所以这里我们在私有成员变量中使用shared_ptr对应的_listensock_ptr管理Sock的类对象
  3. 同样的既然是epoll版本的TCP服务器,所以这里就要使用epoll的相关接口,小编在前面已经将epoll的相关接口封装在了Epoller.hpp中的Epoller类中了,所以这里我们就要实例化一个Epoller的列对象,那么同样的我们期望这个类对象可以自动管理生命周期,所以这里我们在私有成员变量中使用shared_ptr对应的_epoller_ptr管理Epoller的类对象
  4. 那么私有成员变量我们还要定义一个static和const修饰的int类型的变量num,初始值我们给64,用于给Epoller中的EpollerWait的第二个参数传参,即num表示将来我们获取就绪的fd的最大个数,所以此时我们就完成了对EpollServer类的私有成员变量的声明
  5. 那么接下来,我们在构造函数中对私有成员变量进行初始化,我们期望服务器所绑定的端口号是外部传入的,所以这里在构造函数中我们接收端口号port,然后给_listensock_ptr这个成员变量new一个Sock对象,给_epoller_ptr这个成员变量new一个Epoller对象,然后使用接收到的端口号port初始化成员变量_port
  6. 在析构函数中,我们调用_listensock_ptr中的Close关闭listen文件描述符即可,那么我们要关心的是文件描述符的读事件EPOLLIN和写事件EPOLLOUT,那么这里我们规范一点,将读事件EPOLLIN定义为EVENT_IN,将写事件EPOLLOUT定义为EVENT_OUT
#include 
#include 
#include "Log.hpp"
#include "Socket.hpp"
#include "nocopy.hpp"
#include "Epoller.hpp"


uint32_t EVENT_IN = (EPOLLIN);
uint32_t EVENT_OUT = (EPOLLOUT);


class EpollServer : public  nocopy
{
    static const int num = 64;
public:
    EpollServer(uint16_t port)
        :_listensock_ptr(new Sock())
        ,_epoller_ptr(new Epoller())
        ,_port(port)
    {}

    void Init()
    {}

    void Start()
    {}

    ~EpollServer()
    {
        _listensock_ptr->Close();
    }
    
private:
    std::shared_ptr<Sock> _listensock_ptr;
    std::shared_ptr<Epoller> _epoller_ptr;
    uint16_t _port;
};
  1. 接下来我们就开始编写Init初始化服务器了,依次需要完成套接字的创建,服务器的绑定,将套接字设置为监听状态,那么我们依次调用_listensock_ptr中的成员函数Socket完成套接字的创建,调用Bind完成服务器的绑定,调用Listen将_listensock设置为监听状态,那么此时服务器已经初始化完成,并且此时Epoll模型的创建在EpollServer构造函数中已经完成了,并且在Epoller的构造函数中已经打印了日志,那么在这里我们也打印创建listen套接字完成这个日志,并且将listen套接字对应的fd也一并打印出来
void Init()
{
    _listensock_ptr->Socket();
    _listensock_ptr->Bind(_port);
    _listensock_ptr->Listen();

    lg(Info, "create listen socket success, fd: %d", _listensock_ptr->Fd());
}
  1. 接下来我们开始编写Start启动服务器,那么服务器一旦启动就要源源不断的接收来自客户端的连接请求,服务器一旦启动几乎都在运行,所以也就意味着服务器的逻辑要基于一个死循环运行,所以这里我们使用for(;;)进行死循环,listensock上的读事件就绪代表有连接到来,那么我们能否一上来就让服务器accept对应的listensock呢?
  2. 不能,因为我们要实现的是epoll版本的TCP服务器,说好的使用epoll一次等待多个fd,那么一旦这里服务器上来就accept等待客户端的连接,此时在单进程中服务器一次只能等待一个listensock这个一个fd ,别忘了epoll可以一次等待多个fd,所以等待的事情我们就要交给epoll来做,那么使用了epoll我们就可以在单进程场景中,一次等待多个fd,让单进程服务器一次服务多个客户端
  3. 所以此时在进行for循环之前,我们要将listensock添加到epoll中,那么本质就是将listensock和它关心的事件,添加到内核epoll模型中的红黑树rb_tree中,所以此时我们调用_epoller_ptr中的EpollerUpdata进行EPOLL_CTL_ADD新增操作,将listensock对应的fd以及关心的读事件EVENT_IN添加到epoll模型的rb_tree中
  4. 那么在for循环中我们此时就只需要调用_epoller_ptr中的EpollerWait传入struct epoll_event类型的revs数组以及num最大获取就绪fd的个数,将就绪的fd通过数组的形式获取上来即可,那么通过返回值n我们可以获悉有多少个fd就绪了,这个fd的值小于等于num
  5. 所以此时我们EpollerWait的返回值n进行判断即可,如果n大于0,说明此时底层有fd就绪了,所以我们这里将revs数组中的第一个fd打印一下日志,别忘了有可能就绪的fd可能不止一个,而是都处于数组中,具体有多少个,有n个,所以我们就可以对这些就绪的处于数组中的fd及其就绪的事件event进行处理了,那么如何处理?交给派发器Dispatcher处理即可
void Dispatcher(epoll_event revs[], int n)
{

}

void Start()
{
    _epoller_ptr->EpollerUpdate(EPOLL_CTL_ADD, _listensock_ptr->Fd(), EVENT_IN);
    struct epoll_event revs[num];
    for(;;)
    {
        int n = _epoller_ptr->EpollerWait(revs, num);
        if(n > 0)
        {
            lg(Debug, "event happened, fd: %d", revs[0].data.fd);
            
            Dispatcher(revs, n);
        }
        else if(n == 0)
        {
            lg(Info, "time out...");
        }
        else
        {
            lg(Error, "epoll wait error");
        }
    }
}
  1. 就绪的fd的事件分为读事件就绪和写事件就绪,那么为了简便学习,目前在epoll中我们暂时不讨论对写事件就绪进行处理,我们仅对fd就绪事件为的读事件就绪的情况进行处理,那么此时fd的读事件已经就绪了,别忘了fd分为listensock和普通的sock
  2. 对于listensock那么则是读事件就绪了要使用accept将来自客户端的连接从底层获取上来,对于普通sock则是读事件就绪了要将客户端发来的数据进行读取处理,两者的处理逻辑不同,所以listensock的代码处理封装为Accepter连接管理器,将sock的代码处理封装为Recver数据读取器
  3. 那么我们要处理的就绪的fd都在revs数组中放着,数组内有多少个就绪的fd,我们如何知道?那么就是通过EpollerWait的返回值n,n表示数组内的就绪的fd的个数,所以Dispatcher的第二个参数我们就将n传入了,所以我们知道了要遍历revs这个数组,要遍历的上限范围是n
  4. 所以此时我们就可以开始遍历一下revs了,那么别忘了数组的中的对象的类型是struct epoll_event,其中的第一个字段是uint32_t类型的events,即就绪的事件,第二个字段是epoll_data_t类型的data,而在epoll_data_t类型中有fd这个字段,所以我们单单从一个struct epoll_event类型中就可以获悉哪个fd上的哪个event事件就绪了,所以此时我们就可以理解了当初在epoll_ctl传参的时候要传参struct epoll_event类型
  5. 那么此时我们已经可以知道哪个fd上的哪个event事件就绪了,所以我们此时就可以进行判断了,如果是写事件或者其它事件就绪了,在本文的处理逻辑中我们暂时忽略,为了简便学习,我们仅考虑if判断读事件就绪了,即此时fd上的读事件就绪了,那么fd分为listensock和其它的普通sock,那么如果fd是listensock我们就交给连接管理器Accepter处理,否则fd就是其它的普通sock,那么我们就交给数据读取器Recver处理
void Accepter()
{

}

void Recver(int fd)
{

}

void Dispatcher(struct epoll_event revs[], int n)
{
    for(int i = 0; i < n; i++)
    {
        int fd = revs[i].data.fd;
        uint32_t event = revs[i].events;
        if(event & EVENT_IN)
        {
            if(fd == _listensock_ptr->Fd())
            {
                Accepter();
            }
            else
            {
                Recver(fd);
            }
        }
        else if(event & EVENT_OUT)
        {

        }
        else
        {

        }
    }
}
  1. 所以接下来我们编写一下连接管理器Accepter,那么当调用Accepter的时候,意味着此时的listensock上的读事件就绪了,意味着此时有客户端来连接服务器,服务器需要将连接从底层的全连接队列使用accept获取上来,所以此时我们就调用_listensock_ptr中的Accept即可
  2. 那么如果返回的文件描述符sock大于0,说明此时accept获取连接成功,那么此时我们可以直接read读取sock上的数据吗?不可以,因为如果客户端仅仅只是建立连接而不发送数据,所以此时read读取底层的tcp接收缓冲区就会由于没有数据而导致read阻塞等待,所以此时服务器对于其它客户端的连接或者发来的数据就会不响应了
  3. 所以要等待sock上有数据的本质上就是等待sock的读事件就绪,别忘了本文小编要实现的是epoll版本的TCP服务器,要等待的事情都交给epoll即可,epoll可以一次等待多个fd,所以此时我们不能直接read读取数据,而是应该将关心等待sock上的读事件就绪通过EpollerUpdate交给epoll模型中的红黑树rb_tree,让epoll关心这个sock和对应的event事件,接下来我们打印客户端信息的日志即可
void Accepter()
{
    std::string clientip;
    uint16_t clientport;

    int sock = _listensock_ptr->Accept(&clientip, &clientport);
    if(sock > 0)
    {
        _epoller_ptr->EpollerUpdate(EPOLL_CTL_ADD, sock, EVENT_IN);
        lg(Info, "get a new link, client info@ %s:%d", clientip.c_str(), clientport);
    }
}

运行结果如下

  1. 所以此时如上,那么我们在Epoller中在EpollerWait中给epoll_wait设置的超时时间time是3000微秒,即3秒,所以如上图左边前9秒,小编没有让客户端连接服务器,所以左侧每个3秒就会超时一次,那么9秒过后,小编让右侧的telnet充当客户端连接服务器,此时我们的服务器中的epoll可以检测到listensock上的读事件就绪,并且正常调用了连接管理器Accepter可以正常将连接获取上来添加到epoll中让epoll关心这个连接对应的文件描述符sock上的读事件,并且打印了日志,无误
  2. 那么接下来小编在右侧的客户端上进行了输入,即此时客户端给服务器通过tcp连接发送了数据,所以此时左侧服务器中的epoll就会检测到sock上的读事件就绪,那么就会告诉上层sock上的读事件就绪了,赶快来读取吧,epoll会一直通知到上层进行读取
  3. 可是在上层此时小编并没有对数据读取器Recver进行编写,所以自然也就无法读取sock上的数据,进而我们每次在Start中的死循环中调用EpollerWait中封装的epoll_wait都会直接进行返回,然后上层打印fd就绪日志,可是上层由于没有编写Recver无法对sock上的数据进行读取,底层由于上层没有读取就会一直返回通知,所以此时在左侧就会出现满屏的日志信息
  4. 那么下面小编想验证一下非阻塞,即在Epoller中的EpollerWait中封装的epoll_wait的超时时间为0,那么此时epoll就为非阻塞等待,那么由于小编在代码中没有进行任何sleep休眠,所以服务器一旦运行,那么由于是非阻塞,那么一进行调用那么就会返回,超时,然后在调用再返回,超时,所以这样循环,那么超时的日志就会一瞬间将屏幕打满,所以如下,小编在Epoller中的EpollerWait中封装的epoll_wait的超时时间为0,我们期望看到超时的日志一瞬间打满屏幕的现象
int EpollerWait(struct epoll_event revents[], int num)
{
    // int n = epoll_wait(_epfd, revents, num, _timeout);
    int n = epoll_wait(_epfd, revents, num, 0);
    
    return n;
}

运行结果如下,非阻塞情况下一瞬间超时日志就会将屏幕打满

  1. 那么下面小编想验证一下阻塞等待,所以小编就将Epoller中的EpollerWait中封装的epoll_wait的超时时间为-1,那么此时epoll就为阻塞等待直到fd上的事件就绪
 int EpollerWait(struct epoll_event revents[], int num)
 {
     // int n = epoll_wait(_epfd, revents, num, _timeout);
     // int n = epoll_wait(_epfd, revents, num, 0);
     int n = epoll_wait(_epfd, revents, num, -1);
     
     return n;
 }

运行结果如下

  1. 所以此时左侧服务器一旦运行,那么就是阻塞式等待fd上的事件就绪,当小编在右侧使用telnet充当客户端连接服务器的时候,此时服务器的epoll上的listensock上的读事件就绪了,那么此时epoll就会返回,然后告诉上层处理listensock的读事件,所以此时上层处理完成,将连接获取上来放到epoll中,然后打印完成日志,此时左侧服务器又去阻塞式等待fd上的事件就绪了
  2. 所以此时小编也编写一下数据读取器Recver,将数据读取上来,那么首先我们创建用户级缓冲区buffer,然后使用read将数据读取上来,如果read的返回值n大于0,那么代表此时成功的读取了数据,所以此时我们在数据的末尾放上‘’(0等于‘’),然后将这个字符串打印出来,接下来我们也给客户端发送消息,所以此时我们就构建echo_svr为"server echo$ ",然后我们将来客户端的消息添加在echo_svr的后面即可,然后使用write将消息通过连接发送回给客户端
  3. 接下来如果n等于0,说明对方将连接关闭,所以此时服务器也想要将连接关闭,所以服务器这边先打印日志,这里细节来了,epoll要求如果想要使用epoll_ctl的EPOLL_CTL_DEL关闭fd,那么首先要求这个fd有效,所以如果这里我们贸然的上来先使用close关闭fd,然后再调用eopll_ctl就会出错,所以这里我们一定要先使用epoll_ctl的EPOLL_CTL_DEL在内核epoll模型中的rb_tree中删除sock对应的节点,所以这里我们只需要找到sock,并不需要传参event,所以这里的event我们传入0即可,然后再调用close关闭fd
  4. 接下来如果n小于0,那么说明此时读取数据发生错误,那么此时打印日志,使用epoll_ctl的EPOLL_CTL_DEL删除内核epoll模型中rb_tree中对应的sock节点,然后再调用close关闭fd即可,所以此时我们就将数据读取器Recver也编写出来了,所以此时我们来验证一下,服务器是否可以成功的将来自客户端的数据读取,并且给客户端写回数据
void Recver(int fd)
{
    char buffer[1024];
    ssize_t n = read(fd, buffer, sizeof(buffer) - 1);
    if(n > 0)
    {
        buffer[n] = 0;
        std::cout << "get a message: " << buffer << std::endl;

        std::string echo_str = "server echo$ ";
        echo_str += buffer;
        write(fd, echo_str.c_str(), echo_str.size());
    }
    else if(n == 0)
    {
        lg(Info, "client quit, me to, close fd: %d", fd);
        _epoller_ptr->EpollerUpdate(EPOLL_CTL_DEL, fd, 0);
        close(fd);
    }
    else
    {
        lg(Warning, "recv error, fd: %d", fd);
        _epoller_ptr->EpollerUpdate(EPOLL_CTL_DEL, fd, 0);
        close(fd);
    }
}

运行结果如下

  1. 所以运行结果如上,别忘了此时的Epoller中的EpollerWait封装的epoll_wait的超时时间timeout仍然为-1阻塞等待直到fd上的事件就绪,所以此时左侧运行服务器之后,服务器启动后会阻塞等待直到fd上的事件就绪,那么此时右侧使用telnet充当客户端来连接服务器,所以此时服务器的epoll就会检测到listensock上有读事件就绪,然后告诉上层
  2. 所以此时上层经过判断是listensock上的读事件就绪了,所以此时就会将fd交给连接管理器Accepter将连接对应的文件描述符sock获取上来并且添加到epoll中关心读事件,然后打印日志,此时左侧的客户端发送消息,本质上是右侧的服务器检测到sock上的读事件就绪了,所以就会告诉上层
  3. 那么此时小编已经编写了数据读取器Recver将数据读取上来,并且构建服务器的消息发送回给客户端,所以此时客户端就收到了来自服务器的消息并打印,那么此时右侧的客户端想要断开连接,所以就quit退出了telnet,所以此时右侧的客户端将连接断开了,连接的断开相当于左侧的服务器epoll关心的sock上的读事件就绪
  4. 那么左侧的服务器的epoll就会检测到sock的读事件就绪了,所以就会告诉上层,那么上层经过判断,发现就绪的fd不是listensock,就绪的fd是普通sock,所以就会交给数据读取器将数据读取上来,可是此时read一进行读取,发现对方将连接关闭了,所以此时read就会返回0,告诉上层此时连接已经关闭了,无法进行读取了
  5. 所以上层就会打印日志,然后先使用epoll_ctl的EOPLL_CTL_DEL将连接对应的文件描述符sock删除,本质是在内核epoll模型中红黑树rb_tree中找到对应的sock节点进行删除,所以删除完成后,那么返回上层然后再调用close关闭sock,释放维护连接的文件打开对象

六、源代码

Epoller.hpp

#include 
#include 
#include 
#include 
#include "Log.hpp"
#include "nocopy.hpp"


class Epoller : public nocopy
{
    static const int size = 128;

public:
    Epoller()
    {
        _epfd = epoll_create(size);
        if(_epfd == -1)
        {
            lg(Error, "epoll_create error: %s", strerror(errno));
        }
        else
        {
            lg(Info, "epoller_create success, epfd: %d", _epfd);
        }
    }

    int EpollerWait(struct epoll_event revents[], int num)
    {
        // int n = epoll_wait(_epfd, revents, num, _timeout);
        // int n = epoll_wait(_epfd, revents, num, 0);
        int n = epoll_wait(_epfd, revents, num, -1);
        
        return n;
    }

    int EpollerUpdate(int oper, int sock, uint32_t event)
    {
        int n = 0;
        if(oper == EPOLL_CTL_DEL)
        {
            n = epoll_ctl(_epfd, oper, sock, nullptr);
            if(n == -1)
            {
                lg(Error, "epoll_ctl delete error");
            }
        }
        else
        {
            // EPOLL_CTL_ADD || EPOLL_CTL_MOD
            struct epoll_event ev;
            ev.data.fd = sock;
            ev.events = event;
            
            n = epoll_ctl(_epfd, oper, sock, &ev);
            if(n == -1)
            {
                lg(Error, "epoll_ctl error");
            }
        }

        return n;
    }



    ~Epoller()
    {
        if(_epfd >= 0)
        {
            close(_epfd);
        }
    }

private:
    int _epfd;
    int _timeout{3000};
};

EpollServer.hpp

#include 
#include 
#include 
#include 
#include "Log.hpp"
#include "Socket.hpp"
#include "nocopy.hpp"
#include "Epoller.hpp"


uint32_t EVENT_IN = (EPOLLIN);
uint32_t EVENT_OUT = (EPOLLOUT);


class EpollServer : public  nocopy
{
    static const int num = 64;
public:
    EpollServer(uint16_t port)
        :_listensock_ptr(new Sock())
        ,_epoller_ptr(new Epoller())
        ,_port(port)
    {}

    void Init()
    {
        _listensock_ptr->Socket();
        _listensock_ptr->Bind(_port);
        _listensock_ptr->Listen();

        lg(Info, "create listen socket success, fd: %d", _listensock_ptr->Fd());
    }

    void Accepter()
    {
        std::string clientip;
        uint16_t clientport;

        int sock = _listensock_ptr->Accept(&clientip, &clientport);
        if(sock > 0)
        {
            _epoller_ptr->EpollerUpdate(EPOLL_CTL_ADD, sock, EVENT_IN);
            lg(Info, "get a new link, client info@ %s:%d", clientip.c_str(), clientport);
        }
    }

    void Recver(int fd)
    {
        char buffer[1024];
        ssize_t n = read(fd, buffer, sizeof(buffer) - 1);
        if(n > 0)
        {
            buffer[n] = 0;
            std::cout << "get a message: " << buffer << std::endl;

            std::string echo_str = "server echo$ ";
            echo_str += buffer;
            write(fd, echo_str.c_str(), echo_str.size());
        }
        else if(n == 0)
        {
            lg(Info, "client quit, me to, close fd: %d", fd);
            _epoller_ptr->EpollerUpdate(EPOLL_CTL_DEL, fd, 0);
            close(fd);
        }
        else
        {
            lg(Warning, "recv error, fd: %d", fd);
            _epoller_ptr->EpollerUpdate(EPOLL_CTL_DEL, fd, 0);
            close(fd);
        }
    }

    void Dispatcher(struct epoll_event revs[], int n)
    {
        for(int i = 0; i < n; i++)
        {
            int fd = revs[i].data.fd;
            uint32_t event = revs[i].events;
            if(event & EVENT_IN)
            {
                if(fd == _listensock_ptr->Fd())
                {
                    Accepter();
                }
                else
                {
                    Recver(fd);
                }
            }
            else if(event & EVENT_OUT)
            {

            }
            else
            {

            }
        }
    }

    void Start()
    {
        _epoller_ptr->EpollerUpdate(EPOLL_CTL_ADD, _listensock_ptr->Fd(), EVENT_IN);
        struct epoll_event revs[num];
        for(;;)
        {
            int n = _epoller_ptr->EpollerWait(revs, num);
            if(n > 0)
            {
                lg(Debug, "event happened, fd: %d", revs[0].data.fd);
                
                Dispatcher(revs, n);
            }
            else if(n == 0)
            {
                lg(Info, "time out...");
            }
            else
            {
                lg(Error, "epoll wait error");
            }
        }
    }

    ~EpollServer()
    {
        _listensock_ptr->Close();
    }
    
private:
    std::shared_ptr<Sock> _listensock_ptr;
    std::shared_ptr<Epoller> _epoller_ptr;
    uint16_t _port;
};

Log.hpp

#pragma once

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 

#define SIZE 1024

#define Info   0
#define Debug  1
#define Warning 2
#define Error  3
#define Fatal  4

#define Screen 1     //输出到屏幕上
#define Onefile 2    //输出到一个文件中
#define Classfile 3  //根据事件等级输出到不同的文件中

#define LogFile "log.txt" //日志名称


class Log
{
public:
    Log()
    {
        printMethod = Screen;
        path = "./log/";
    }

    void Enable(int method) //改变日志打印方式
    {
        printMethod = method;
    }

    ~Log()
    {}

    std::string levelToString(int level)
    {
        switch(level)
        {
            case Info:
                return "Info";
            case Debug:
                return "Debug";
            case Warning:
                return "Warning";
            case Error:
                return "Error";
            case Fatal:
                return "Fatal";
            default:
                return "";
        }
    }

    void operator()(int level, const char* format, ...)
    {
        //默认部分 = 日志等级 + 日志时间
        time_t t = time(nullptr);
        struct tm* ctime = localtime(&t);
        char leftbuffer[SIZE];
        snprintf(leftbuffer, sizeof(leftbuffer), "[%s][%d-%d-%d %d:%d:%d]", levelToString(level).c_str(), 
        ctime->tm_year + 1900, ctime->tm_mon + 1, ctime->tm_mday, 
        ctime->tm_hour, ctime->tm_min, ctime->tm_sec);

        va_list s;
        va_start(s, format);
        char rightbuffer[SIZE];
        vsnprintf(rightbuffer, sizeof(rightbuffer), format, s);
        va_end(s);

        char logtxt[2 * SIZE];
        snprintf(logtxt, sizeof(logtxt), "%s %s", leftbuffer, rightbuffer);

        printLog(level, logtxt);
    }

    void printLog(int level, const std::string& logtxt)
    {
        switch(printMethod)
        {
            case Screen:
                std::cout << logtxt << std::endl;
                break;
            case Onefile:
                printOneFile(LogFile, logtxt);
                break;
            case Classfile:
                printClassFile(level, logtxt);
                break;
            default:
                break;
        }
    }

    void printOneFile(const std::string& logname, const std::string& logtxt)
    {
        std::string _logname = path + logname;
        int fd = open(_logname.c_str(), O_WRONLY | O_CREAT | O_APPEND, 0666);
        if(fd < 0)
            return;
        write(fd, logtxt.c_str(), logtxt.size());
        close(fd);
    }

    void printClassFile(int level, const std::string& logtxt)
    {
        std::string filename = LogFile;
        filename += ".";
        filename += levelToString(level);

        printOneFile(filename, logtxt);
    }


private:
    int printMethod;
    std::string path;
};

Log lg;

Main.cc

#include "EpollServer.hpp"
#include 


int main()
{
    std::unique_ptr<EpollServer> epoll_svr(new EpollServer(8080));

    epoll_svr->Init();
    epoll_svr->Start();

    // Epoller ep;
    // Epoller ep1 = ep;

    return 0;
}

makefile

epoll_server:Main.cc
	g++ -o $@ $^ -std=c++11
.PHONY:clean
clean:
	rm -f epoll_server

nocopy.hpp

#pragma once


class nocopy
{
public:
    nocopy(){};

    nocopy(const nocopy& ) = delete;
    
    const nocopy& operator=(const nocopy& ) = delete;
};

Socket.hpp

#pragma once

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include "Log.hpp"


const int backlog = 10;

enum{
    SocketErr = 1,
    BindErr,
    ListenErr,
};


class Sock
{
public:
    Sock()
    {}

    void Socket()
    {
        sockfd_ = socket(AF_INET, SOCK_STREAM, 0);
        if(sockfd_ < 0)
        {
            lg(Fatal, "socket error, %s : %d", strerror(errno), errno);
            exit(SocketErr);
        }

        int opt = 1;
        setsockopt(sockfd_, SOL_SOCKET, SO_REUSEPORT, &opt, sizeof(opt));
    }

    void Bind(uint16_t port)
    {
        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;
        socklen_t len = sizeof(local);

        if(bind(sockfd_, (struct sockaddr*)&local, len) < 0)
        {
            lg(Fatal, "bind error, %s : %d", strerror(errno), errno);
            exit(BindErr);            
        }
    }

    void Listen()
    {
        if(listen(sockfd_, backlog) < 0)
        {
            lg(Fatal, "listen error, %s : %d", strerror(errno), errno);
            exit(ListenErr);
        }
    }

    int Accept(std::string* clientip, uint16_t* clientport)
    {
        struct sockaddr_in peer;
        socklen_t len = sizeof(peer);

        int newfd = accept(sockfd_, (struct sockaddr*)&peer, &len);
        if(newfd < 0)
        {
            lg(Warning, "accept error, %s : %d", strerror(errno), errno);
            return -1;
        }
        
        char ipstr[128];
        inet_ntop(AF_INET, &(peer.sin_addr), ipstr, sizeof(ipstr));
        *clientip = ipstr;
        *clientport = ntohs(peer.sin_port);

        return newfd;
    }
    
    bool Connect(const std::string& serverip, uint16_t serverport)
    {
        struct sockaddr_in peer;
        memset(&peer, 0, sizeof(peer));
        peer.sin_family = AF_INET;
        peer.sin_port = htons(serverport);
        inet_pton(AF_INET, serverip.c_str(), &(peer.sin_addr));
        socklen_t len = sizeof(peer);

        int n = connect(sockfd_, (struct sockaddr*)&peer, len);
        if(n == -1)
        {
            std::cerr << "connect to " << serverip << ':' << serverport << "error" << std::endl;
            return false;
        }

        return true;
    }

    void Close()
    {
        if(sockfd_ > 0)
        {
            close(sockfd_);
        }
    }

    int Fd()
    {
        return sockfd_;
    }
    
    ~Sock()
    {}
private:
    int sockfd_;
};

总结

以上就是今天的博客内容啦,希望对读者朋友们有帮助
水滴石穿,坚持就是胜利,读者朋友们可以点个关注
点赞收藏加关注,找到小编不迷路!

本文地址:https://www.yitenyun.com/1493.html

搜索文章

Tags

#ios面试 #ios弱网 #断点续传 #ios开发 #objective-c #ios #ios缓存 #服务器 #python #pip #conda #远程工作 #kubernetes #笔记 #平面 #容器 #linux #学习方法 香港站群服务器 多IP服务器 香港站群 站群服务器 #Trae #IDE #AI 原生集成开发环境 #Trae AI #分阶段策略 #模型协议 #人工智能 #运维 #kylin #github #git #docker #后端 #数据库 #Conda # 私有索引 # 包管理 #数信院生信服务器 #Rstudio #生信入门 #生信云服务器 #物联网 #websocket #进程控制 #低代码 #爬虫 #音视频 #学习 #内网穿透 #网络 #cpolar #开发语言 #云原生 #iventoy #VmWare #OpenEuler #MobaXterm #ubuntu #Dell #PowerEdge620 #内存 #硬盘 #RAID5 #vscode #mobaxterm #深度学习 #计算机视觉 #node.js #华为云 #部署上线 #动静分离 #Nginx #新人首发 #fastapi #html #css #FTP服务器 #unity #c# #游戏引擎 #缓存 #开源 #hadoop #hbase #hive #zookeeper #spark #kafka #flink #算法 #大数据 #科技 #自然语言处理 #神经网络 #golang #java #redis #银河麒麟高级服务器操作系统安装 #银河麒麟高级服务器V11配置 #设置基础软件仓库时出错 #银河麒高级服务器系统的实操教程 #生产级部署银河麒麟服务系统教程 #Linux系统的快速上手教程 #harmonyos #鸿蒙PC #安全 #nginx #tcp/ip #vllm #大模型 #Streamlit #Qwen #本地部署 #AI聊天机器人 #RTP over RTSP #RTP over TCP #RTSP服务器 #RTP #TCP发送RTP #web安全 #需求分析 #centos #ssh #ide #jvm #我的世界 #android #腾讯云 #udp #电脑 #自动化 #jmeter #功能测试 #软件测试 #自动化测试 #职场和发展 #课程设计 #prometheus #gpu算力 #grafana #qt #c++ #asp.net大文件上传 #asp.net大文件上传下载 #asp.net大文件上传源码 #ASP.NET断点续传 #asp.net上传文件夹 #c语言 #stm32 #jar #ping通服务器 #读不了内网数据库 #bug菌问答团队 #架构 #asp.net #面试 #flask #http #fiddler #分布式 #华为 #银河麒麟 #系统升级 #信创 #国产化 #ModelEngine #多个客户端访问 #IO多路复用 #回显服务器 #TCP相关API #AI编程 #计算机网络 #编辑器 #研发管理 #禅道 #禅道云端部署 #RAID #RAID技术 #磁盘 #存储 #elasticsearch #性能优化 #C++ #凤希AI伴侣 #云计算 #json #openlayers #bmap #tile #server #vue #spring boot #vuejs #eBPF #网络协议 #Harbor #todesk #uni-app #小程序 #notepad++ #信令服务器 #Janus #MediaSoup #jenkins #windows #flutter #数码相机 #SSH #X11转发 #Miniconda #mcu #改行学it #创业创新 #程序员创富 #MCP #MCP服务器 #sqlserver #debian #deepseek #1024程序员节 #php #前端 #claude #cpp #项目 #高并发 #LoRA # RTX 3090 # lora-scripts #arm开发 #企业开发 #ERP #项目实践 #.NET开发 #C#编程 #编程与数学 #gemini #gemini国内访问 #gemini api #gemini中转搭建 #Cloudflare #流媒体 #NAS #飞牛NAS #监控 #NVR #EasyNVR #游戏 #MC #数据结构 #链表 #链表的销毁 #链表的排序 #链表倒置 #判断链表是否有环 #mysql #screen 命令 #mvp #个人开发 #设计模式 #儿童书籍 #儿童诗歌 #童话故事 #经典好书 #儿童文学 #好书推荐 #经典文学作品 #DisM++ # GLM-4.6V # 系统维护 #金融 #mcp #金融投资Agent #Agent #京东云 #版本控制 #Git入门 #开发工具 #代码托管 #AIGC #ida #制造 #个人博客 #svn #n8n #毕设 #嵌入式编译 #ccache #distcc #vue.js #AI论文写作工具 #学术论文创作 #论文效率提升 #MBA论文写作 #ollama #ai #llm #智能路由器 #django #RustDesk #IndexTTS 2.0 #本地化部署 #oracle #SA-PEKS # 关键词猜测攻击 # 盲签名 # 限速机制 #树莓派4b安装系统 #毕业设计 #车辆排放 #Spring AI #STDIO协议 #Streamable-HTTP #McpTool注解 #服务器能力 #Android #Bluedroid #时序数据库 #我的世界服务器搭建 #minecraft #智能手机 #AI #大模型学习 #Ansible #Playbook #AI服务器 #javascript #压力测试 #libosinfo #ssl #maven #gitlab #单片机 #嵌入式硬件 #TCP #客户端 #嵌入式 #DIY机器人工房 #windows11 #microsoft #系统修复 #高级IO #select #nas #语音识别 #说话人验证 #声纹识别 #CAM++ #ansible #gitea #macos #微服务 #p2p #Windows #万悟 #联通元景 #智能体 #镜像 #结构体 #scala #测试用例 #测试工具 #transformer #chatgpt #webrtc #idm #网站 #截图工具 #批量处理图片 #图片格式转换 #图片裁剪 #视频去字幕 #微信小程序 #微信 #健身房预约系统 #健身房管理系统 #健身管理系统 #Android16 #音频性能实战 #音频进阶 #阿里云 #SSE # AI翻译机 # 实时翻译 #密码学 #鸭科夫 #逃离鸭科夫 #鸭科夫联机 #鸭科夫异地联机 #开服 #北京百思可瑞教育 #百思可瑞教育 #北京百思教育 #无人机 #Deepoc #具身模型 #开发板 #未来 #apache #r-tree #聊天小程序 #NFC #智能公交 #服务器计费 #数据挖掘 #FP-增长 #WEB #risc-v #tdengine #涛思数据 #流量监控 #算力一体机 #ai算力服务器 #SSH公钥认证 # PyTorch # 安全加固 #Proxmox VE #虚拟化 #VMware #laravel #spring #diskinfo # TensorFlow # 磁盘健康 #PyTorch #CUDA #Triton #交互 #部署 #GPU服务器 #8U #硬件架构 #语言模型 #DeepSeek #昇腾300I DUO #NPU #CANN #ui #opencv #cosmic #搜索引擎 #tomcat #pytorch #H5 #跨域 #发布上线后跨域报错 #请求接口跨域问题解决 #跨域请求代理配置 #request浏览器跨域 #运维开发 #远程桌面 #远程控制 #东方仙盟 #游戏机 #JumpServer #堡垒机 #API限流 # 频率限制 # 令牌桶算法 #iBMC #UltraISO #黑群晖 #虚拟机 #无U盘 #纯小白 #支付 #1panel #vmware #振镜 #振镜焊接 #teamviewer #蓝湖 #Axure原型发布 #bash #单元测试 #集成测试 #YOLO # Triton # 目标检测 #网络安全 #pycharm #lua #llama #web #webdav #uv #uvx #uv pip #npx #Ruff #pytest #SRS #直播 #蓝耘智算 #milvus #springboot #知识库 #910B #昇腾 #ci/cd #web server #请求处理流程 #aws #MQTT协议 #react.js #政务 #深度优先 #DFS #rocketmq #selenium #守护进程 #复用 #screen #系统架构 # 双因素认证 #Clawdbot #个人助理 #数字员工 #扩展屏应用开发 #android runtime #rustdesk #postgresql #连接数据库报错 #VS Code调试配置 #umeditor粘贴word #ueditor粘贴word #ueditor复制word #ueditor上传word图片 #YOLOFuse # Base64编码 # 多模态检测 #IPv6 #DNS #进程 #操作系统 #进程创建与终止 #shell #unity3d #服务器框架 #Fantasy #源码 #闲置物品交易系统 #tensorflow #arm #SPA #单页应用 #web3.py #jetty #swagger #visual studio code #java-ee #prompt #YOLOv8 # Docker镜像 #麒麟OS #计算机 #mamba #信息可视化 #claude code #codex #code cli #ccusage #Ascend #MindIE # 高并发部署 #journalctl #epoll #sqlite #webpack #openresty #学术写作辅助 #论文创作效率提升 #AI写论文实测 #wordpress #雨云 #电气工程 #C# #PLC #jupyter #经验分享 #安卓 #负载均衡 #sql #intellij-idea #大模型部署 #mindie #大模型推理 #业界资讯 #实时音视频 #Go并发 #高并发架构 #Goroutine #系统设计 #Dify #ARM架构 #鲲鹏 #.net #net core #kestrel #web-server #asp.net-core #langchain #大模型开发 #程序员 #SSH Agent Forwarding # 容器化 #数据分析 #SSH反向隧道 # Miniconda # Jupyter远程访问 #Host #渗透测试 #SSRF #集成学习 #serverless #EMC存储 #存储维护 #NetApp存储 #简单数论 #埃氏筛法 #openEuler #Hadoop #eureka #C语言 #mongodb #cursor #yum #uvicorn #uvloop #asgi #event #https #homelab #Lattepanda #Jellyfin #Plex #Emby #Kodi #PTP_1588 #gPTP #rtsp #转发 #Termux #Samba #Linux #三维 #3D #三维重建 #分类 #CVE-2025-61686 #漏洞 #路径遍历高危漏洞 #neo4j #NoSQL #SQL #Llama-Factory # 大模型推理 #rust #fpga开发 #进程等待 #wait #waitpid #HeyGem # 服务器IP # 端口7860 # REST API # GLM-4.6V-Flash-WEB #pdf #大模型教程 #AI大模型 # keep-alive #echarts #GPU #AutoDL ##租显卡 # GPU租赁 # 自建服务器 #web服务器 #clickhouse #iphone #H5网页 #网页白屏 #H5页面空白 #资源加载问题 #打包部署后网页打不开 #HBuilderX #CTF #VMWare Tool #AI智能体 #MinIO服务器启动与配置详解 #OPCUA #心理健康服务平台 #心理健康系统 #心理服务平台 #心理健康小程序 #arm64 #esp32教程 #插件 #开源软件 #DHCP #agent #ai大模型 #模版 #函数 #类 #笔试 #agi #串口服务器 #Modbus #MOXA #GATT服务器 #蓝牙低功耗 #ms-swift # 一锤定音 # 大模型微调 #散列表 #哈希算法 #adb #青少年编程 #论文笔记 #硬件 #cesium #可视化 #dify #信号处理 #PowerBI #企业 #idea #intellij idea #CPU利用率 #c++20 #googlecloud #5G #vnstat #memcache #文心一言 #大剑师 #nodejs面试题 #vp9 #C2000 #TI #实时控制MCU #AI服务器电源 #前端框架 #reactjs #web3 #攻防演练 #Java web #红队 #leetcode # 树莓派 # ARM架构 # 自动化部署 # VibeThinker #Aluminium #Google #GB28181 #SIP信令 #SpringBoot #视频监控 #WT-2026-0001 #QVD-2026-4572 #smartermail #银河麒麟操作系统 #openssh #华为交换机 #信创终端 #UDP的API使用 #硬件工程 #处理器 #飞牛nas #fnos #Ubuntu服务器 #硬盘扩容 #命令行操作 #智能体来了 #智能体对传统行业冲击 #行业转型 #AI赋能 #系统管理 #服务 #Modbus-TCP #Socket网络编程 #管道Pipe #system V #RAG #LLM #chat #azure #远程开发 #SAP #ebs #metaerp #oracle ebs #muduo库 # 高并发 #YOLO26 #目标检测 #muduo #TcpServer #accept #高并发服务器 #ShaderGraph #图形 #VMware Workstation16 #服务器操作系统 #国产化OS #postman #html5 #计算几何 #斜率 #方向归一化 #叉积 #copilot # 批量管理 #chrome #微PE #硬盘克隆 #DiskGenius #媒体 #Anaconda配置云虚拟环境 #机器学习 #交通物流 #vivado license #MS #Materials #手机h5网页浏览器 #安卓app #苹果ios APP #手机电脑开启摄像头并排查 #IO #fabric #可信计算技术 #能源 #openHiTLS #TLCP #DTLCP #商用密码算法 #ONLYOFFICE #MCP 服务器 #ArkUI #ArkTS #鸿蒙开发 #Nacos #服务器繁忙 #powerbi #go #CPU #测评 #CCE #Dify-LLM #Flexus #puppeteer #KMS #slmgr #数据仓库 #spine #智能家居 #POC #问答 #交付 #xlwings #Excel #mybatis #翻译 #推荐算法 #spring cloud #bootstrap #nfs #iscsi #文件管理 #文件服务器 #信息与通信 #tcpdump #embedding #kmeans #聚类 #文件IO #输入输出流 #log #系统安全 # 大模型 # 模型训练 #scanf #printf #getchar #putchar #cin #cout #大语言模型 #Java #浏览器自动化 #python #企业级存储 #网络设备 #Smokeping #pve #LangGraph #CLI #Python #JavaScript #langgraph.json #SSH免密登录 # CUDA #程序人生 #蓝桥杯 #paddleocr #vps #zotero #WebDAV #同步失败 #代理模式 #工具集 #大模型应用 #API调用 #PyInstaller打包运行 #服务端部署 #ARM服务器 # 多模态推理 #排序算法 #jdk #排序 #统信UOS #win10 #qemu #pencil #pencil.dev #设计 #ddos #欧拉 #aiohttp #asyncio #异步 #软件 #本地生活 #电商系统 #商城 #.netcore # IndexTTS 2.0 # 自动化运维 #全链路优化 #实战教程 #win11 #儿童AI #图像生成 #星图GPU # 模型微调 #LobeChat #vLLM #GPU加速 #麒麟 #SSH保活 #rdp #everything #AB包 #海外服务器安装宝塔面板 #ZooKeeper #ZooKeeper面试题 #面试宝典 #深入解析 #ComfyUI # 推理服务器 #n8n解惑 #Tracker 服务器 #响应最快 #torrent 下载 #2026年 #Aria2 可用 #迅雷可用 #BT工具通用 #elk #rabbitmq #esp32 arduino #Zabbix #CosyVoice3 #语音合成 #FASTMCP #高斯溅射 # 语音合成 #产品运营 #Puppet # IndexTTS2 # TTS #联机教程 #局域网联机 #局域网联机教程 #局域网游戏 #模拟退火算法 #hibernate #云服务器 #个人电脑 #广播 #组播 #并发服务器 #x86_64 #数字人系统 #MC群组服务器 #yolov12 #研究生life #gpu #nvcc #cuda #nvidia #其他 #TensorRT # 推理优化 #unix #C/C++ #编程 #c++高并发 #百万并发 #机器人 #es安装 #zabbix #CS2 #debian13 #信创国产化 #达梦数据库 #RXT4090显卡 #RTX4090 #深度学习服务器 #硬件选型 #模型训练 #IntelliJ IDEA #Spring Boot #SQL注入主机 #uip #k8s #wsl #Jetty # CosyVoice3 # 嵌入式服务器 #树莓派 #温湿度监控 #WhatsApp通知 #IoT #MySQL #建筑缺陷 #红外 #数据集 #LangFlow # 智能运维 # 性能瓶颈分析 #VibeVoice # 云服务器 #devops #戴尔服务器 #戴尔730 #装系统 #SMTP # 内容安全 # Qwen3Guard #junit #黑客技术 #文件上传漏洞 #ThingsBoard MCP #Kylin-Server #国产操作系统 #服务器安装 #代理 #平板 #零售 #智能硬件 #vncdotool #链接VNC服务器 #如何隐藏光标 # 服务器IP访问 # 端口映射 #A2A #GenAI #gateway #Comate #遛狗 #bug #自动化运维 #FHSS #UOS #海光K100 #统信 #算力建设 #数据安全 #注入漏洞 #duckdb #wpf #服务器解析漏洞 #nodejs #SSH密钥 # ControlMaster #练习 #基础练习 #数组 #循环 #九九乘法表 #计算机实现 #dynadot #域名 #ETL管道 #向量存储 #数据预处理 #DocumentReader #esb接口 #走处理类报异常 #firefox #safari #ffmpeg #le audio #蓝牙 #低功耗音频 #通信 #连接 #memory mcp #Cursor #网路编程 #smtp #smtp服务器 #PHP #银河麒麟部署 #银河麒麟部署文档 #银河麒麟linux #银河麒麟linux部署教程 #Qwen3-14B # 大模型部署 # 私有化AI # 远程连接 #AI 推理 #NV #ServBay #SFTP #指针 #anaconda #虚拟环境 #ranger #MySQL8.0 #SSH跳板机 # Python3.11 #LVDS #高速ADC #DDR #AI技术 #word #驱动开发 #ESP32 # OTA升级 # 黄山派 #ansys #ansys问题解决办法 # WebUI # 网络延迟 #screen命令 #mariadb # Connection refused #Gunicorn #WSGI #Flask #并发模型 #容器化 #性能调优 #视频 #ceph #ambari #门禁 #梯控 #智能一卡通 #门禁一卡通 #消费一卡通 #智能梯控 #一卡通 #源代码管理 #超时设置 #客户端/服务器 #网络编程 #挖矿 #Linux病毒 #sql注入 #ai编程 #excel #雨云服务器 #Minecraft服务器 #教程 #MCSM面板 #鸿蒙 # 服务器配置 # GPU #VPS #搭建 #数据恢复 #视频恢复 #视频修复 #RAID5恢复 #流媒体服务器恢复 #智慧校园解决方案 #智慧校园一体化平台 #智慧校园选型 #智慧校园采购 #智慧校园软件 #智慧校园专项资金 #智慧校园定制开发 #模型上下文协议 #MultiServerMCPC #load_mcp_tools #load_mcp_prompt #Gateway #认证服务器集成详解 #框架搭建 #状态模式 #AI-native #dba #Tokio #react native #weston #x11 #x11显示服务器 #WinSCP 下载安装教程 #FTP工具 #服务器文件传输 # 批量部署 #RSO #机器人操作系统 #ASR #SenseVoice # TTS服务器 # 键鼠锁定 #glibc #中间件 #远程连接 #CVE-2025-68143 #CVE-2025-68144 #CVE-2025-68145 #工程设计 #预混 #扩散 #燃烧知识 #层流 #湍流 #证书 #scrapy #AI写作 #winscp #markdown #建站 #后端框架 #node #参数估计 #矩估计 #概率论 # 数字人系统 # 远程部署 #ue5 #LE Audio #BAP #Docker #Node.js # child_process #策略模式 #matlab #bond #服务器链路聚合 #网卡绑定 #性能测试 #LoadRunner #仙盟创梦IDE #运维工具 #GLM-4.6V-Flash-WEB # AI视觉 # 本地部署 #网络攻击模型 #动态规划 #pyqt #dlms #dlms协议 #逻辑设备 #逻辑设置间权限 #scikit-learn #随机森林 #安全威胁分析 #Minecraft #PaperMC #我的世界服务器 #ipmitool #BMC # 黑屏模式 #C #STDIO传输 #SSE传输 #WebMVC #WebFlux #企业微信 #3d #IndexTTS2 # 阿里云安骑士 # 木马查杀 #kong #Kong Audio #Kong Audio3 #KongAudio3 #空音3 #空音 #中国民乐 #人大金仓 #Kingbase #小艺 #搜索 #Spring AOP #计组 #数电 #多模态 #微调 #超参 #LLamafactory #生信 #产品经理 #就业 #Xshell #Finalshell #生物信息学 #组学 #CMake #Make #多进程 #python技巧 #V11 #kylinos #raid #raid阵列 #上下文工程 #langgraph #意图识别 #KMS激活 #java大文件上传 #java大文件秒传 #java大文件上传下载 #java文件传输解决方案 #Java程序员 #Java面试 #后端开发 #Spring源码 #Spring #numpy #CSDN #论文阅读 #软件工程 #Langchain-Chatchat # 国产化服务器 # 信创 #视觉检测 #visual studio #传感器 #MicroPython #RK3576 #瑞芯微 #硬件设计 #PyCharm # 远程调试 # YOLOFuse #gRPC #注册中心 #database #pjsip #iot #智慧城市 #数字化转型 #实体经济 #商业模式 #软件开发 #数智红包 #商业变革 #创业干货 #c #人脸识别sdk #视频编解码 #人脸识别 #开源工具 #sglang #防毒面罩 #防尘面罩 #编程助手 #m3u8 #HLS #移动端H5网页 #APP安卓苹果ios #监控画面 直播视频流 #Prometheus #挖漏洞 #日志分析 #勒索病毒 #勒索软件 #加密算法 #.bixi勒索病毒 #数据加密 #决策树 #CA证书 #HistoryServer #Spark #YARN #jobhistory #DooTask #交换机 #三层交换机 #UEFI #BIOS #Legacy BIOS #内存接口 # 澜起科技 # 服务器主板 # 显卡驱动备份 #Socket #计算机毕业设计 #程序定制 #毕设代做 #课设 #webgl #区块链 #性能 #优化 #RAM #KMS 激活 #AI智能棋盘 #Rock Pi S #边缘计算 #wireshark #nacos #银河麒麟aarch64 # 服务器迁移 # 回滚方案 #大模型入门 #Keycloak #Quarkus #AI编程需求分析 #开关电源 #热敏电阻 #PTC热敏电阻 #文件传输 #电脑文件传输 #电脑传输文件 #电脑怎么传输文件到另一台电脑 #电脑传输文件到另一台电脑 #身体实验室 #健康认知重构 #系统思维 #微行动 #NEAT效应 #亚健康自救 #ICT人 #云开发 #asp.net上传大文件 #测速 #iperf #iperf3 #漏洞挖掘 #gpt #SSH别名 #BoringSSL #企业存储 #RustFS #对象存储 #高可用 #云计算运维 #模块 # 公钥认证 # 权限修复 #STUN #turn #ICE #群晖 #目标跟踪 #音乐 # ARM服务器 # 鲲鹏 #http头信息 #Ubuntu #ESP32编译服务器 #Ping #DNS域名解析 #Coturn #TURN #log4j # HiChatBox # 离线AI #TCP服务器 #开发实战 #SMARC #ARM #全文检索 #银河麒麟服务器系统 # 代理转发 # 跳板机 #面向对象 #基础语法 #标识符 #常量与变量 #数据类型 #运算符与表达式 #空间计算 #原型模式 #nosql #汽车 #可撤销IBE #服务器辅助 #私钥更新 #安全性证明 #双线性Diffie-Hellman #主板 #总体设计 #电源树 #框图 #Reactor #短剧 #短剧小程序 #短剧系统 #微剧 #数学建模 #数模美赛 #数据访问 #传统行业 #I/O模型 #并发 #水平触发、边缘触发 #多路复用 #CNAS #CMA #程序文件 #SSH复用 # 远程开发 #磁盘配额 #存储管理 #形考作业 #国家开放大学 #系统运维 #网络安全大赛 #C++ UA Server #SDK #跨平台开发 #eclipse #servlet #机器视觉 #6D位姿 #outlook #错误代码2603 #无网络连接 #2603 #mssql #实时检测 #卷积神经网络 #lucene #DAG #云服务器选购 #Saas #线程 #b树 #具身智能 #Fun-ASR # 语音识别 #HarmonyOS APP #密码 #AI电商客服 #spring ai #oauth2 #数据可视化 #nmodbus4类库使用教程 #docker-compose #rtmp #coffeescript #声源定位 #MUSIC #windbg分析蓝屏教程 #寄存器 #Buck #NVIDIA #算力 #交错并联 #DGX #ROS # 局域网访问 # 批量处理 #内存治理 #IFix # 高温监控 #fs7TF # 远程访问 #npu # 环境迁移 #matplotlib #安全架构 #gerrit #opc ua #opc #远程软件 # GLM-TTS # 数据安全 #xshell #host key #TTS私有化 # IndexTTS # 音色克隆 #内网 #分布式数据库 #集中式数据库 #业务需求 #选型误 #代理服务器 #rsync # 数据同步 #ip #blender #设计师 #图像处理 #游戏美术 #技术美术 #多线程 #claudeCode #content7 #跳槽 #工作 #网安应急响应 #odoo # GLM # 服务连通性 #HarmonyOS #Apple AI #Apple 人工智能 #FoundationModel #Summarize #SwiftUI #支持向量机 # 串口服务器 # NPort5630 #appche # GPU集群 #服务器开启 TLS v1.2 #IISCrypto 使用教程 #TLS 协议配置 #IIS 安全设置 #服务器运维工具 #ftp #sftp #uniapp #合法域名校验出错 #服务器域名配置不生效 #request域名配置 #已经配置好了但还是报错 #uniapp微信小程序 #YOLO识别 #YOLO环境搭建Windows #YOLO环境搭建Ubuntu # 轻量化镜像 # 边缘计算 #华为od #华为机试 #OpenHarmony #Python办公自动化 #Python办公 #SSH跳转 #TTS #量子计算 #samba #mtgsig #美团医药 #美团医药mtgsig #美团医药mtgsig1.2 #opc模拟服务器 #cpu #套接字 #I/O多路复用 #字节序 #报表制作 #职场 #用数据讲故事 #语音生成 #统信操作系统 #AI部署 # ms-swift #PN 结 #服务器线程 # SSL通信 # 动态结构体 #RWK35xx #语音流 #实时传输 #游戏策划 #游戏程序 #用户体验 #超算中心 #PBS #lsf #TLS协议 #HTTPS #漏洞修复 #运维安全 #lvs #adobe # TURN # NAT穿透 #MCP服务器注解 #异步支持 #方法筛选 #声明式编程 #自动筛选机制 #数据迁移 #JNI #pxe #express #cherry studio #程序开发 #程序设计 #大作业 #2026年美赛C题代码 #2026年美赛 #gmssh #宝塔 #Exchange #free #vmstat #sar #sentinel #宝塔面板部署RustDesk #RustDesk远程控制手机 #手机远程控制 #系统安装 #铁路桥梁 #DIC技术 #箱梁试验 #裂纹监测 #四点弯曲 #可再生能源 #绿色算力 #风电 # 远程运维 #MinIO #数据报系统 #麦克风权限 #访问麦克风并录制音频 #麦克风录制音频后在线播放 #用户拒绝访问麦克风权限怎么办 #uniapp 安卓 苹果ios #将音频保存本地或上传服务器 #若依 #TRO #TRO侵权 #TRO和解 #智能制造 #供应链管理 #工业工程 #库存管理 #Discord机器人 #云部署 #程序那些事 #AI应用编程 #r语言 #服务器IO模型 #非阻塞轮询模型 #多任务并发模型 #异步信号模型 #多路复用模型 #前端开发 #EN4FE #领域驱动 #自由表达演说平台 #演说 #海外短剧 #海外短剧app开发 #海外短剧系统开发 #短剧APP #短剧APP开发 #短剧系统开发 #海外短剧项目 #移动端h5网页 #调用浏览器摄像头并拍照 #开启摄像头权限 #拍照后查看与上传服务器端 #摄像头黑屏打不开问题 #AI Agent #开发者工具 #国产开源制品管理工具 #Hadess #一文上手 #工业级串口服务器 #串口转以太网 #串口设备联网通讯模块 #串口服务器选型 #okhttp #范式 #晶振 #计算机外设 #WinDbg #Windows调试 #内存转储分析 #入侵 #日志排查 #Karalon #AI Test #流程图 #图论 #健康医疗 #ET模式 #非阻塞 #remote-ssh #工程实践 #租显卡 #训练推理 #AI应用 #图像识别 #高考 #轻量化 #低配服务器 #Beidou #北斗 #SSR #Anything-LLM #IDC服务器 #私有化部署 #clawdbot #API #taro #wps #Linux多线程 #bigtop #hdp #hue #kerberos #simulink #Matrox MIL #二次开发 #docker安装seata #vertx #vert.x #vertx4 #runOnContext #信息安全 #信息收集 #poll #WRF #WRFDA #VoxCPM-1.5-TTS # 云端GPU # PyCharm宕机 #重构 #机器人学习 #Syslog #系统日志 #日志监控 #生产服务器问题查询 #日志过滤 #Autodl私有云 #深度服务器配置 # 水冷服务器 # 风冷服务器 #视觉理解 #Moondream2 #多模态AI # AI部署 #材料工程 #智能电视 #VMware创建虚拟机 #远程更新 #缓存更新 #多指令适配 #物料关联计划 #AI生成 # outputs目录 # 自动化 #攻击溯源 #stl #IIS Crypto #warp #CS336 #Assignment #Experiments #TinyStories #Ablation #UDP套接字编程 #UDP协议 #网络测试 #二值化 #Canny边缘检测 #轮廓检测 #透视变换 #vue上传解决方案 #vue断点续传 #vue分片上传下载 #vue分块上传下载 #rag #ossinsight #AE #jquery #节日 #fork函数 #进程创建 #进程终止 #分子动力学 #化工仿真 #小智 #session #游戏服务器断线 #期刊 #SCI #Linly-Talker # 数字人 # 服务器稳定性 #JADX-AI 插件 #starrocks #电子电气架构 #系统工程与系统架构的内涵 #自动驾驶 #Routine #OSS #L6 #L10 #L9 #阿里云RDS #国产PLM #瑞华丽 #瑞华丽PLM #PLM #composer #symfony #java-zookeeper #软件需求 #项目申报系统 #项目申报管理 #项目申报 #企业项目申报 #H3C #dubbo #tcp/ip #网络 #新浪微博 #传媒 #DuckDB #协议 #汇编 #高仿永硕E盘的个人网盘系统源码 #Ward #思爱普 #SAP S/4HANA #ABAP #NetWeaver #typescript #npm #xss #日志模块 #音诺ai翻译机 #AI翻译机 # Ampere Altra Max #考研 #文生视频 #WAN2.2 #AI视频生成 #反向代理 #Arduino BLDC #核辐射区域探测机器人 #电梯 #电梯运力 #电梯门禁 # GPU服务器 # tmux #idc #esp32 #mosquito #静脉曲张 #腿部健康 #运动 #RK3588 #RK3588J #评估板 #核心板 #嵌入式开发 #边缘AI # Kontron # SMARC-sAMX8 #resnet50 #分类识别训练 #运维 #AI视频创作系统 #AI视频创作 #AI创作系统 #AI工具 #AI创作工具 #华为od机试 #华为od机考 #华为od最新上机考试题库 #华为OD题库 #华为OD机试双机位C卷 #od机考题库 #Python3.11 #OpenAI #故障 #FRP #AI工具集成 #容器化部署 #分布式架构 #2025年 #数据采集 #浏览器指纹 #edge #迭代器模式 #观察者模式 # IP配置 # 0.0.0.0 #自动化巡检 #VSCode # Qwen3Guard-Gen-8B #0day漏洞 #DDoS攻击 #漏洞排查 #restful #ajax #ipv6