最新资讯

  • Linux:TCP和守护进程

Linux:TCP和守护进程

2025-05-17 21:00:48 1 阅读

一、TCP网络程序

1.1 TCP服务端

成员变量:

int _listensock; // 监听的文件描述符

string _ip;      // 服务端ip

uint16_t _port;  // 端口号

bool _isrunning; // 服务器是否在运行

1.1.1 InitServer-创建服务端

1、创建套接字socket

socket()打开一个网络通讯端口,如果成功的话,就像open()一样返回一个文件描述符;

应用程序可以像读写文件一样用read/write在网络上收发数据;

如果socket()调用出错则返回-1;

对于IPv4, family参数指定为AF_INET;

对于TCP协议,type参数指定为SOCK_STREAM, 表示面向流的传输协议

protocol参数的介绍从略,指定为0即可。

2、绑定套接字bind

服务器程序所监听的网络地址和端口号通常是固定不变的,客户端程序得知服务器程序的地址和端口号后 就可以向服务器发起连接; 服务器需要调用bind绑定一个固定的网络地址和端口号;

bind()成功返回0,失败返回-1。

bind()的作用是将参数sockfd和myaddr绑定在一起, 使sockfd这个用于网络通讯的文件描述符监听myaddr所描述的地址和端口号;

struct sockaddr *是一个通用指针类型,myaddr参数实际上可以接受多种协议的sockaddr结 构体,而它们的长度各不相同,所以需要第三个参数addrlen指定结构体的长度;

我们的程序中对myaddr参数是这样初始化的:

注意:其实大部分的接口都已经帮我们考虑到大小端的问题了,只不过ip和port需要我们写到OS里,所以需要我们自己去转化!! 

(1)将整个结构体清零;

(2)设置地址类型为AF_INET;

(3)网络地址为INADDR_ANY(或者是0.0.0.0), 这个宏表示本地的任意IP地址,因为服务器可能有多个网卡,每个网卡也可能绑 定多个IP地址, 这样设置可以在所有的IP地址上监听,直到与某个客户端建立了连接时才确定下来到底用哪个IP地址;

(4)端口号为SERV_PORT, 我们定义为8080;

这两步和之前UDP的基本一样

3、TCP是面向连接的,服务器一般都是一个比较被动的状态,要等待客户端和他建立连接关系。所以他需要不断保持一个监听的状态 listen

 listen()声明sockfd处于监听状态, 并且最多允许有backlog个客户端处于连接等待状态, 如果接收到更多 的连接请求就忽略, 这里设置不会太大(一般是5), 

listen()成功返回0,失败返回-1;

全部代码:

void InitServer() // 创建服务器
  {
    // 1/创建套接字
    _listensock = socket(AF_INET, SOCK_STREAM, 0); // 面向字节流;
    if (_listensock < 0)
    {
      lg(Fatal, "create socket,errno:%d,errstring:%s", errno, strerror(errno));
      exit(SocketError);
    }
    lg(Info, "create socket success,_listsock:%d", _listensock);
    // 2/开始绑定
    struct sockaddr_in local;
    bzero(&local, sizeof(local)); // 先清空,然后再填进去
    local.sin_family = AF_INET;
    local.sin_port = htons(_port); // 转网络序列
    inet_aton(_ip.c_str(), &local.sin_addr);
    // 开始绑定
    if (bind(_listensock, (sockaddr *)&local, sizeof(local)) < 0) // 如果绑定失败
    {
      lg(Fatal, "bind errno,errno:%d,errstring:%s", errno, strerror(errno));
      exit(BindError);
    }
    lg(Info, "bind socket success,_listsock:%d", _listensock);
    // 3/tcp和udp的区别就是要面向连接 要被动地等待别人来连接
    if (listen(_listensock, backlog) < 0)
    {
      lg(Fatal, "listen errno,errno:%d,errstring:%s", errno, strerror(errno));
    }
    lg(Info, "listen socket success,_listsock:%d", _listensock);
  }

1.1.2 Run-运行服务器(单进程版)

1、接收客户端的连接请求 accept

三次握手完成后, 服务器调用accept()接受连接;

如果服务器调用accept()时还没有客户端的连接请求,就阻塞等待直到有客户端连接上来;

addr是一个传出参数,accept()返回时传出客户端的地址和端口号;

如果给addr 参数传NULL,表示不关心客户端的地址;

addrlen参数是一个传入传出参数(value-result argument), 传入的是调用者提供的, 缓冲区addr的长度 以避免缓冲区溢出问题, 传出的是客户端地址结构体的实际长度(有可能没有占满调用者提供的缓冲区);

 我们的服务器程序结构是这样的: 

accept的返回值是什么意义??

      我们会发现accept返回的也是一个文件描述符,那么这个文件描述符跟我们刚开始的那个listensock是什么关系呢??

举例子:假如我们在假期去一个旅游圣地,到中午的时候很多当地的餐馆为了生意都会安排人站在外头去拉客,比方说你们一行人走到一家鱼庄面前,这时候鱼庄门口有个张三马上靠过来开始给你介绍这里餐馆的特色,邀请你去鱼庄吃饭,这个时候你们正好也饿了于是就进去了 ,但是你们进去的时候张三并没有进去,而是喊了李四这个服务员过来给你们服务,自己又继续出去拉客了,后来张三又不断拉来新的客人,又不断有王五,赵六……服务员也出来了,所以张三只关注于拉客,而李四等服务员只提供服务,他们各自专注着自己的事情却可以把整个鱼庄经营得非常好!!!

     所以我们之前的listensock就是我们的张三,他就是专门负责和客户端建立连接的,而accept返回值的sockfd是就相当于是我们的李四,是专门给客户端提供服务的!

2、开始给客户端提供服务 

我们用telnet模拟客户端就可以进行测试了!!

      但是这样有一个很尴尬的地方就是,我们当前这个单进程版的必须等到这个服务结束了才会去进行下一个客户端的连接,这样显然是不符合我们的要求的!(有点像客流量很多 但是餐馆只有一张桌子  这样效率很低!!)!所以我们接下来要尝试 多进程版、多线程版、线程池版

1.1.3 Run-运行服务器(多进程版)

多进程思路:让子进程替我去完成工作,而我继续去响应链接!!

(1)父进程把文件描述符表拷贝给子进程后,父进程就要把sockfd给关了(让服务完全由子进程去做,如果子进程退出了意味着服务结束,这样正好可以把这个文件给关了) 而子进程可以把listensockfd给关了(让链接完全由父进程去做,防止子进程误操作)

(2)但是如果父进程阻塞等待的话,又会和单进程一样,而如果用非阻塞轮询又会浪费cpu资源,且增加程序设计的复杂性  所以我们要思考其他办法!

 方法1:让孙子进程去做 然后子进程退出 这样的话父进程立马可以返回 而孙子进程会被系统领养  由系统回收

方法2:直接将SIGCHLD信号设成SIG_IGN 那么系统就不会把退出的进程转成僵尸进程

1.1.4 Run-运行服务器 (多线程版)

但是多进程太耗费资源了!!所以我们应该考虑多线程版!!

(1)多线程不需要关闭文件描述符 因为是共享的所以没有多余的

(2)如果join的话又会阻塞住,所以我们可以直接将线程给分离了,这样主线程就不关心了!

(3)定义一个类将属性传给线程 

 如果线程调用的函数写在里面,默认有this指针,所以必须把他设置成静态成员函数!!

但是静态成员函数不能调用非静态成员函数,所以我们可以把对象指针传进去,通过这个对象指针来调用成员函数。

1.1.5 Run-运行服务器(线程池版汉英翻译)

      我不希望你客户端连接成功后我才去创建一个线程,而一个客户断开了又得释放进程,这样效率太低了!!而且我不想给你提供这种长服务(就是你当前客户端如果请求太多的话,我就得一直专门服务你,这就是长服务),而多线程并不适合长服务,因为线程的个数是确定的,不能让你一直给一个客户端服务,所以我们要尝试把他修改成短服务(就是这个客户端一旦接受了你的一次请求他就断掉  继续去服务别的客户端)!!

     所以我们(1)一方面需要通过线程池来避免线程被重复创建和释放的过程,(2)另一方面把长服务设置成短服务!(3)然后将具体任务封装起来交给线程池去完成,这样可以解耦

 我们可以将这个任务设置成英汉翻译

dict.txt

apple:苹果...
banana:香蕉...
red:红色...
yellow:黄色...
the: 这
be: 是
to: 朝向/给/对
and: 和
I: 我
in: 在...里
that: 那个
have: 有
will: 将
for: 为了
but: 但是
as: 像...一样
what: 什么
so: 因此
he: 他
her: 她
his: 他的
they: 他们
we: 我们
their: 他们的
his: 它的
with: 和...一起
she: 她
he: 他(宾格)
it: 它

 Init.hpp(读取一个文件 然后分割到哈希表中)

#pragma once

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

const std::string dictname = "./dict.txt";
const std::string sep = ":";

//yellow:黄色...
static bool Split(std::string &s, std::string *part1, std::string *part2)
{
    auto pos = s.find(sep);
    if(pos == std::string::npos) return false;
    *part1 = s.substr(0, pos);
    *part2 = s.substr(pos+1);
    return true;
}

class Init
{
public:
    Init()
    {
        std::ifstream in(dictname);
        if(!in.is_open())
        {
            lg(Fatal, "ifstream open %s error", dictname.c_str());
            exit(1);
        }
        std::string line;
        while(std::getline(in, line))
        {
            std::string part1, part2;
            Split(line, &part1, &part2);
            dict.insert({part1, part2});
        }
        in.close();
    }
    std::string translation(const std::string &key)
    {
        auto iter = dict.find(key);
        if(iter == dict.end()) return "Unknow";
        else return iter->second;
    }
private:
    std::unordered_map dict;
};

Task.hpp 任务

#pragma once
#include 
#include 
#include "Log.hpp"
#include "Init.hpp"

extern Log lg;
Init init;

class Task
{
public:
    Task(int sockfd, const std::string &clientip, const uint16_t &clientport)
        : sockfd_(sockfd), clientip_(clientip), clientport_(clientport)
    {
    }
    Task()
    {
    }
    void run()
    {
        // 测试代码
        char buffer[4096];
        // Tcp是面向字节流的,你怎么保证,你读取上来的数据,是"一个" "完整" 的报文呢?
        ssize_t n = read(sockfd_, buffer, sizeof(buffer)); // BUG?
        if (n > 0)
        {
            buffer[n] = 0;
            std::cout << "client key# " << buffer << std::endl;
            std::string echo_string = init.translation(buffer);

            // sleep(5);
            // // close(sockfd_);
            // lg(Warning, "close sockfd %d done", sockfd_);

            // sleep(2);
            n = write(sockfd_, echo_string.c_str(), echo_string.size()); // 100 fd 不存在
            if(n < 0)
            {
                lg(Warning, "write error, errno : %d, errstring: %s", errno, strerror(errno));
            }
        }
        else if (n == 0)
        {
            lg(Info, "%s:%d quit, server close sockfd: %d", clientip_.c_str(), clientport_, sockfd_);
        }
        else
        {
            lg(Warning, "read error, sockfd: %d, client ip: %s, client port: %d", sockfd_, clientip_.c_str(), clientport_);
        }
        close(sockfd_);
    }
    void operator()()
    {
        run();
    }
    ~Task()
    {
    }

private:
    int sockfd_;
    std::string clientip_;
    uint16_t clientport_;
};

线程池ThreadPool:

#pragma once

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

struct ThreadInfo
{
    pthread_t tid;
    std::string name;
};

static const int defalutnum = 10;

template 
class ThreadPool
{
public:
    void Lock()
    {
        pthread_mutex_lock(&mutex_);
    }
    void Unlock()
    {
        pthread_mutex_unlock(&mutex_);
    }
    void Wakeup()
    {
        pthread_cond_signal(&cond_);
    }
    void ThreadSleep()
    {
        pthread_cond_wait(&cond_, &mutex_);
    }
    bool IsQueueEmpty()
    {
        return tasks_.empty();
    }
    std::string GetThreadName(pthread_t tid)
    {
        for (const auto &ti : threads_)
        {
            if (ti.tid == tid)
                return ti.name;
        }
        return "None";
    }

public:
    static void *HandlerTask(void *args)
    {
        ThreadPool *tp = static_cast *>(args);
        std::string name = tp->GetThreadName(pthread_self());
        while (true)
        {
            tp->Lock();

            while (tp->IsQueueEmpty())
            {
                tp->ThreadSleep();
            }
            T t = tp->Pop();
            tp->Unlock();

            t();
        }
    }
    void Start()
    {
        int num = threads_.size();
        for (int i = 0; i < num; i++)
        {
            threads_[i].name = "thread-" + std::to_string(i + 1);
            pthread_create(&(threads_[i].tid), nullptr, HandlerTask, this);
        }
    }
    T Pop()
    {
        T t = tasks_.front();
        tasks_.pop();
        return t;
    }
    void Push(const T &t)
    {
        Lock();
        tasks_.push(t);
        Wakeup();
        Unlock();
    }
    static ThreadPool *GetInstance()
    {
        if (nullptr == tp_) // ???
        {
            pthread_mutex_lock(&lock_);
            if (nullptr == tp_)
            {
                std::cout << "log: singleton create done first!" << std::endl;
                tp_ = new ThreadPool();
            }
            pthread_mutex_unlock(&lock_);
        }

        return tp_;
    }

private:
    ThreadPool(int num = defalutnum) : threads_(num)
    {
        pthread_mutex_init(&mutex_, nullptr);
        pthread_cond_init(&cond_, nullptr);
    }
    ~ThreadPool()
    {
        pthread_mutex_destroy(&mutex_);
        pthread_cond_destroy(&cond_);
    }
    ThreadPool(const ThreadPool &) = delete;
    const ThreadPool &operator=(const ThreadPool &) = delete; // a=b=c
private:
    std::vector threads_;
    std::queue tasks_;

    pthread_mutex_t mutex_;
    pthread_cond_t cond_;

    static ThreadPool *tp_;
    static pthread_mutex_t lock_;
};

template 
ThreadPool *ThreadPool::tp_ = nullptr;

template 
pthread_mutex_t ThreadPool::lock_ = PTHREAD_MUTEX_INITIALIZER;

1.1.6 服务端写入时客户端退出了怎么办

对一个对端已经关闭的socket调用两次write, 第二次将会生成SIGPIPE信号, 该信号默认结束进程.

SIGPIPE信号详解 - 冷冰若水 - 博客园

1.1.7 服务端全部代码 

#include 
#include 
#include 
#include 
#include  //套接字类型的头文件
#include     //bzero的头文件
#include 
#include 
#include "Log.hpp"
#include 
#include 
#include 
#include 
#include"ThreadPool.hpp"
#include "Task.hpp"
using namespace std;

typedef function func_t;
Log lg; // 命令对象 用来打印日志信息
enum
{
  UsageError = 1, // 使用有误
  SocketError,    // 创建套接字有误
  BindError,      // 绑定有误
  ListenError,    // 监听有误
};

const int defaultfd = -1;
const uint16_t defaultport = 8080;
const string defaultip = "0.0.0.0";
const int size = 1024;
const int backlog = 10; // 但是一般不要设置的太大

class ThreadData
{
public:
  ThreadData(int fd, uint16_t &port, const string &ip, TcpServer *t) : sockfd(fd), clientport(port),clientip(ip), tsvr(t)
  {
  }
  public:
  int sockfd;
  string clientip;
  uint16_t clientport;
  TcpServer *tsvr;//通过对象让静态成员函数调用 非静态成员方法
};

class TcpServer
{
public:
  TcpServer(uint16_t &port, const string &ip = defaultip) : _listensock(defaultfd), _port(port), _ip(ip), _isrunning(false)
  {
  }

  static void *Routine(void *args)
  {
    pthread_detach(pthread_self());
    ThreadData *td = static_cast(args);
    td->tsvr->Service(td->sockfd, td->clientport, td->clientip); //通过传一个对象指针来调用类内的成员函数
    delete td;
    return nullptr;
  }

  void Service(int sockfd, uint16_t clientport, string clientip)
  {
    char buffer[size];
    while (true)
    {
      // 服务端要先接收客户端的数据
      ssize_t n = read(sockfd, buffer, sizeof(buffer)); // 读到缓冲区中
      if (n > 0)                                        // 大于0表示读取成功 =0表示当前没有数据可读了  <0说明读取失败
      {
        buffer[n] = 0; // 把我们读到的信息当成是字符串的形式来处理
        cout << "client say#" << buffer << endl;
        string echo_string = "tcpserver echo#";
        echo_string += buffer;
        // 对buffer简单加工完之后往客户端写入
        write(sockfd, echo_string.c_str(), echo_string.size());
      }
      else if (n == 0) // 没有什么数据可读的了 所以八成是客户端链接断开了 我服务端不能崩
      {
        lg(Info, "client quit, sockfd: %d, client ip: %s, client port: %d", sockfd, clientip.c_str(), clientport);
        break;
      }
      else // 读取失败  可能是文件描述符被关闭了
      {
        lg(Warning, "read error, sockfd: %d, client ip: %s, client port: %d", sockfd, clientip.c_str(), clientport);
      }
    }
  }

  void InitServer() // 创建服务器
  {
    // 1/创建套接字
    _listensock = socket(AF_INET, SOCK_STREAM, 0); // 面向字节流;
    if (_listensock < 0)
    {
      lg(Fatal, "create socket,errno:%d,errstring:%s", errno, strerror(errno));
      exit(SocketError);
    }
    lg(Info, "create socket success,_listsock:%d", _listensock);
    // 2/开始绑定
    struct sockaddr_in local;
    bzero(&local, sizeof(local)); // 先清空,然后再填进去
    local.sin_family = AF_INET;
    local.sin_port = htons(_port); // 转网络序列
    inet_aton(_ip.c_str(), &local.sin_addr);
    // 开始绑定
    if (bind(_listensock, (sockaddr *)&local, sizeof(local)) < 0) // 如果绑定失败
    {
      lg(Fatal, "bind errno,errno:%d,errstring:%s", errno, strerror(errno));
      exit(BindError);
    }
    lg(Info, "bind socket success,_listsock:%d", _listensock);
    // 3/tcp和udp的区别就是要面向连接 要被动地等待别人来连接
    if (listen(_listensock, backlog) < 0)
    {
      lg(Fatal, "listen errno,errno:%d,errstring:%s", errno, strerror(errno));
    }
    lg(Info, "listen socket success,_listsock:%d", _listensock);
  }
  ///
  void Run() // 启动服务器
  {
ThreadPool::GetInstance()->Start();
    signal(SIGPIPE,SIG_IGN);
    _isrunning = true;
    lg(Info, "tcpServer is running....");
    // 1、accept尝试获取新链接
    while (_isrunning) // 不断获取新链接
    {
      struct sockaddr_in client;
      socklen_t len = sizeof(client);
      int sockfd = accept(_listensock, (struct sockaddr *)&client, &len); // 这个id是用来做服务的!!
      if (sockfd < 0)                                                     // 如果获取失败 应该获取下一个 而不是直接结束
      {
        lg(Warning, "accept error, errno: %d, errstring: %s", errno, strerror(errno));
        continue;
      }
      lg(Info, "accept success,sockfd:%d", sockfd);
      // 2、将客户端的信息弄出来
      uint16_t clientport = ntohs(client.sin_port);
      char clientip[32]; // 输出型参数
      inet_ntop(AF_INET, &(client.sin_addr), clientip, sizeof(clientip));
      // 根据新连接来进行通信服务-version单进程版
      // Service(sockfd, clientport, clientip);
      // close(sockfd);
      // version2 多进程版-让子进程帮我做服务,而我继续去链接
      // pid_t id=fork();
      // if(id==0) //child 让他去服务
      // {
      //    close(_listensock);//关掉 防止误操作
      //    if(fork()>0) exit(0);//子进程退掉 让孙子进程来做
      //    Service(sockfd, clientport, clientip);//孙子进程此时已经被 system领养了
      //    close(sockfd);
      //    exit(0);
      // }
      // //father
      // pid_t rid=waitpid(id,nullptr,0);//子进程一进去就退出了,所以父进程会马上返回继续去链接
      // (void)rid;//rid没用过 所以用一下防止警告
      // //signal(SIGCHLD,SIG_IGN);
      // version3 多线程版!!
      // ThreadData *td = new ThreadData(sockfd, clientport, clientip, this);
      // pthread_t tid;
      // pthread_create(&tid, nullptr, Routine, td);
      //version4 线程池英汉词典
       Task t(sockfd, clientip, clientport);
       ThreadPool::GetInstance()->Push(t);
    }
  }
  ~TcpServer()
  {
  }

private:
  int _listensock; // 监听的文件描述符
  string _ip;      // 服务端ip
  uint16_t _port;  // 端口号
  bool _isrunning; // 服务器是否在运行
};

 1.2 客户端

客户端帮我们发送connect请求的时候,会自动bind 

客户端需要调用connect()连接服务器;

connect和bind的参数形式一致, 区别在于bind的参数是自己的地址, 而connect的参数是对方的地址;

connect()成功返回0,出错返回-1; 

单进程版客户端:

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

using namespace std;

void Usage(std::string proc)
{
    std::cout << "

Usage: " << proc << " serverip serverport
"
              << std::endl;
}

// ./tcpclient serverip serverport
int main(int argc,char* argv[]) //必须知道服务器的ip和端口号
{
    if(argc!=3)
    {
      Usage(argv[0]);
      exit(0);
    }
    string serverip = argv[1];
    uint16_t serverport = std::stoi(argv[2]);

    //1、第一步 创建套接字
    int sockfd=socket(AF_INET, SOCK_STREAM, 0);
    if(sockfd<0)
    {
        cerr << "socker error" << endl;
        return 1;
    }
    //2/OS帮助们bind
    struct sockaddr_in server;//输出型参数
    bzero(&server, sizeof(server));
    server.sin_family = AF_INET;
    server.sin_port = htons(serverport); 
   // server.sin_addr.s_addr = inet_addr(serverip.c_str());//字符串转四字节
    inet_pton(AF_INET,serverip.c_str(), &(server.sin_addr));
    socklen_t len = sizeof(server);
    //3、向服务端发送链接请求 
    int n=connect(sockfd,(struct sockaddr*)&server,len);
    if(n<0)//如果链接失败
    {
      cerr<<"connect error……" < 0)
        {
            inbuffer[s] = 0;
            cout << inbuffer << endl;
        }
    } 
    close(sockfd);
}

线程池版英汉翻译客户端:

(1)需要改成短服务,所以链接在请求一次后就得断掉,所以while循环必须写在链接的前面

(2)我们平时掉线了 就是服务端和客户端断开了,这个时候我们客户端要继续尝试跟服务端建立连接,当然也要限制连接次数  所以可以用一个do while循环放在链接那里

因为每处理一次请求就要断掉,所以while循环必须写到链接前面

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

using namespace std;

void Usage(std::string proc)
{
    std::cout << "

Usage: " << proc << " serverip serverport
"
              << std::endl;
}

// ./tcpclient serverip serverport
int main(int argc, char *argv[]) // 必须知道服务器的ip和端口号
{
    if (argc != 3)
    {
        Usage(argv[0]);
        exit(0);
    }
    string serverip = argv[1];
    uint16_t serverport = std::stoi(argv[2]);

    // 2/OS帮助们bind
    struct sockaddr_in server; // 输出型参数
    bzero(&server, sizeof(server));
    server.sin_family = AF_INET;
    server.sin_port = htons(serverport);
    // server.sin_addr.s_addr = inet_addr(serverip.c_str());//字符串转四字节
    inet_pton(AF_INET, serverip.c_str(), &(server.sin_addr));
    // 3、向服务端发送链接请求
    while (true)
    {
        int cnt = 5;             // 重连次数
        int isreconnect = false; // 是否要尝试重连
        // 创建套接字
        int sockfd =0; 
        sockfd=socket(AF_INET, SOCK_STREAM, 0);
        if (sockfd < 0)
        {
            cerr << "socker error" << endl;
            return 1;
        }
        do
        {
            int n = connect(sockfd, (struct sockaddr *)&server, sizeof(server));
            if (n < 0) // 如果链接失败
            {
                isreconnect = true;
                --cnt;
                cerr << "connect error……" << endl;
            }
            else
                break; // 链接成功就跳出去
        } while (cnt && isreconnect);
        if (cnt == 0)
        {
            cerr << "user ofline" << endl;
            break;
        }
        // 链接成功
        cout << "connect sucess" << endl;
        string message;      // 用来
        char inbuffer[1024]; // 接收读取的缓冲区
        cout << "please enter@";
        getline(cin, message); // 将获取的信息放到message中 发到服务端
        // 1. 数据 2. 给谁发
        ssize_t n = write(sockfd, message.c_str(), message.size()); // 可以直接通过write写到文件里
        if (n < 0)
        {
            std::cerr << "write error..." << std::endl;
            // break; 短服务不用出去
        }
        // 一般不会写失败,因为服务器一般都不关
        // 从文件里读
        n = read(sockfd, inbuffer, sizeof(inbuffer)); // 读到我们的缓冲区里
        // 会将结果带回来
        if (n > 0)
        {
            inbuffer[n] = 0;
            cout << inbuffer << endl;
        }
    close(sockfd);
}
     return 0;
}

当然如果执行的是长服务的话,一旦写入失败就要break出去重连!!

 二、守护进程

     服务端在我们ctrl c或者关掉xshell的时候就会被杀死,但是我们希望无论如何这个服务端是一直在跑的!!所以我们必须守护进程!!

2.1 Session和前后台进程

 每当一个用户登录的就是 默认就会形成一个session,然后分配一个bash进程

 前台进程后后台进程的关键在于谁拥有键盘文件!

1、执行可执行程序的时候在后面加个&  该进程就会变成后台进程

2、通过jobs命令可以看到所有后台任务

3、该序号叫做后台进程任务号,我们可以使用fg+序号将后台进程提到前台

4、如果我们将一个后台进程提到前台之后后悔了,我们可以ctrl+z向前台进程发送19号信号,此时当前台进程被暂停时,bash进程就会自动移到前台进程(因为在命令行中,前台必须存在),而暂停的进程自动放到后台。    然后通过bg+序号将因为暂停被放在后台的进程恢复运行!

2.2 进程间关系

1、PGID叫进程组ID,一个组的是一样的 ,只启动一个进程的话就自成一组

2、sessionid用的就是bash进程的pid,而多个进程组在同一个session里面sid是一样的

3、如果我们关掉OS,那么后台进程会收到用户登录和退出的影响  因此我们需要守护进程化

2.3 如何做到

要尝试自成一个会话!! 

 

 

setsid  自成会话 不能是组长,所以我们必须fork出子进程,然后退出父进程,让子进程执行后面的代码,所以 守护进程的本质也是孤儿进程!

#pragma once

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

const std::string nullfile = "/dev/null";

void Daemon(const std::string &cwd = "")
{
    // 1. 忽略其他异常信号
    signal(SIGCLD, SIG_IGN);
    signal(SIGPIPE, SIG_IGN);
    signal(SIGSTOP, SIG_IGN);

    // 2. 将自己变成独立的会话
    if (fork() > 0)
        exit(0);
    setsid();

    // 3. 更改当前调用进程的工作目录
    if (!cwd.empty())
        chdir(cwd.c_str());

    // 4. 标准输入,标准输出,标准错误重定向至/dev/null
    int fd = open(nullfile.c_str(), O_RDWR);
    if(fd > 0)
    {
        dup2(fd, 0);
        dup2(fd, 1);
        dup2(fd, 2);
        close(fd);
    }
}

(1)忽略其他异常信号

(2)用setsid将自己变成独立会话

(3)更改当前的工作目录

(4)标准输出、输入、错误重定向到/dev/null (守护进程必须和他们解关联,如果我们不往显示器而是文件写入的话还好,但如果我们直接关闭描述符的话显然会导致printf和cout出错!!而/dev/null就是相当于是一个垃圾桶文件,可以把不关心的内容丢到里面去)

 

所以我们将上述代码在Run函数中运行 然后同时把我们的日志改成写到文件中!!

如果我们想杀掉的话就得用kill -9 PID 

 2.4 为什么我们能远程登录Linux呢?

其实ssh就是守护进程,我们向他发送链接请求,认证后再登录,然后分配一个会话,然后将命令再远端执行完后再返回给你 

 一般来说守护进程我们一般在他的名字后面加一个-D

三、TCP协议的通讯流程 

3.1 TCP的三次握手和四次挥手 

3.2 TCP通信全双工 

TCP是全双工的 ,因为发送和接受缓冲区是分开的,多线程时虽然不能多人读,但是支持同时读写!!

3.3 如何理解链接 

        对于服务器来说,同时存在大量连接,那么谁来打开、谁来关闭、连接状态是什么,所以OS必须要先描述再组织来管理链接

本文地址:https://www.vps345.com/11079.html

搜索文章

Tags

PV计算 带宽计算 流量带宽 服务器带宽 上行带宽 上行速率 什么是上行带宽? CC攻击 攻击怎么办 流量攻击 DDOS攻击 服务器被攻击怎么办 源IP docker 容器 运维 java-rabbitmq java 服务器安全 网络安全策略 防御服务器攻击 安全威胁和解决方案 程序员博客保护 数据保护 安全最佳实践 服务器 linux 游戏 云计算 Deepseek Deepseek-R1 大模型 私有化部署 推理模型 网络工程师 网络管理 软考 2024 2024年上半年 下午真题 答案 javascript 前端 chrome edge deepseek DeepSeek-R1 API接口 c++ centos python 机器学习 深度学习 人工智能 神经网络 计算机视觉 卷积神经网络 数据库 oracle 关系型 安全 分布式 RTSP xop RTP RTSPServer 推流 视频 ddos android 网络安全 web安全 DNS ubuntu YOLO conda pytorch vscode yolov5 redis ssh harmonyos 华为 物联网 网络 macos windows vue.js spring boot nginx gcc centos 7 tcp/ip 网络协议 ip协议 c语言 Flask FastAPI Waitress Gunicorn uWSGI Uvicorn 进程 操作系统 进程控制 Ubuntu unix 自动化 英语 springsecurity6 oauth2 授权服务器 前后端分离 Ollama Qwen2.5-coder 离线部署 服务器无法访问 ip地址无法访问 无法访问宝塔面板 宝塔面板打不开 ROS 自动驾驶 oracle fusion oracle中间件 stm32 嵌入式硬件 单片机 audio vue音乐播放器 vue播放音频文件 Audio音频播放器自定义样式 播放暂停进度条音量调节快进快退 自定义audio覆盖默认样式 计算机外设 bug pycharm ide kubernetes prometheus grafana 云原生 ai nlp FTP服务器 opencv https 游戏引擎 学习 负载均衡 agi AIGC MacMini Mac 迷你主机 mini Apple arm开发 架构 指令 数据库系统 php 开发语言 远程桌面 CH340 串口驱动 CH341 uart 485 fstab 笔记 C 环境变量 进程地址空间 flutter Google pay Apple pay 模型联网 API CherryStudio mysql 经验分享 学习方法 YOLOv8 NPU Atlas800 A300I pro asi_bench adb 阿里云 llama 算法 自然语言处理 语言模型 django gnu centos-root /dev/mapper yum clean all df -h / du -sh apache 交互 银河麒麟 c# vue3 excel 在线预览 xlsx xls文件 在浏览器直接打开解析xls表格 前端实现vue3打开excel 文件地址url或接口文档流二进 rocketmq vite qt arm tomcat outlook 错误代码2603 无网络连接 2603 typescript 计算机网络 UEFI Legacy MBR GPT U盘安装操作系统 node.js virtualenv 科技 个人开发 react.js 前端面试题 持续部署 教程 环境搭建 Java Maven 实时音视频 音视频 rpc 数据分析 开发环境 ansible git npm 华为云 华为od 后端 Ubuntu20.04 GLIBC 2.35 网络结构图 dify 知识库 本地化部署 部署 程序人生 像素流送api 像素流送UE4 像素流送卡顿 像素流送并发支持 opengl MCP ui 目标检测 intellij-idea ACL 流量控制 基本ACL 规则配置 C语言 Ubuntu DeepSeek DeepSeek Ubuntu DeepSeek 本地部署 DeepSeek 知识库 DeepSeek 私有化知识库 本地部署 DeepSeek DeepSeek 私有化部署 flask jvm 虚拟机 ip spring cloud compose ipython devops github 人工智能生成内容 Dify mount挂载磁盘 wrong fs type LVM挂载磁盘 Centos7.9 deepseek-r1 大模型本地部署 智能路由器 dell服务器 程序 编程 内存 性能分析 服务器配置 生物信息学 开源 milvus 性能优化 sql RAID RAID技术 磁盘 存储 elasticsearch grub 版本升级 扩容 高级IO epoll 目标跟踪 OpenVINO 推理应用 cron crontab日志 宝塔面板访问不了 宝塔面板网站访问不了 宝塔面板怎么配置网站能访问 宝塔面板配置ip访问 宝塔面板配置域名访问教程 宝塔面板配置教程 机器人 AI编程 远程工作 缓存 visualstudio 面试 svn jenkins gitee stm32项目 http 反向代理 filezilla 无法连接服务器 连接被服务器拒绝 vsftpd 331/530 qps 高并发 AutoDL zabbix mariadb matplotlib fonts-noto-cjk 外网访问 内网穿透 端口映射 集成学习 集成测试 burpsuite 安全工具 mac安全工具 burp安装教程 渗透工具 maxkb ARG 交换机 硬件 设备 GPU PCI-Express 建站 rust腐蚀 vnc 镜像源 selenium 策略模式 asp.net大文件上传 asp.net大文件上传下载 asp.net大文件上传源码 ASP.NET断点续传 虚拟显示器 远程控制 雨云 NPS python2 ubuntu24.04 AI-native gpt jellyfin nas JDK LInux Windows 自定义客户端 SAS ffmpeg 视频编解码 pip rsyslog live555 rtsp rtp 并查集 leetcode mcu 数据结构 小程序 微信小程序域名配置 微信小程序服务器域名 微信小程序合法域名 小程序配置业务域名 微信小程序需要域名吗 微信小程序添加域名 eureka fastapi mcp mcp-proxy mcp-inspector fastapi-mcp agent sse udp 本地环回 bind mysql离线安装 ubuntu22.04 mysql8.0 k8s Linux xrdp 远程连接 adobe asm 环境迁移 word图片自动上传 word一键转存 复制word图片 复制word图文 复制word公式 粘贴word图文 粘贴word公式 鸿蒙 快捷键 旋转屏幕 自动操作 ubuntu 18.04 安装教程 gitea USB转串口 WSL2 IP 地址 大数据 spark hive 前端框架 uni-app 上传视频文件到服务器 uniApp本地上传视频并预览 uniapp移动端h5网页 uniapp微信小程序上传视频 uniapp app端视频上传 uniapp uview组件库 数据挖掘 iftop 网络流量监控 golang debian PVE 多线程服务器 Linux网络编程 kafka AI大模型 大模型技术 本地部署大模型 VMware ollama 压力测试 大模型压力测试 EvalScope 虚拟现实 websocket Alexnet ArkUI 鸿蒙系统 ArkTS 镜像下载 freebsd linuxdeployqt 打包部署程序 appimagetool kali 共享文件夹 Maxkb RAG技术 本地知识库 GPU环境配置 Ubuntu22 CUDA PyTorch Anaconda安装 bash TCP WebServer perf es6 qt6.3 g726 firewall notepad JAVA gitlab unity 课程设计 kylin 统信 NFS 思科 运维开发 mac react next.js 部署next.js ubuntu20.04 ros ros1 Noetic 20.04 apt 安装 postman 测试工具 sublime text 编辑器 rtsp服务器 rtsp server android rtsp服务 安卓rtsp服务器 移动端rtsp服务 大牛直播SDK vim json HarmonyOS Next frp maven 嵌入式 阻塞队列 生产者消费者模型 服务器崩坏原因 桥接模式 vmware windows虚拟机 虚拟机联网 linux驱动开发 ue4 着色器 ue5 虚幻 QT 5.12.12 QT开发环境 Ubuntu18.04 LDAP 硬件工程 cuda zotero WebDAV 同步失败 代理模式 n8n 工作流 系统安全 进程优先级 调度队列 进程切换 pygame windows 服务器安装 网易邮箱大师 EtherNet/IP串口网关 EIP转RS485 EIP转Modbus EtherNet/IP网关协议 EIP转RS485网关 EIP串口服务器 ip命令 新增网卡 新增IP 启动网卡 管道 匿名管道 protobuf 序列化和反序列化 安装 虚拟局域网 信息与通信 高效I/O yum docker-compose docker compose 电脑 NVML nvidia-smi 防火墙 端口号 开放端口 访问列表 C++ Doris搭建 docker搭建Doris Doris搭建过程 linux搭建Doris Doris搭建详细步骤 Doris部署 大模型入门 大模型教程 程序员创富 openvpn server openvpn配置教程 centos安装openvpn list 腾讯云 rag ragflow 大模型部署 华为认证 jupyter spring llm DeepSeek Chatbox 孤岛惊魂4 AI 智能体开发 p2p IMM 网络药理学 生信 gromacs 分子动力学模拟 MD 动力学模拟 多线程 VMware Tools vmware tools安装 vmwaretools安装步骤 vmwaretools安装失败 vmware tool安装步骤 vm tools安装步骤 vm tools安装后不能拖 vmware tools安装步骤 1024程序员节 无人机 PX4 Linux PID IM即时通讯 QQ 微信 企业微信 剪切板对通 HTML FORMAT burp suite 抓包 axure 富文本编辑器 安卓模拟器 Hyper-V WinRM TrustedHosts gpu算力 电子信息 通信工程 毕业 Linux 维护模式 Ubuntu 22.04 MySql 算家云 算力租赁 移动端开发 dubbo 监控k8s集群 集群内prometheus 命名管道 客户端与服务端通信 微信小程序 pillow hadoop big data 爬虫 cpu 实时 使用 聚类 nvm 串口服务器 开机黑屏 命令 CPU 使用率 系统监控工具 linux 命令 C# MQTTS 双向认证 emqx 链表 zip unzip KylinV10 麒麟操作系统 Vmware transformer ssl cursor FTP 服务器 EtherCAT转Modbus EtherCAT转485网关 ECT转485串口服务器 ECT转Modbus485协议 ECT转Modbus串口网关 ECT转Modbus串口服务器 xcode IIS .net core Hosting Bundle .NET Framework vs2022 ssh漏洞 ssh9.9p2 CVE-2025-23419 私有化 本地部署 open webui 实时互动 驱动开发 媒体 磁盘挂载 新盘添加 partedUtil 图像处理 深度求索 私域 7z 硅基流动 ChatBox Kylin-Server 国产操作系统 服务器安装 mongodb Cline jar gradle oceanbase rc.local 开机自启 systemd 麒麟 3d 数学建模 云电竞 云电脑 todesk etcd 数据安全 RBAC css css3 html5 网络爬虫 linux环境变量 jdk iTerm2 Linux无人智慧超市 LInux多线程服务器 QT项目 LInux项目 单片机项目 报错 glibc 系统架构 硬件架构 计算机科学与技术 SRS 流媒体 直播 云原生开发 K8S k8s管理系统 国产数据库 瀚高数据库 数据迁移 下载安装 tar ollama api ollama外网访问 visual studio code wsl photoshop vue Dell R750XS ocr 毕设 安卓 RDP IIS服务器 IIS性能 日志监控 localhost 温湿度数据上传到服务器 Arduino HTTP web elk 进程信号 ios 虚拟化 智慧农业 开源鸿蒙 团队开发 VPN wireguard 灵办AI 智能手机 软件需求 rust 其他 clickhouse 读写锁 Invalid Host allowedHosts 计算生物学 生物信息 基因组 隐藏文件 安全威胁分析 numpy kubeless WSL2 上安装 Ubuntu MCP server C/S LLM 微服务 nacos express 具身智能 Isaac Sim 虚拟仿真 CNNs 图像分类 一切皆文件 多层架构 解耦 浪潮信息 AI服务器 CPU 主板 电源 网卡 postgresql rustdesk GaN HEMT 氮化镓 单粒子烧毁 辐射损伤 辐照效应 微信分享 Image wxopensdk DevOps 软件交付 数据驱动 应用场景 宝塔面板无法访问 intellij idea 区块链 行情服务器 股票交易 速度慢 切换 股票量化接口 股票API接口 java-ee Mac内存不够用怎么办 webdav HCIE 数通 卡死 miniapp 真机调试 调试 debug 断点 网络API请求调试方法 .netcore Docker Desktop 教育电商 游戏程序 vm eclipse 热榜 gateway Clion Nova ResharperC++引擎 Centos7 远程开发 redhat rdp 实验 王者荣耀 python3.11 pyside6 界面 flash-attention umeditor粘贴word ueditor粘贴word ueditor复制word ueditor上传word图片 ueditor导入word ueditor导入pdf ueditor导入ppt iot 论文阅读 虚拟机安装 webrtc 冯诺依曼体系 fork wait waitpid exit 离线部署dify html GameFramework HybridCLR Unity编辑器扩展 自动化工具 工具 VM搭建win2012 win2012应急响应靶机搭建 攻击者获取服务器权限 上传wakaung病毒 应急响应并溯源 挖矿病毒处置 应急响应综合性靶场 k8s部署 MySQL8.0 高可用集群(1主2从) deepseak 文心一言 豆包 KIMI 腾讯元宝 WebUI DeepSeek V3 web3.py VMware安装mocOS macOS系统安装 程序员 edge浏览器 onlyoffice samba kotlin android studio iphone 互信 vSphere vCenter 软件定义数据中心 sddc ruoyi vmamba efficientVIT YOLOv8替换主干网络 TOLOv8 chatgpt oneapi 大模型微调 vr 考试 su sudo sudo原理 su切换 昇腾 大模型训练/推理 推理问题 mindie MQTT 消息队列 远程过程调用 Windows环境 大语言模型 LLMs jmeter 软件测试 grep Cursor langchain deep learning Trae叒更新了? tcp 产品经理 microsoft pdf linux内核 termux database yolov8 Ubuntu22.04 开发人员主页 ci/cd 代码规范 iperf3 带宽测试 TrueLicense Python IPMI WLAN rancher 软件卸载 系统清理 rabbitmq SSH Xterminal Mermaid 可视化图表 自动化生成 Kali Linux 黑客 渗透测试 信息收集 ecmascript nextjs reactjs lvm 磁盘分区 ssrf 失效的访问控制 HTML audio 控件组件 vue3 audio音乐播放器 Audio标签自定义样式默认 vue3播放音频文件音效音乐 自定义audio播放器样式 播放暂停调整声音大小下载文件 VMware安装Ubuntu Ubuntu安装k8s Agent CrewAI 僵尸进程 华为OD 华为OD机试真题 可以组成网络的服务器 ufw nftables 5G shell Hive环境搭建 hive3环境 Hive远程模式 dash 正则表达式 webgl diskgenius 信号处理 tcpdump 大文件分片上传断点续传及进度条 如何批量上传超大文件并显示进度 axios大文件切片上传详细教 node服务器合并切片 vue3大文件上传报错提示错误 vu大文件秒传跨域报错cors OpenCore 创意 社区 DevEco Studio db VMware创建虚拟机 源码剖析 rtsp实现步骤 流媒体开发 进程程序替换 execl函数 execv函数 execvp函数 execvpe函数 putenv函数 odoo 服务器动作 Server action Qualcomm WoS QNN AppBuilder openjdk Mac软件 macbook 知识图谱 linux上传下载 docker run 数据卷挂载 交互模式 ftp服务 文件上传 webpack 回显服务器 UDP的API使用 ai小智 语音助手 ai小智配网 ai小智教程 智能硬件 esp32语音助手 diy语音助手 matlab remote-ssh aws scikit-learn netty 远程服务 宝塔 政务 分布式系统 监控运维 Prometheus Grafana MobaXterm 文件传输 RustDesk自建服务器 rustdesk服务器 docker rustdesk React Next.js 开源框架 powerpoint 升级 CVE-2024-7347 漏洞 kernel 软件工程 软件构建 压测 ECS Linux Vim ESP32 camera Arduino 主从复制 统信UOS bonding 链路聚合 沙盒 需求分析 提示词 网络用户购物行为分析可视化平台 大数据毕业设计 ROS2 gpt-3 GCC aarch64 编译安装 HPC ECT转Modbus协议 EtherCAT转485协议 ECT转Modbus网关 HP Anyware 传统数据库升级 银行 VSCode MAVROS 四旋翼无人机 常用命令 文本命令 目录命令 nohup 异步执行 智能体 autogen openai coze arkUI arkTs 话题通信 服务通信 内网渗透 靶机渗透 MacOS录屏软件 openEuler Trae IDE AI 原生集成开发环境 Trae AI 职场和发展 嵌入式实习 大模型应用 DIFY UOS 统信操作系统 bcompare Beyond Compare 网卡的名称修改 eth0 ens33 高德地图 鸿蒙接入高德地图 HarmonyOS5.0 Cookie mybatis word 负载测试 GPU训练 c openssl seatunnel 腾讯云大模型知识引擎 SSE IO ping++ 深度优先 图论 并集查找 换根法 树上倍增 NAS Termux Samba 底层实现 ragflow 源码启动 Linux的基础指令 llama3 Chatglm 开源大模型 迁移指南 SWAT 配置文件 服务管理 网络共享 键盘 开发 cudnn anaconda wireshark 隐藏目录 文件系统 管理器 通配符 linux安装配置 Reactor 设计模式 服务器扩容没有扩容成功 OpenManus 中兴光猫 换光猫 网络桥接 自己换光猫 嵌入式系统开发 版本 券商 股票交易接口api 类型 特点 arcgis 银河麒麟服务器操作系统 系统激活 uni-file-picker 拍摄从相册选择 uni.uploadFile H5上传图片 微信小程序上传图片 HiCar CarLife+ CarPlay QT RK3588 mysql安装报错 windows拒绝安装 视频平台 录像 视频转发 性能测试 视频流 CLion 矩阵 fpga开发 磁盘清理 开机自启动 桌面快捷方式 playbook 剧本 跨平台 ubuntu24.04.1 生活 客户端 pyicu 博客 图片增强 增强数据 windwos防火墙 defender防火墙 win防火墙白名单 防火墙白名单效果 防火墙只允许指定应用上网 防火墙允许指定上网其它禁止 firewalld retry 重试机制 工业4.0 KingBase 搜索引擎 微信开放平台 微信公众平台 微信公众号配置 x64 SIGSEGV xmm0 junit zookeeper 显卡驱动持久化 GPU持久化 SSH 密钥生成 SSH 公钥 私钥 生成 Ubuntu 24 常用命令 Ubuntu 24 Ubuntu vi 异常处理 Github加速 Mac上Github加速 Chrome浏览器插件 Kali 渗透 权限 lighttpd安装 Ubuntu配置 Windows安装 服务器优化 Ardupilot 豆瓣 追剧助手 迅雷 弹性计算 云服务器 裸金属服务器 弹性裸金属服务器 源代码管理 大版本升 升级Ubuntu系统 Webserver 异步 unity3d Portainer搭建 Portainer使用 Portainer使用详解 Portainer详解 Portainer portainer kylin v10 麒麟 v10 蓝耘科技 元生代平台工作流 ComfyUI H3C 飞牛nas fnos Office dns googlecloud 单元测试 sublime text3 软链接 硬链接 springcloud 小智 Web服务器 多线程下载工具 网络编程 PYTHON mock mock server 模拟服务器 mock服务器 Postman内置变量 Postman随机数据 ArkTs .net vasp安装 initramfs Linux内核 Grub 云服务 ELF加载 Web应用服务器 中间件 mq 国产化 个人博客 宠物 毕业设计 免费学习 宠物领养 宠物平台 webstorm 计算机 监控 自动化运维 ollama下载加速 kamailio sip VoIP 大数据平台 联网 easyconnect 代理 CosyVoice DrissionPage 相机 电路仿真 multisim 硬件工程师 硬件工程师学习 电路图 电路分析 仪器仪表 动静态库 Linux的权限 强制清理 强制删除 mac废纸篓 办公自动化 pdf教程 CentOS 华为证书 HarmonyOS认证 华为证书考试 sqlserver Reactor反应堆 AI写作 数据可视化 推荐算法 海康 springboot 设置代理 实用教程 export env 变量 xml ShapeFile GeoJSON Nginx Docker Hub docker pull daemon.json Carla 智能驾驶 功能测试 自动化测试 r语言 数据集 智能电视 命令行 基础入门 fpga fd 文件描述符 视频监控 网站 数据采集 Crawlee Playwright log4j 低代码 nohup后台启动 双系统 远程 执行 sshpass 操作 servlet minicom 串口调试工具 计算机系统 显示器 curl wget 单一职责原则 Ubuntu共享文件夹 共享目录 Linux共享文件夹 信号 内核 系统开发 binder 车载系统 framework 源码环境 iis 安装MySQL 移动云 CDN 安防软件 京东云 firefox iNode Macos EMQX 通信协议 Docker Docker Compose Kubernetes k8s集群资源管理 go 多产物 物联网开发 自动化编程 java-rocketmq 监控k8s 监控kubernetes 云桌面 微软 AD域控 证书服务器 Netty 概率论 考研 系统 黑苹果 浏览器自动化 WSL win11 无法解析服务器的名称或地址 kind systemctl composer 全文检索 图搜索算法 perl alias unalias 别名 安全漏洞 信息安全 免密 登录 公钥 私钥 visual studio 支持向量机 copilot ebpf uprobe ux 鲲鹏 npu rime 桌面环境 WebVM LLM Web APP Streamlit 流水线 脚本式流水线 gunicorn AD域 Xinference string模拟实现 深拷贝 浅拷贝 经典的string类问题 三个swap sysctl.conf vm.nr_hugepages 代码调试 ipdb hosts hosts文件管理工具 开源软件 视觉检测 client-go k8s二次开发 stable diffusion 3GPP 卫星通信 MS Materials 蓝桥杯 TRAE Redis Desktop 锁屏不生效 MQTT协议 消息服务器 代码 IP配置 netplan 远程看看 远程协助 直播推流 beautifulsoup sqlite3 RAGFLOW 机柜 1U 2U Windsurf openwrt 大屏端 chatbox selete armbian u-boot shell脚本免交互 expect linux免交互 框架搭建 飞牛NAS 飞牛OS MacBook Pro harmonyOS面试题 ros2 moveit 机器人运动 cnn 换源 国内源 Debian DigitalOcean GPU服务器购买 GPU服务器哪里有 GPU服务器 加解密 Yakit yaklang 环境配置 ftp 云耀服务器 强化学习 显示过滤器 ICMP Wireshark安装 图文教程 VMware虚拟机 macOS系统安装教程 macOS最新版 虚拟机安装macOS Sequoia 基础环境 实战案例 AI作画 less import save load 迁移镜像 单例模式 执法记录仪 智能安全帽 smarteye vscode1.86 1.86版本 ssh远程连接 windows日志 CUPS 打印机 Qt5 EVE-NG Linux权限 权限命令 特殊权限 拓扑图 sdkman ESXi top Linux top top命令详解 top命令重点 top常用参数 java-zookeeper mac设置host docker搭建pg docker搭建pgsql pg授权 postgresql使用 postgresql搭建 数码 联机 僵尸毁灭工程 游戏联机 开服 openstack Xen KVM c/c++ 串口 deepseek-v3 ktransformers iBMC UltraISO P2P HDLC 模拟实现 nac 802.1 portal autoware 用户管理 rpa thingsboard docker搭建nacos详解 docker部署nacos docker安装nacos 腾讯云搭建nacos centos7搭建nacos 笔灵AI AI工具 HistoryServer Spark YARN jobhistory 域名服务 DHCP 符号链接 配置 工具分享 NVM Node Yarn PM2 iventoy VmWare OpenEuler threejs 3D okhttp YOLOv12 磁盘监控 路径解析 状态管理的 UDP 服务器 Arduino RTOS 图形化界面 多进程 g++ g++13 v10 软件 file server http server web server GRE HarmonyOS Tabs组件 TabContent TabBar TabsController 导航页签栏 滚动导航栏 yaml Ultralytics 可视化 AimRT RAG 检索增强生成 文档解析 大模型垂直应用 Python教程 pycharm安装 Wi-Fi minio Spring Security 技术共享 rnn 显卡驱动 我的世界 我的世界联机 AP配网 AK配网 小程序AP配网和AK配网教程 WIFI设备配网小程序UDP开 UDP miniconda SysBench 基准测试 服务器时间 线程 容器技术 模拟器 Xshell jina IPv4 子网掩码 公网IP 私有IP 计算机学习路线 编程语言选择 Linux24.04 deepin swift 医疗APP开发 app开发 金融 trea idea jetty undertow massa sui aptos sei DBeaver 语音识别 wordpress 产测工具框架 IMX6ULL 管理框架 MNN Qwen pthread Linux find grep gaussdb llama.cpp mamba Vmamba DeepSeek行业应用 Heroku 网站部署 Dell HPE 联想 浪潮 springboot远程调试 java项目远程debug docker远程debug java项目远程调试 springboot远程 可执行程序 lio-sam SLAM trae 思科模拟器 Cisco nuxt3 测试用例 论文笔记 asp.net上传文件夹 asp.net上传大文件 .net core断点续传 .net mvc断点续传 Obsidian Dataview 聊天室 代理服务器 多端开发 智慧分发 应用生态 鸿蒙OS 自学笔记 小米 澎湃OS Android 群晖 飞牛 Node-Red 编程工具 流编程 算力 服务器管理 宝塔面板 配置教程 网站管理 ECS服务器 nvidia kvm muduo Linux awk awk函数 awk结构 awk内置变量 awk参数 awk脚本 awk详解 程序化交易 PTrade QMT 量化交易 量化股票 NLP模型 NLP fast Claude Desktop Claude MCP Windows Cli MCP 星河版 workflow 软件商店 信创 livecd systemtools wsl2 vsxsrv 社交电子 高效远程协作 TrustViewer体验 跨设备操作便利 智能远程控制 SoC finebi conda配置 conda镜像源 可信计算技术 安全架构 网络攻击模型 服务器部署 本地拉取打包 直流充电桩 充电桩 远程登录 telnet 稳定性 看门狗 Typore anythingllm open-webui docker国内镜像 tensorflow SEO 显示管理器 lightdm gdm crosstool-ng 同步 备份 MAC SecureCRT vscode 1.86 网站搭建 serv00 linux 命令 sed 命令 prompt AI提示词优化 Nuxt.js comfyui comfyui教程 ShenTong vpn 自动化任务管理 7-zip wpf banner easyui web3 区块链项目 网络穿透 分子对接 autodock mgltools PDB PubChem Docker快速入门 即时通信 NIO yum源切换 更换国内yum源 分析解读 音乐服务器 Navidrome 音流 bot 毕昇JDK PyQt PySide6 致远OA OA服务器 服务器磁盘扩容 上传视频至服务器代码 vue3批量上传多个视频并预览 如何实现将本地视频上传到网页 element plu视频上传 ant design vue vue3本地上传视频及预览移除 交叉编译 CORS 跨域 游戏机 欧拉系统 micropython esp32 mqtt 边缘计算 DeepSeek r1 Open WebUI 飞腾处理器 弹性服务器 健康医疗 Linux环境 卸载 列表 半虚拟化 硬件虚拟化 Hypervisor csrf 能力提升 面试宝典 技术 IT信息化 模拟退火算法 田俊楠 code-server mosquitto SVN Server tortoise svn solidworks安装 Kylin OS 免费域名 域名解析 小艺 Pura X cmake android-studio pgpool 密码学 业界资讯 js bat 服务器繁忙 UOS1070e rtc echarts 信息可视化 网页设计 WINCC 代码托管服务 AISphereButler hibernate Java Applet URL操作 服务器建立 Socket编程 网络文件读取 Ark-TS语言 ukui 麒麟kylinos openeuler can 线程池 W5500 OLED u8g2 TCP服务器 服务器ssl异常解决 chfs ubuntu 16.04 VR手套 数据手套 动捕手套 动捕数据手套 配置原理 Masshunter 质谱采集分析软件 使用教程 科研软件 火绒安全 VPS 输入法 空间 查错 av1 电视盒子 机顶盒ROM 魔百盒刷机 pyautogui Alist rclone mount 挂载 网盘 cmos Pyppeteer resolv.conf 创业创新 MVS 海康威视相机 SSH 服务 SSH Server OpenSSH Server GeneCards OMIM TTD pyscenic 生信教程 DenseNet hdc WebRTC uniapp pppoe radius 实时内核 恒源云 keepalived 进程间通信 qemu libvirt sonoma 自动更新 RTMP 应用层 Echarts图表 折线图 柱状图 异步动态数据 鸿蒙开发 可视化效果 实时云渲染 云渲染 3D推流 neo4j 数据仓库 数据库开发 数据库架构 OD机试真题 服务器能耗统计 网络原理 LVM lvresize 磁盘扩容 pvcreate 重启 排查 系统重启 日志 原因 chrome devtools chromedriver RAGFlow 可用性测试 HarmonyOS NEXT 原生鸿蒙 IPv4/IPv6双栈 双栈技术 网路规划设计 ensp综合实验 IPv4过渡IPv6 IPv4与IPv6 ArcTS ArcUI GridItem 服务网格 istio 电视剧收视率分析与可视化平台 cd 目录切换 聊天服务器 套接字 Socket shard 智能音箱 智能家居 IPMITOOL BMC 硬件管理 opcua opcda KEPServer安装 token sas dba 端口测试 ubuntu24 vivado24 XCC Lenovo apt lsb_release /etc/issue /proc/version uname -r 查看ubuntu版本 docker命令大全 繁忙 解决办法 替代网站 汇总推荐 AI推理 virtualbox 终端 DocFlow OpenManage 服务器数据恢复 数据恢复 存储数据恢复 raid5数据恢复 磁盘阵列数据恢复 embedding 银河麒麟操作系统 xpath定位元素 wsgiref Web 服务器网关接口 OpenHarmony nfs 服务器部署ai模型 SSL 域名 skynet 本地部署AI大模型 机架式服务器 1U工控机 国产工控机 Jellyfin gru wps LORA 蓝牙 Helm k8s集群 图形渲染 RoboVLM 通用机器人策略 VLA设计哲学 vlm fot robot 视觉语言动作模型 网络文件系统 技能大赛 小游戏 五子棋 影刀 #影刀RPA# api Linux的基础开发工具 pyqt xfce 实习 大模型面经 大模型学习 EasyConnect 内网环境 d3d12 源代码 黑客技术 流式接口 VGG网络 卷积层 池化层 URL CentOS Stream 网工 opensearch helm 服务器主板 AI芯片 MI300x 大模型推理 源码 文件存储服务器组件 open Euler dde 混合开发 环境安装 自定义登录信息展示 motd 美化登录 chromium dpi 大文件秒传跨域报错cors cpp-httplib 玩游戏 postgres Dify重启后重新初始化 Qwen3 qwen3 32b vllm 设备树 scapy 游戏服务器 TrinityCore 魔兽世界 zerotier SSL证书 uv Bug解决 Qt platform OpenCV Zoertier 内网组网 金仓数据库 2025 征文 数据库平替用金仓 在线office 崖山数据库 YashanDB harmonyosnext chrome 浏览器下载 chrome 下载安装 谷歌浏览器下载 文件分享 MDK 嵌入式开发工具 雨云服务器 百度云 矩池云 数据下载 数据传输 商用密码产品体系 archlinux kde plasma 向日葵 高效日志打印 串口通信日志 服务器日志 系统状态监控日志 异常记录日志 AList fnOS 华为机试 saltstack make命令 makefile文件 相差8小时 UTC 时间 框架 laravel yashandb ceph Ubuntu Server Ubuntu 22.04.5 etl Bandizip Mac解压 Mac压缩 压缩菜单 cpolar 无桌面 risc-v 裸机装机 linux磁盘分区 裸机安装linux 裸机安装ubuntu 裸机安装kali 裸机 终端工具 远程工具 邮件APP 免费软件 三级等保 服务器审计日志备份 glm4 历史版本 下载 联想开天P90Z装win10 llamafactory 微调 Claude dity make bootstrap eNSP 网络规划 VLAN 企业网络 匿名FTP 邮件传输代理 SSL支持 chroot监狱技术 AnythingLLM AnythingLLM安装 safari 网页服务器 web服务器 充电桩平台 充电桩开源平台 ecm bpm k8s资源监控 annotations自动化 自动化监控 监控service 监控jvm 序列化反序列化 searxng docker desktop 镜像 宕机切换 服务器宕机 Docker引擎已经停止 Docker无法使用 WSL进度一直是0 镜像加速地址 录音麦克风权限判断检测 录音功能 录音文件mp3播放 小程序实现录音及播放功能 RecorderManager 解决录音报错播放没声音问题 ajax accept netlink libnl3 电脑桌面出现linux图标 电脑桌面linux图标删除不了 电脑桌面Liunx图标删不掉 linux图标删不掉 小番茄C盘清理 便捷易用C盘清理工具 小番茄C盘清理的优势尽显何处? 教你深度体验小番茄C盘清理 C盘变红?!不知所措? C盘瘦身后电脑会发生什么变化? cocos2d 3dcoat Minecraft DOIT 四博智联 工厂方法模式 NAT转发 NAT Server 数据管理 数据治理 数据编织 数据虚拟化 ruby idm 软件开发 信任链 cfssl 免费 用户缓冲区 树莓派 VNC 支付 微信支付 开放平台 互联网医院 GRUB引导 Linux技巧 iDRAC R720xd 对比 meld DiffMerge powerbi ssh远程登录 日志分析 系统取证 打不开xxx软件 无法检查其是否包含恶意软件 mvc cocoapods XFS xfs文件系统损坏 I_O error node es SenseVoice Charles AI代码编辑器 lb 协议 FunASR ASR 蓝桥杯C++组 Attention X11 Xming OpenSSH 我的世界服务器搭建 跨域请求 #STC8 #STM32 minecraft 线程同步 线程互斥 条件变量 接口返回 ISO镜像作为本地源 游戏开发 finalsheel 云计算面试题 材料工程 kerberos IMX317 MIPI H265 VCU TCP协议 事件驱动 磁盘镜像 服务器镜像 服务器实时复制 实时文件备份 UFW Erlang OTP gen_server 热代码交换 事务语义 gitee go 集群管理 Multi-Agent 备份SQL Server数据库 数据库备份 傲梅企业备份网络版 seleium hugo 嵌入式Linux IPC 模板 泛型编程 端口聚合 windows11 EMUI 回退 降级 AI员工 dns是什么 如何设置电脑dns dns应该如何设置 react native 银河麒麟桌面操作系统 达梦 DM8 xss win服务器架设 windows server Logstash 日志采集 homeassistant AI agent 物理地址 页表 虚拟地址 分布式训练 System V共享内存 进程通信 mapreduce NVIDIA 国标28181 监控接入 语音广播 流程 SIP SDP 软负载 状态模式 image 授时服务 北斗授时 金仓数据库概述 金仓数据库的产品优化提案 网络建设与运维 qt项目 qt项目实战 qt教程 流程图 mermaid 端口 查看 ss RagFlow deployment daemonset statefulset cronjob Qwen2.5-VL aac VS Code 根服务器 内存管理 nosql hexo post.io 企业邮箱 搭建邮箱 小智AI服务端 xiaozhi TTS docker部署翻译组件 docker部署deepl docker搭建deepl java对接deepl 翻译组件使用 element-ui 上传视频并预览视频 vue上传本地视频及进度条功能 vue2选择视频上传到服务器 upload上传视频组件插件 批量上传视频 限制单个上传视频 Modbus TCP MinIO AD 域管理 IO模型 openvino deekseek notepad++ win向maOS迁移数据 WireGuard 异地组网 OS rsync 查看显卡进程 fuser ArtTS MacOS HTTP 服务器控制 ESP32 DeepSeek qtcreator 备选 调用 示例 web环境 bigdata 软考设计师 中级设计师 SQL 软件设计师 子系统 钉钉 IPv6 IPv6测试 IPv6测速 IPv6检测 IPv6查询 语法 AzureDataStudio 查询数据库服务IP地址 SQL Server 抓包工具 动态库 GCC编译器 -fPIC -shared 医院门诊管理系统 智能合约 哈希算法 sqlite qt5 客户端开发 机械臂 银河麒麟高级服务器 外接硬盘 Kylin mm-wiki搭建 linux搭建mm-wiki mm-wiki搭建与使用 mm-wiki使用 mm-wiki详解 flink 三次握手 docker部署Python ranger MySQL8.0 做raid 装系统 内网服务器 内网代理 内网通信 企业网络规划 华为eNSP 规格说明书 顽固图标 启动台 运维监控 数据库管理 飞书 figma autodl xshell termius iterm2 红黑树封装map和set SystemV 知行EDI 电子数据交换 知行之桥 EDI GoogLeNet csrutil mac恢复模式进入方法 恢复模式 动态规划 mcp服务器 client close Headless Linux 极限编程 高频交易 合成模型 扩散模型 图像生成 Python 视频爬取教程 Python 视频爬取 Python 视频教程 mybase Radius 僵尸世界大战 游戏服务器搭建 脚本 Sealos Anolis nginx安装 linux插件下载 移动开发 怎么卸载MySQL MySQL怎么卸载干净 MySQL卸载重新安装教程 MySQL5.7卸载 Linux卸载MySQL8.0 如何卸载MySQL教程 MySQL卸载与安装 捆绑 链接 谷歌浏览器 youtube google gmail ldap GIS 遥感 WebGIS fiddler 分布式账本 共识算法 ai工具 阿里云ECS 项目部署到linux服务器 项目部署过程 超融合 Async注解 yum换源 pxe h.264 prometheus数据采集 prometheus数据模型 prometheus特点 网络搭建 神州数码 神州数码云平台 云平台 paddle whistle PPI String Cytoscape CytoHubba regedit 开机启动 增强现实 沉浸式体验 技术实现 案例分析 AR 鸿蒙NEXT logstash 问题解决 Ubuntu 24.04.1 轻量级服务器 虚幻引擎 NFC 近场通讯 智能门锁 玩机技巧 软件分享 软件图标 浏览器开发 AI浏览器 信创终端 中科方德 sentinel 搭建个人相关服务器 midjourney 静态IP GRANT REVOKE linux子系统 忘记密码 烟花代码 烟花 元旦 性能调优 安全代理 本地知识库部署 DeepSeek R1 模型 web开发 移动魔百盒 轮播图 佛山戴尔服务器维修 佛山三水服务器维修 Python基础 Python技巧 swoole 无法访问wordpess后台 打开网站页面错乱 linux宝塔面板 wordpress更换服务器 proxy模式 架构与原理 多个客户端访问 IO多路复用 TCP相关API Putty 花生壳 设计规范 干货分享 黑客工具 密码爆破 Apache Beam 批流统一 案例展示 数据分区 容错机制 流量运营 C++软件实战问题排查经验分享 0xfeeefeee 0xcdcdcdcd 动态库加载失败 程序启动失败 程序运行权限 标准用户权限与管理员权限 deepseek r1 ubantu IDEA tailscale derp derper 中转 triton 模型分析 线性代数 电商平台 抗锯齿 vue-i18n 国际化多语言 vue2中英文切换详细教程 如何动态加载i18n语言包 把语言json放到服务器调用 前端调用api获取语言配置文件 docker search 多路转接 Unity Dedicated Server Host Client 无头主机 欧标 OCPP lua VM虚拟机 USB网络共享 音乐库 项目部署 brew ubuntu安装 linux入门小白 navicat c/s easyTier 组网 dock 加速 PostgreSQL15数据库 漏洞报告生成 带外管理 webview 服务器正确解析请求体 计算虚拟化 弹性裸金属 watchtower AI Agent 字节智能运维 元服务 应用上架 接口优化 HAProxy 学习路线 STL OpenGL 导航栏 风扇控制软件 解决方案 lrzsz ABAP authing 李心怡 存储维护 NetApp存储 EMC存储 网络库 西门子PLC 通讯 智能问答 向量数据库 Spring AI Milvus 北亚数据恢复 oracle数据恢复 Qt QModbus 文件共享 ardunio BLE 大大通 第三代半导体 碳化硅 SPI mcp协议 go-zero sequoiaDB 零售 华为鸿蒙系统 ArkTS语言 Component 生命周期 条件渲染 Image图片组件 粘包问题 pythonai PlaywrightMCP macOS tidb 制造 安装部署 milvus安装 火山引擎 命令模式 RK3568 caddy 定义 核心特点 优缺点 适用场景 输入系统 qwen2vl vmware tools 进程池实现 mujoco Unity插件 数字证书 签署证书 solr 搜狗输入法 中文输入法 bert Metastore Catalog 蜂窝网络 频率复用 射频单元 无线协议接口RAN 主同步信号PSS broadcom access blocked 破解 devmem MLLMs VLM gpt-4v Ubuntu 24.04 搜狗输入法闪屏 Ubuntu中文输入法 AWS CAD瓦片化 栅格瓦片 矢量瓦片 Web可视化 DWG解析 金字塔模型 华为昇腾910b3 容器化 Serverless 环境 非root java毕业设计 微信小程序医院预约挂号 医院预约 医院预约挂号 小程序挂号 Python学习 Python编程 LVS vCenter服务器 ESXi主机 监控与管理 故障排除 日志记录 数字比特流 模拟信号 将二进制数据映射到模拟波形上 频谱资源 振幅频率相位 载波高频正弦波 LSTM 手机 wifi驱动 切换root 惠普服务器 惠普ML310e Gen8 惠普ML310e Gen8V2 NTP服务器 Web3 Telegram 代码复审 实时日志 logs orbslam2 能效分析 能源 springboot容器部署 springboot容器化部署 微服务容器化负载均衡配置 微服务容器多节点部署 微服务多节点部署配置负载均衡 烟雾检测 yolo检测 消防检测 4 - 分布式通信、分布式张量 rtcp 动态域名 端口开放 nginx默认共享目录 CPU架构 服务器cpu fabric h.265 超级终端 多任务操作 提高工作效率 anonymous 腾讯云服务器 轻量应用服务器 linux系统入门 linux命令 BCLinux