最新资讯

  • 【Linux】线程ID与互斥、同步(锁、条件变量)

【Linux】线程ID与互斥、同步(锁、条件变量)

2025-04-30 14:37:44 1 阅读

  🌈个人主页:秦jh__https://blog.csdn.net/qinjh_?spm=1010.2135.3001.5343
🔥 系列专栏:https://blog.csdn.net/qinjh_/category_12625432.html

目录

线程ID及进程地址空间布局

 线程简单封装

 线程互斥

进程线程间的互斥相关背景概念

 锁简单封装

 互斥量实现原理

 Linux线程同步

 同步概念与竞态条件

 条件变量

认识接口 

 生产者消费者模型

 基于BlockingQueue的生产者消费者模型

BlockQueue.hpp

main.cc

Task.hpp


前言

    💬 hello! 各位铁子们大家好哇。

             今日更新了Linux线程的内容
    🎉 欢迎大家关注🔍点赞👍收藏⭐️留言📝

线程ID及进程地址空间布局

 运行代码,这个很大的数字就是线程id。

通过 ps -aL 指令,看到LWP跟线程id是不一样的。

给用户提供的线程的id,不是内核中的lwp,而是pthread库维护的一个唯一值。

我们把上面的数字转换成十六进制,可以看出这是一个地址

 理解库

动态库被运行时加载,动态库没被加载前在磁盘中。

pthread库本质是一个文件。

我们刚刚写的可执行程序,它也是个文件,所以他也在磁盘中。可执行程序内部用线程库来创建多线程。

程序运行时,会变成一个进程,加载到内存中,内存中就有该进程的代码和数据。创建线程时,要先把库加载到内存中,然后再映射到该进程的地址空间才能用。映射要映射到堆栈之间的共享区。如果有多个多线程进程,它只需要把共享区的代码,经过页表映射到已经加载到内存的库,此时多个进程就可以使用同一个库里的方法来创建线程。

Linux只维护轻量级进程,linux中的pcb里与执行流相关的属性都是轻量级进程的属性,所有的属性都是围绕lwp展开的。我们在用户层的概念是线程,要的是线程的id,与线程相关的内容在Linux中是没有的,它没有维护。所以这部分属性由库来进行维护。  

为了更好的管理线程,创建线程时,库会为我们的每一个线程申请一个内存块(描述线程的相关结构体字段属性)。

未来要找一个线程的所有属性,只要找到线程控制块的地址即可。所以pthread_t id就是一个地址。

pthread_t类型的线程ID,本质就是线程属性集合的起始虚拟地址 ---- 在pthread库中维护。

 由上图可得,一个全局变量,本身就是被所有线程共享的。

 如果我们想让两个线程各自私有一份变量,g++有一个编译选项 __thread  

用__thread修饰这个全局变量即可。运行后,主线程和新线程gval的地址也不一样了。这种情况叫线程的局部存储,原始代码里只看到一个gval,但是他们用的是各自的gval。

这种情况只在Linux中有效。__thread只能用来修饰内置类型。

 线程简单封装

#pragma once
#include 
#include 
#include 

namespace ThreadMoudle
{
    // 线程要执行的方法,后面我们随时调整
    typedef void (*func_t)(const std::string &name); // 函数指针类型

    class Thread
    {
    public:
        void Excute()
        {
            std::cout << _name << " is running" << std::endl;
            _isrunning = true;
            _func(_name);
            _isrunning = false;
        }
    public:
        Thread(const std::string &name, func_t func):_name(name), _func(func)
        {
            std::cout << "create " << name << " done" << std::endl;
        }
        static void *ThreadRoutine(void *args) // 变成static,内部没有this指针,pthread_create就能匹配上了
        {
            Thread *self = static_cast(args); // 获得了当前对象
            self->Excute();
            return nullptr;
        }
        bool Start()
        {
            int n = ::pthread_create(&_tid, nullptr, ThreadRoutine, this);//传this,线程函数才能拿到_func方法
            if(n != 0) return false;
            return true;
        }
        std::string Status()
        {
            if(_isrunning) return "running";
            else return "sleep";
        }
        void Stop()
        {
            if(_isrunning)
            {
                ::pthread_cancel(_tid);
                _isrunning = false;
                std::cout << _name << " Stop" << std::endl;
            }
        }
        void Join()
        {
            ::pthread_join(_tid, nullptr);
            std::cout << _name << " Joined" << std::endl;
        }
        std::string Name()
        {
            return _name;
        }
        ~Thread()
        {
        }

    private:
        std::string _name;
        pthread_t _tid;
        bool _isrunning;
        func_t _func; // 线程要执行的回调函数
    };
} // namespace ThreadModle

 线程互斥

进程线程间的互斥相关背景概念

  • 临界资源:多线程执行流共享的资源就叫做临界资源
  • 临界区:每个线程内部,访问临界资源的代码,就叫做临界区
  • 互斥:任何时刻,互斥保证有且只有一个执行流进入临界区,访问临界资源,通常对临界资源起保护作用
  • 原子性:不会被任何调度机制打断的操作,该操作只有两态,要么完成,要么未完成 

 下面通过抢票代码演示:

void route(const std::string &name)
{
    while(true)
    {
        if(tickets > 0)
        {
            // 抢票过程
            usleep(1000); // 1ms -> 抢票花费的时间
            printf("who: %s, get a ticket: %d
", name.c_str(), tickets);
            tickets--;
        }
        else
        {
            break;
        }
    }
}

int main()
{
    Thread t1("thread-1", route);
    Thread t2("thread-2", route);
    Thread t3("thread-3", route);
    Thread t4("thread-4", route);

    t1.Start();
    t2.Start();
    t3.Start();
    t4.Start();

    t1.Join();
    t2.Join();
    t3.Join();
    t4.Join();
}

线程就用我们前面封装的线程。每抢到一张票就--,直到没票为止。

由上面结果可知,抢票抢到负数去了。因此多线程并发访问公共资源时可能会引发异常。 

  • 大部分情况,线程使用的数据都是局部变量,变量的地址空间在线程栈空间内,这种情况,变量归属单个线程,其他线程无法获得这种变量。
  • 但有时候,很多变量都需要在线程间共享,这样的变量称为共享变量,可以通过数据的共享,完成线程之间的交互。
  • 多个线程并发的操作共享变量,会带来一些问题。

 为什么会出现抢到负数的问题呢?

判断的过程是计算,由cpu来做。判断的过程:数据要先从内存移动到对应的寄存器中,然后再进行逻辑判断,然后才能得到结果,最后cpu再决定要执行if还是else。

之前讲过,cpu内寄存器只有一套,但是寄存器里的数据可以有多套。

这里有四个线程进行抢票,  如果一个线程在判断到一半时被切换了,他需要把寄存器里的数值带走,等到被唤醒的时候,他又要把数值恢复。

比如:假设票只剩一张了,线程a已经在判断完了,当他准备抢票的时候,他被切换了。此时他就把上下文数据保存。线程b被唤醒了,他也来进行票数判断,因为刚刚线程a还没来得及抢票,票数没--,所以线程b也判断成功,他也要抢票。线程a此时被唤醒,就往后执行代码进行抢票,然后--,票数就变成0。线程b又被唤醒,票数又--,就变成负数了。

如何解决上面的问题呢?加锁!

 pthread_mutex_t是互斥锁类型。

互斥锁在任何时刻,只允许一个线程进行资源访问。

有了锁,我们往往需要初始化和销毁锁,初始化有两种做法:

  1. 如果定义的是全局或者静态的锁,可以只使用pthread_mutex_t 锁的名字 =PTHREAD_MUTEX_INITIALIZER  
  2. 如果定义的这把锁是动态申请的,比如new或栈上开辟的,必须使用pthread_mutex_init函数来进行初始化。参数1就是你自己定义的锁,参数2是属性,直接设为nullptr即可。

用完锁后,还需要销毁,用pthread_mutex_destroy函数,参数是锁的地址。如果锁是静态或者全局的,我们不需要destroy,全局的或者静态的变量会随着进程的运行而一直存在,进程结束他也就自动释放了。初始化和销毁的返回值,成功返回0,失败返回-1。

一旦有了锁,我们就需要对临界区进行保护, 就需要加锁和解锁。要对某个区域加锁,就要调用pthread_mutex_lock函数来加锁,参数就是你定义的锁。要解锁,就用pthread_mutex_unlock函数。

lock的情况:

  • 互斥量处于未锁状态,该函数会将互斥量锁定,同时返回成功
  • 发起函数调用时,其他线程已经锁定互斥量,或者存在其他线程同时申请互斥量,但没有竞争到互斥量, 那么pthread_ lock调用会陷入阻塞(执行流被挂起),等待互斥量解锁。

所谓对临界区资源进行保护,本质是对临界区代码进行保护。 

把前面抢票的代码改一下,加锁。运行结果发现不会再抢到负数了,而且运行时间明显比之前要长。 

  1.  加锁的范围,粒度要尽量小,即临界区尽量小。
  2. 所有线程申请锁,前提是所有线程都看得到这把锁,因此锁本身也是共享资源。所以加锁的过程必须是原子的
  3. 原子性:要么不做,要么完成了。没有中间状态,就是原子性。
  4. 如果线程申请锁失败了,线程就要被阻塞
  5. 如果线程申请锁成功了,就继续往后运行,执行临界区代码。

 在执行临界区代码期间,线程可以被切换。假设线程1正在执行临界区代码,此时线程被切换了,其他线程也无法进入临界区,因为线程1并没有释放锁。

结论:我这个线程访问临界区,对其他线程来说是原子的。因为对于其他线程,我要么没有申请锁,要么释放了锁,这样对他们才有意义。

下面用局部的锁,使用前,先对线程重新封装

上面是增加的内容,下面是完整的代码

#pragma once
#include 
#include 
#include 

namespace ThreadMoudle
{
    class ThreadData
    {
    public:
        ThreadData(const std::string &name,pthread_mutex_t *lock):_name(name),_lock(lock)
        {}
    public:
        std::string _name;
        pthread_mutex_t* _lock;
    };


    // 线程要执行的方法,后面我们随时调整
    typedef void (*func_t)(ThreadData* td); // 函数指针类型
 
    class Thread
    {
    public:
        void Excute()
        {
            std::cout << _name << " is running" << std::endl;
            _isrunning = true;
            _func(_td);
            _isrunning = false;
        }
    public:
        Thread(const std::string &name, func_t func,ThreadData* td):_name(name), _func(func),_td(td)
        {
            std::cout << "create " << name << " done" << std::endl;
        }
        static void *ThreadRoutine(void *args) // 变成static,内部没有this指针,pthread_create就能匹配上了
        {
            Thread *self = static_cast(args); // 获得了当前对象
            self->Excute();
            return nullptr;
        }
        bool Start()
        {
            int n = ::pthread_create(&_tid, nullptr, ThreadRoutine, this);//传this,线程函数才能拿到_func方法
            if(n != 0) return false;
            return true;
        }
        std::string Status()
        {
            if(_isrunning) return "running";
            else return "sleep";
        }
        void Stop()
        {
            if(_isrunning)
            {
                ::pthread_cancel(_tid);
                _isrunning = false;
                std::cout << _name << " Stop" << std::endl;
            }
        }
        void Join()
        {
            ::pthread_join(_tid, nullptr);
            std::cout << _name << " Joined" << std::endl;
            delete _td;
        }
        std::string Name()
        {
            return _name;
        }
        ~Thread()
        {
        }

    private:
        std::string _name;
        pthread_t _tid;
        bool _isrunning;
        func_t _func; // 线程要执行的回调函数
        ThreadData* _td;
    };
} // namespace ThreadModle

运行代码,结果如下图,4个线程的锁都是同一个地址,说明他们访问的都是同一把锁。

最后再把线程函数代码换成抢票的代码即可,结果跟用全局的一样。

因此锁的使用既可以用全局的,也可以以参数的形式传递给线程。

 锁简单封装

因为是临时对象,在循环结束后会自动调用析构函数销毁。

 互斥量实现原理

  • 经过上面的例子,大家已经意识到单纯的 i++ 或者 ++i 都不是原子的,有可能会有数据一致性问题
  • 为了实现互斥锁操作,大多数体系结构都提供了swap或exchange指令,该指令的作用是把寄存器和内存单元的数据相交换,由于只有一条指令,保证了原子性,即使是多处理器平台,访问内存的 总线周期也有先后,一个处理器上的交换指令执行时另一个处理器的交换指令只能等待总线周期。 

 Linux线程同步

 同步概念与竞态条件

  • 同步:在保证数据安全的前提下,让线程能够按照某种特定的顺序访问临界资源,从而有效避免饥饿问题,叫做同步
  • 竞态条件:因为时序问题,而导致程序异常,我们称之为竞态条件。 

如上图,线程2一直抢到票,其他线程一直抢不到,这时候就需要线程同步 

 条件变量

  • 当一个线程互斥地访问某个变量时,它可能发现在其它线程改变状态之前,它什么也做不了。
  • 例如一个线程访问队列时,发现队列为空,它只能等待,直到其它线程将一个节点添加到队列中。这种情况就需要用到条件变量。 

 举例:有A,B,C三个人,一个盘子。B拿出苹果放到盘子上,另外两人就可以到盘子上拿。为了在放苹果的时候,其他人不能来拿,就要加锁,盘子就是临界区。因为另外两人想拿苹果,就一直申请锁,导致B放不了苹果。此时就需要一个铃铛。A,C两人在外面排队,当B放好苹果后就摇铃铛,此时A和C就会根据排队的顺序依次进去拿苹果。

上面的铃铛就是条件变量,人就是线程。摇铃铛后,可以规定是唤醒一个线程还是唤醒全部。

认识接口 

条件变量是 pthread_cond_t 的数据类型。它的使用跟前面互斥锁一样,可以定义成局部或者全局的。如果是全局或者静态的,可以直接使用 PTHREAD_COND_INITIALIZER 初始化。

如果是局部的,就用pthread_cond_init 函数初始化,使用完了就destroy销毁掉。

线程条件不满足时,线程就要等待,要在指定的条件变量上等待。

cond:要在这个条件变量上等待

等待完成后,就要进行唤醒。

pthread_cond_signal 表示唤醒一个线程。 pthread_cond_broadcast 表示唤醒所有线程。

条件变量接口使用例子:

#include 
#include 
#include 
#include 

const int num = 5;
pthread_mutex_t gmutex = PTHREAD_MUTEX_INITIALIZER;
pthread_cond_t gcond = PTHREAD_COND_INITIALIZER;

void *Wait(void *args)
{
    std::string name = static_cast(args);
    while (true)
    {
        pthread_mutex_lock(&gmutex);

        pthread_cond_wait(&gcond, &gmutex); // 这里就是线程等待的位置
        usleep(10000);
        std::cout << "I am : " << name << std::endl;

        pthread_mutex_unlock(&gmutex);

        // usleep(100000);
    }
}

int main()
{
    pthread_t threads[num];
    for (int i = 0; i < num; i++)
    {
        char *name = new char[1024];
        snprintf(name, 1024, "thread-%d", i + 1);
        pthread_create(threads + i, nullptr, Wait, (void *)name);
        usleep(10000);
    }

    sleep(1);
    // 唤醒其他线程
    while (true)
    {
        // pthread_cond_signal(&gcond);
        pthread_cond_broadcast(&gcond);
        std::cout << "唤醒一个线程...." << std::endl;
        sleep(2);
    }

    for (int i = 0; i < num; i++)
    {
        pthread_join(threads[i], nullptr);
    }

    return 0;
}

运行结果:

 生产者消费者模型

生产者消费者模式就是通过一个容器来解决生产者和消费者的强耦合问题。生产者和消费者彼此之间不直接通讯,而通过阻塞队列来进行通讯,所以生产者生产完数据之后不用等待消费者处理,直接扔给阻塞队列,消费者不找生产者要数据,而是直接从阻塞队列里取,阻塞队列就相当于一个缓冲区,平衡了生产者和消费者的处理能力。这个阻塞队列就是用来给生产者和消费者解耦的。

生产者消费者模型优点:

  • 解耦
  • 支持并发
  • 支持忙闲不均

 为了方便记忆,这里有一个“321”原则:

  1. 一个交易场所(一段内存空间)
  2. 两种角色(生产、消费角色)

  3. 三种关系(生产和生产、消费和消费 、生产和消费)前两种是互斥关系,最后一种是互斥和同步的关系

 基于BlockingQueue的生产者消费者模型

在多线程编程中阻塞队列(Blocking Queue)是一种常用于实现生产者和消费者模型的数据结构。其与普通的队列区别在于,当队列为空时,从队列获取元素的操作将会被阻塞,直到队列中被放入了元素;当队列满时,往队列里存放元素的操作也会被阻塞,直到有元素被从队列中取出

BlockQueue.hpp

#pragma once

#include 
#include 
#include 
#include 

const static int defaultcap = 5;

template 
class BlockQueue
{
private:
    bool IsFull()
    {
        return _block_queue.size() == _max_cap;
    }
    bool IsEmpty()
    {
        return _block_queue.empty();
    }

public:
    BlockQueue(int cap = defaultcap) : _max_cap(cap)
    {
        pthread_mutex_init(&_mutex, nullptr);
        pthread_cond_init(&_p_cond, nullptr);
        pthread_cond_init(&_c_cond, nullptr);
    }
    // 假设:2个消费者
    void Pop(T *out)
    {
        pthread_mutex_lock(&_mutex);
        while (IsEmpty()) // while可以保证代码的鲁棒性(健壮性),不用if,因为如果只有一个生产品
        {                 //,有两个消费者,一次性唤醒两个消费者的话,他们会竞争锁,其中一个拿完东西后,另一个才能重新拿到锁
                          //另一个拿到锁后,不用if的话,就跳过判断了,此时队列是空的,就会异常,所以用while
            // 添加尚未满足,但是线程被异常唤醒的情况,叫做伪唤醒!
            pthread_cond_wait(&_c_cond, &_mutex); // 两个消费者都在这里等待了
        }
        // 1. 没有空 || 2. 被唤醒了
        *out = _block_queue.front();
        _block_queue.pop();
        // if(_block_queue.size() > hight_water)
        //     pthread_cond_signal(&_p_cond);
        pthread_mutex_unlock(&_mutex);
        pthread_cond_signal(&_p_cond);
  
    }
    // 一个生产者
    void Equeue(const T &in)
    {
        pthread_mutex_lock(&_mutex);
        while (IsFull()) 
        {
            // 满了,生产者不能生产,必须等待
            // 可是在临界区里面啊!
            // 被调用的时候:除了让自己继续排队等待,还会自己释放传入的锁
            // 函数返回的时候,不就还在临界区了!
            // 返回时:必须先参与锁的竞争,重新加上锁,该函数才会返回!
            pthread_cond_wait(&_p_cond, &_mutex);
        }
        // 1. 没有满 || 2. 被唤醒了
        _block_queue.push(in); // 生产到阻塞队列
        pthread_mutex_unlock(&_mutex);
        // 让消费者消费
        pthread_cond_signal(&_c_cond);//解锁和唤醒的顺序可以交换,不影响
    }
    ~BlockQueue()
    {
        pthread_mutex_destroy(&_mutex);
        pthread_cond_destroy(&_p_cond);
        pthread_cond_destroy(&_c_cond);
    }

private:
    std::queue _block_queue; // 临界资源
    int _max_cap;
    pthread_mutex_t _mutex;
    pthread_cond_t _p_cond; // 生产者条件变量
    pthread_cond_t _c_cond; // 消费者条件变量

    // int low_water = _max_cap/3
    // int hight_water _max_cap/3*2
};

main.cc

#include "BlockQueue.hpp"
#include "Task.hpp"
#include 
#include 
#include 

void *Consumer(void *args)
{
    BlockQueue *bq = static_cast *>(args);
    while(true)
    {
        // 1. 获取数据
        task_t t;
        bq->Pop(&t);
        // 2. 处理数据
        // t.Excute();
        t();
        // std::cout << "Consumer -> " << t.result()  << std::endl;
    }
}

void *Productor(void *args)
{
    srand(time(nullptr) ^ getpid());
    BlockQueue *bq = static_cast *>(args);
    while(true)
    {
        // 1. 构建数据/任务
        // int x = rand() % 10 + 1; // [1, 10]
        // usleep(x * 1000);
        // int y = rand() % 10 + 1; // [1, 10]
        // Task t(x, y);
        // 2. 生产数据
        bq->Equeue(Download);
        std::cout << "Productor -> Download" << std::endl;

        sleep(1);
    }
}

int main()
{
    BlockQueue *bq = new BlockQueue();
    pthread_t c1,c2, p1,p2,p3;
    pthread_create(&c1, nullptr, Consumer, bq);
    pthread_create(&c2, nullptr, Consumer, bq);
    pthread_create(&p1, nullptr, Productor, bq);
    pthread_create(&p2, nullptr, Productor, bq);
    pthread_create(&p3, nullptr, Productor, bq);

    pthread_join(c1, nullptr);
    pthread_join(c2, nullptr);
    pthread_join(p1, nullptr);
    pthread_join(p2, nullptr);
    pthread_join(p3, nullptr);

    return 0;
}

Task.hpp

#pragma once

#include
#include

// typedef std::function task_t;
using task_t = std::function;

void Download()
{
    std::cout << "我是一个下载的任务" << std::endl;
}


// // 要做加法
// class Task
// {
// public:
//     Task()
//     {
//     }
//     Task(int x, int y) : _x(x), _y(y)
//     {
//     }
//     void Excute()
//     {
//         _result = _x + _y;
//     }
//     void operator ()()
//     {
//         Excute();
//     }
//     std::string debug()
//     {
//         std::string msg = std::to_string(_x) + "+" + std::to_string(_y) + "=?";
//         return msg;
//     }
//     std::string result()
//     {
//         std::string msg = std::to_string(_x) + "+" + std::to_string(_y) + "=" + std::to_string(_result);
//         return msg;
//     }

// private:
//     int _x;
//     int _y;
//     int _result;
// };

多线程生产者消费者模型高效是因为:一个生产者在生产任务的时候,其他生产者在构建任务,一个消费者在获取任务的时候,其他消费者在处理任务。

为什么线程在等待的时候,都是在加锁和解锁之间等待?

答:无论是生产者还是消费者,都必须先检查资源的状态。检查就是要访问,所以检查之前就要加锁,等待必须在临界区里进行等待,因为判断结果是在临界区里的。  

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

搜索文章

Tags

PV计算 带宽计算 流量带宽 服务器带宽 上行带宽 上行速率 什么是上行带宽? CC攻击 攻击怎么办 流量攻击 DDOS攻击 服务器被攻击怎么办 源IP 服务器 linux 运维 游戏 云计算 javascript 前端 chrome edge python MCP ssh 阿里云 网络 网络安全 网络协议 llama 算法 opencv 自然语言处理 神经网络 语言模型 ubuntu 进程 操作系统 进程控制 Ubuntu deepseek Ollama 模型联网 API CherryStudio harmonyos 华为 开发语言 typescript 计算机网络 数据库 centos oracle 关系型 安全 分布式 RTSP xop RTP RTSPServer 推流 视频 udp unity tcp/ip php android 宝塔面板访问不了 宝塔面板网站访问不了 宝塔面板怎么配置网站能访问 宝塔面板配置ip访问 宝塔面板配置域名访问教程 宝塔面板配置教程 Dell R750XS vue.js audio vue音乐播放器 vue播放音频文件 Audio音频播放器自定义样式 播放暂停进度条音量调节快进快退 自定义audio覆盖默认样式 GaN HEMT 氮化镓 单粒子烧毁 辐射损伤 辐照效应 ssl fastapi mcp mcp-proxy mcp-inspector fastapi-mcp agent sse 深度学习 YOLO 目标检测 计算机视觉 人工智能 filezilla 无法连接服务器 连接被服务器拒绝 vsftpd 331/530 HCIE 数通 macos adb numpy rust http java 鸿蒙 面试 性能优化 jdk intellij-idea 架构 c# Flask FastAPI Waitress Gunicorn uWSGI Uvicorn flutter Hyper-V WinRM TrustedHosts 笔记 计算机外设 电脑 mac 软件需求 golang 后端 ide ai AI编程 物联网 iot 华为云 YOLOv12 pycharm windows asm react.js 前端面试题 node.js 持续部署 vscode 代码调试 ipdb c++ oceanbase rc.local 开机自启 systemd 麒麟 微信 微信分享 Image wxopensdk ping++ 运维开发 conda 机器学习 深度优先 图论 并集查找 换根法 树上倍增 智能手机 NAS Termux Samba Linux ddos ollama llm docker 容器 chatgpt 大模型 llama3 Chatglm 开源大模型 mongodb 学习方法 科技 个人开发 word图片自动上传 word一键转存 复制word图片 复制word图文 复制word公式 粘贴word图文 粘贴word公式 jenkins gitee spring boot 数据结构 学习 c语言 经验分享 pytorch HarmonyOS Next 银河麒麟服务器操作系统 系统激活 sql KingBase ollama下载加速 uni-app 负载均衡 ESP32 单片机 豆瓣 追剧助手 迅雷 nas WSL2 rabbitmq Windsurf mysql aws googlecloud 开源 websocket tomcat LDAP 自动化 蓝耘科技 元生代平台工作流 ComfyUI C 环境变量 进程地址空间 实时音视频 maven intellij idea vim 智能路由器 外网访问 内网穿透 端口映射 redis mybatis github git ansible nginx dubbo 大数据 大数据平台 安装教程 GPU环境配置 Ubuntu22 CUDA PyTorch Anaconda安装 rust腐蚀 .net ffmpeg 音视频 僵尸进程 产品经理 agi microsoft 嵌入式硬件 温湿度数据上传到服务器 Arduino HTTP 信息与通信 gitlab 博客 Ubuntu Server Ubuntu 22.04.5 DeepSeek-R1 API接口 多线程服务器 Linux网络编程 监控 自动化运维 爬虫 数据挖掘 网络用户购物行为分析可视化平台 大数据毕业设计 1024程序员节 AI Agent pillow live555 rtsp rtp json html5 firefox https WSL win11 无法解析服务器的名称或地址 apache 小程序 微信小程序域名配置 微信小程序服务器域名 微信小程序合法域名 小程序配置业务域名 微信小程序需要域名吗 微信小程序添加域名 web安全 Kali Linux 黑客 渗透测试 信息收集 django flask web3.py vue3 HTML audio 控件组件 vue3 audio音乐播放器 Audio标签自定义样式默认 vue3播放音频文件音效音乐 自定义audio播放器样式 播放暂停调整声音大小下载文件 .netcore 创意 社区 源码剖析 rtsp实现步骤 流媒体开发 远程控制 远程看看 远程协助 cpu 内存 实时 使用 前端框架 C语言 低代码 AIGC jmeter 软件测试 DigitalOcean GPU服务器购买 GPU服务器哪里有 GPU服务器 Qwen2.5-coder 离线部署 YOLOv8 NPU Atlas800 A300I pro asi_bench cuda cudnn anaconda 微服务 springcloud Dify 嵌入式 linux驱动开发 arm开发 统信UOS bonding 链路聚合 mount挂载磁盘 wrong fs type LVM挂载磁盘 Centos7.9 pip sqlserver qt stm32项目 stm32 进程信号 transformer debian PVE dell服务器 go 代理模式 硬件架构 系统架构 JAVA Java spring cloud linux 命令 sed 命令 集成学习 集成测试 IIS .net core Hosting Bundle .NET Framework vs2022 RoboVLM 通用机器人策略 VLA设计哲学 vlm fot robot 视觉语言动作模型 具身智能 XFS xfs文件系统损坏 I_O error es jvm 云原生 devops springboot shell 磁盘监控 部署 服务器配置 设置代理 实用教程 图形化界面 jupyter 生物信息学 wsl2 wsl openEuler virtualenv Google pay Apple pay dify ragflow gitea 服务器主板 AI芯片 android studio 交互 GCC crosstool-ng html file server http server web server zabbix bash 远程连接 rdp 实验 minicom 串口调试工具 测试工具 远程工作 DNS 多层架构 解耦 minio 腾讯云 postgresql eureka Linux awk awk函数 awk结构 awk内置变量 awk参数 awk脚本 awk详解 spring MQTT协议 消息服务器 代码 DeepSeek 分析解读 职场和发展 交换机 硬件 设备 GPU PCI-Express 微信小程序 机器人 jetty undertow express 华为od UOS 统信操作系统 yum 5G 3GPP 卫星通信 kylin 指令 mq rocketmq kafka Erlang OTP gen_server 热代码交换 事务语义 r语言 数据可视化 ip命令 新增网卡 新增IP 启动网卡 MNN Qwen ip ui sqlite3 音乐服务器 Navidrome 音流 docker命令大全 MacMini Mac 迷你主机 mini Apple 远程桌面 大语言模型 ruoyi DeepSeek行业应用 Heroku 网站部署 小艺 Pura X 服务器繁忙 串口服务器 excel 并查集 leetcode hugo 命令 Docker Hub docker pull 镜像源 daemon.json Ubuntu 24 常用命令 Ubuntu 24 Ubuntu vi 异常处理 思科模拟器 思科 Cisco prometheus kubernetes 监控k8s 监控kubernetes java-ee 压力测试 测试用例 功能测试 kind AI写作 AI作画 react next.js 部署next.js QQ 聊天室 WebUI DeepSeek V3 AI大模型 其他 gpu算力 编辑器 camera Arduino 电子信息 VR手套 数据手套 动捕手套 动捕数据手套 LLM 大模型面经 Deepseek 大模型学习 muduo X11 Xming k8s 弹性计算 虚拟化 KVM 计算虚拟化 弹性裸金属 efficientVIT YOLOv8替换主干网络 TOLOv8 av1 电视盒子 机顶盒ROM 魔百盒刷机 游戏程序 windwos防火墙 defender防火墙 win防火墙白名单 防火墙白名单效果 防火墙只允许指定应用上网 防火墙允许指定上网其它禁止 3d 数学建模 数据库架构 数据管理 数据治理 数据编织 数据虚拟化 命名管道 客户端与服务端通信 idm grafana 直流充电桩 充电桩 IPMI unix 漏洞 远程 执行 sshpass 操作 跨域 protobuf 序列化和反序列化 安装 安全威胁分析 vscode 1.86 webstorm Linux无人智慧超市 LInux多线程服务器 QT项目 LInux项目 单片机项目 Nuxt.js 缓存 SSH Xterminal 强制清理 强制删除 mac废纸篓 netty gpt linux上传下载 ArcTS 登录 ArcUI GridItem 鸿蒙系统 arkUI 云服务器 裸金属服务器 弹性裸金属服务器 p2p unity3d ios Ubuntu DeepSeek DeepSeek Ubuntu DeepSeek 本地部署 DeepSeek 知识库 DeepSeek 私有化知识库 本地部署 DeepSeek DeepSeek 私有化部署 银河麒麟 kylin v10 麒麟 v10 监控k8s集群 集群内prometheus Playwright 自动化测试 VMware安装Ubuntu Ubuntu安装k8s webrtc 网络穿透 chrome devtools selenium chromedriver SSH 服务 SSH Server OpenSSH Server Netty 即时通信 NIO Cursor 状态管理的 UDP 服务器 Arduino RTOS Ark-TS语言 驱动开发 mcu k8s集群资源管理 云原生开发 postman mock mock server 模拟服务器 mock服务器 Postman内置变量 Postman随机数据 hadoop okhttp CORS 虚拟机 VMware vmware 卡死 ros2 moveit 机器人运动 jar gradle nuxt3 C# MQTTS 双向认证 emqx matlab ai小智 语音助手 ai小智配网 ai小智教程 智能硬件 esp32语音助手 diy语音助手 边缘计算 cmos IIS服务器 IIS性能 日志监控 能力提升 面试宝典 技术 IT信息化 pygame code-server MQTT mosquitto 消息队列 SysBench 基准测试 数据分析 telnet 远程登录 小游戏 五子棋 计算机 程序员 VMware安装mocOS macOS系统安装 sdkman sqlite MS Materials openssl 密码学 gateway Clion Nova ResharperC++引擎 Centos7 远程开发 业界资讯 wireshark 显示过滤器 ICMP Wireshark安装 模拟退火算法 gcc kamailio sip VoIP ci/cd echarts 信息可视化 网页设计 nvidia 数据库系统 rime 软件工程 hibernate 课程设计 remote-ssh mysql离线安装 ubuntu22.04 mysql8.0 源码 毕业设计 国产操作系统 ukui 麒麟kylinos openeuler 统信 虚拟机安装 系统安全 框架搭建 混合开发 环境安装 JDK 系统开发 binder 车载系统 framework 源码环境 visual studio code ESXi Dell HPE 联想 浪潮 命令行 基础入门 编程 Linux PID centos-root /dev/mapper yum clean all df -h / du -sh elasticsearch asp.net大文件上传 asp.net大文件上传源码 ASP.NET断点续传 asp.net上传文件夹 asp.net上传大文件 .net core断点续传 .net mvc断点续传 京东云 web 火绒安全 孤岛惊魂4 uniapp vue 数据集 恒源云 RAID RAID技术 磁盘 存储 vSphere vCenter 软件定义数据中心 sddc RTMP 应用层 playbook 剧本 雨云 NPS open webui 政务 分布式系统 监控运维 Prometheus Grafana 传统数据库升级 银行 LLMs mariadb selete 高级IO IPMITOOL BMC 硬件管理 opcua opcda KEPServer安装 oneapi tcpdump 大模型微调 网站搭建 serv00 微信开放平台 微信公众平台 微信公众号配置 飞牛NAS 飞牛OS MacBook Pro pdf 设计模式 VSCode cnn 华为认证 网络工程师 蓝桥杯 移动云 springsecurity6 oauth2 授权服务器 token sas 鲲鹏 FTP 服务器 visualstudio 企业微信 Linux24.04 deepin Reactor C++ SSL 域名 Anolis nginx安装 linux插件下载 etcd 数据安全 RBAC 服务器数据恢复 数据恢复 存储数据恢复 raid5数据恢复 磁盘阵列数据恢复 金融 NFS Trae IDE AI 原生集成开发环境 Trae AI SWAT 配置文件 服务管理 网络共享 硬件工程 嵌入式实习 Redis Desktop pyqt 半虚拟化 硬件虚拟化 Hypervisor Kylin-Server 服务器安装 iftop 网络流量监控 EasyConnect micropython esp32 mqtt Cline 技能大赛 make命令 makefile文件 ecmascript nextjs reactjs 向日葵 RustDesk自建服务器 rustdesk服务器 docker rustdesk 黑客技术 docker搭建nacos详解 docker部署nacos docker安装nacos 腾讯云搭建nacos centos7搭建nacos 流式接口 本地部署 api springboot远程调试 java项目远程debug docker远程debug java项目远程调试 springboot远程 pgpool 搜索引擎 ssrf 失效的访问控制 WebRTC openwrt 田俊楠 ux 多线程 安卓 svn open Euler dde 程序人生 eNSP 网络规划 VLAN 企业网络 服务器管理 宝塔面板 配置教程 网站管理 网工 opensearch helm linux环境变量 xrdp k8s资源监控 annotations自动化 自动化监控 监控service 监控jvm safari 系统 string模拟实现 深拷贝 浅拷贝 经典的string类问题 三个swap 游戏服务器 TrinityCore 魔兽世界 客户端 多进程 adobe elk Python 网络编程 聊天服务器 套接字 TCP Socket 文件系统 路径解析 环境迁移 崖山数据库 YashanDB Ubuntu 24.04.1 轻量级服务器 redhat W5500 OLED u8g2 TCP服务器 chfs ubuntu 16.04 群晖 文件分享 WebDAV 中间件 iis firewalld 雨云服务器 Docker Compose docker compose docker-compose 银河麒麟操作系统 国产化 rpc 远程过程调用 Windows环境 RAGFLOW RAG 检索增强生成 文档解析 大模型垂直应用 直播推流 服务器部署ai模型 rsyslog bug TRAE 毕设 相差8小时 UTC 时间 OD机试真题 华为OD机试真题 服务器能耗统计 Portainer搭建 Portainer使用 Portainer使用详解 Portainer详解 Portainer portainer ue4 着色器 ue5 虚幻 kvm 无桌面 媒体 risc-v 目标跟踪 OpenVINO 推理应用 CPU 主板 电源 网卡 医疗APP开发 app开发 ipython DevEco Studio 开机自启动 北亚数据恢复 oracle数据恢复 联想开天P90Z装win10 Invalid Host allowedHosts ceph AI代码编辑器 bootstrap threejs 3D bot Docker css ecm bpm 繁忙 解决办法 替代网站 汇总推荐 AI推理 压测 ECS 深度求索 私域 知识库 dba 宕机切换 服务器宕机 kali 共享文件夹 XCC Lenovo 图像处理 Minecraft embedding CLion list skynet 腾讯云大模型知识引擎 zotero 同步失败 报错 thingsboard curl wget fpga开发 灵办AI iDRAC R720xd freebsd 昇腾 npu 超融合 服务器无法访问 ip地址无法访问 无法访问宝塔面板 宝塔面板打不开 edge浏览器 前后端分离 tensorflow trae arm 相机 权限 键盘 FunASR ASR 佛山戴尔服务器维修 佛山三水服务器维修 自动化任务管理 prompt easyui langchain IO模型 我的世界服务器搭建 飞牛nas fnos 王者荣耀 Wi-Fi log4j Spring Security 我的世界 我的世界联机 数码 yum源切换 更换国内yum源 ubuntu24.04.1 ISO镜像作为本地源 视觉检测 n8n 工作流 workflow 云电竞 云电脑 todesk vr ocr VMware创建虚拟机 ruby tidb GLIBC 文心一言 saltstack DocFlow 安全架构 备份SQL Server数据库 数据库备份 傲梅企业备份网络版 免费域名 域名解析 远程服务 IM即时通讯 剪切板对通 HTML FORMAT 信号 IMX317 MIPI H265 VCU Linux的权限 wordpress 无法访问wordpess后台 打开网站页面错乱 linux宝塔面板 wordpress更换服务器 怎么卸载MySQL MySQL怎么卸载干净 MySQL卸载重新安装教程 MySQL5.7卸载 Linux卸载MySQL8.0 如何卸载MySQL教程 MySQL卸载与安装 宠物 免费学习 宠物领养 宠物平台 gaussdb dns是什么 如何设置电脑dns dns应该如何设置 x64 SIGSEGV SSE xmm0 稳定性 看门狗 xss 在线预览 xlsx xls文件 在浏览器直接打开解析xls表格 前端实现vue3打开excel 文件地址url或接口文档流二进 Typore pppoe radius AI agent 架构与原理 云服务 单例模式 国标28181 视频监控 监控接入 语音广播 流程 SIP SDP 交叉编译 小番茄C盘清理 便捷易用C盘清理工具 小番茄C盘清理的优势尽显何处? 教你深度体验小番茄C盘清理 C盘变红?!不知所措? C盘瘦身后电脑会发生什么变化? 程序员创富 算力 Claude Radius npm less IDEA AnythingLLM AnythingLLM安装 基础环境 策略模式 Linux环境 流水线 脚本式流水线 iBMC UltraISO 输入法 lua vue-i18n 国际化多语言 vue2中英文切换详细教程 如何动态加载i18n语言包 把语言json放到服务器调用 前端调用api获取语言配置文件 DenseNet 网络结构图 社交电子 CrewAI 高效远程协作 TrustViewer体验 跨设备操作便利 智能远程控制 aarch64 编译安装 HPC qemu libvirt EMQX 通信协议 WebVM 7z 音乐库 飞牛 junit can 线程池 keepalived tcp sonoma 自动更新 显示管理器 lightdm gdm xshell termius iterm2 同步 备份 建站 neo4j 数据仓库 数据库开发 database 阻塞队列 生产者消费者模型 服务器崩坏原因 laravel midjourney 无人机 xcode 服务网格 istio 网络药理学 生信 gromacs 分子动力学模拟 MD 动力学模拟 sentinel USB网络共享 自动驾驶 自动化编程 HTTP 服务器控制 ESP32 DeepSeek 备选 网站 调用 示例 银河麒麟桌面操作系统 Kylin OS ssh漏洞 ssh9.9p2 CVE-2025-23419 AD域 致远OA OA服务器 服务器磁盘扩容 游戏机 xml fd 文件描述符 软件构建 etl vasp安装 查询数据库服务IP地址 SQL Server lsb_release /etc/issue /proc/version uname -r 查看ubuntu版本 虚拟显示器 语音识别 AutoDL 干货分享 黑客工具 密码爆破 技术共享 语法 ros EtherNet/IP串口网关 EIP转RS485 EIP转Modbus EtherNet/IP网关协议 EIP转RS485网关 EIP串口服务器 图形渲染 重启 排查 系统重启 日志 原因 黑苹果 powerpoint eclipse qt项目 qt项目实战 qt教程 捆绑 链接 谷歌浏览器 youtube google gmail 执法记录仪 智能安全帽 smarteye alias unalias 别名 Ubuntu共享文件夹 共享目录 Linux共享文件夹 银河麒麟高级服务器 外接硬盘 Kylin 根服务器 反向代理 实战案例 华为机试 单元测试 Docker引擎已经停止 Docker无法使用 WSL进度一直是0 镜像加速地址 perf AISphereButler EtherCAT转Modbus ECT转Modbus协议 EtherCAT转485网关 ECT转Modbus串口网关 EtherCAT转485协议 ECT转Modbus网关 hive DBeaver kerberos cfssl 产测工具框架 IMX6ULL 管理框架 regedit 开机启动 回显服务器 UDP的API使用 openstack Xen 做raid 装系统 Java Applet URL操作 服务器建立 Socket编程 网络文件读取 大模型入门 大模型教程 升级 CVE-2024-7347 影刀 #影刀RPA# VPS frp 内网服务器 内网代理 内网通信 VM搭建win2012 win2012应急响应靶机搭建 攻击者获取服务器权限 上传wakaung病毒 应急响应并溯源 挖矿病毒处置 应急响应综合性靶场 Hive环境搭建 hive3环境 Hive远程模式 需求分析 规格说明书 webgl 考研 本地部署AI大模型 Logstash 日志采集 LInux flash-attention 飞书 实习 web3 c uni-file-picker 拍摄从相册选择 uni.uploadFile H5上传图片 微信小程序上传图片 矩阵 ROS big data 聚类 AP配网 AK配网 小程序AP配网和AK配网教程 WIFI设备配网小程序UDP开 实时互动 mamba 虚拟局域网 DIFY VS Code 单一职责原则 epoll 工业4.0 显卡驱动 AD 域管理 大模型应用 spark HistoryServer Spark YARN jobhistory GoogLeNet Headless Linux OpenSSH 移动魔百盒 游戏开发 USB转串口 CH340 gpt-3 asp.net大文件上传下载 邮件APP 免费软件 小智AI服务端 xiaozhi TTS 可信计算技术 网络攻击模型 僵尸世界大战 游戏服务器搭建 匿名管道 zookeeper HarmonyOS nfs 序列化反序列化 IPv4 子网掩码 公网IP 私有IP 主从复制 磁盘镜像 服务器镜像 服务器实时复制 实时文件备份 banner 人工智能生成内容 自定义客户端 SAS 游戏引擎 minecraft seleium 迁移指南 GIS 遥感 WebGIS 大大通 第三代半导体 碳化硅 docker run 数据卷挂载 交互模式 毕昇JDK 链表 Windows ai工具 v10 软件 ldap armbian u-boot iphone rclone AList webdav fnOS RAGFlow 本地知识库部署 DeepSeek R1 模型 P2P HDLC 双系统 GRUB引导 Linux技巧 线程 URL 分布式训练 项目部署到linux服务器 项目部署过程 ftp matplotlib MI300x Linux的基础指令 vscode1.86 1.86版本 ssh远程连接 uv glibc LLM Web APP Streamlit cpp-httplib SRS 流媒体 直播 c/c++ 串口 开发环境 SSL证书 历史版本 下载 sysctl.conf vm.nr_hugepages 软负载 AI-native Docker Desktop UOS1070e yolov8 clickhouse 代码托管服务 wps outlook rnn MacOS录屏软件 python3.11 Vmamba 视频编解码 dash 正则表达式 性能测试 读写锁 信号处理 odoo 服务器动作 Server action vpn seatunnel g++ g++13 nac 802.1 portal Cookie arcgis 高效日志打印 串口通信日志 服务器日志 系统状态监控日志 异常记录日志 rustdesk 串口驱动 CH341 uart 485 模拟器 教程 上传视频至服务器代码 vue3批量上传多个视频并预览 如何实现将本地视频上传到网页 element plu视频上传 ant design vue vue3本地上传视频及预览移除 用户缓冲区 模拟实现 pyautogui dns 支付 微信支付 开放平台 ArkTs ArkUI 互信 rag ragflow 源码启动 swoole 运维监控 三级等保 服务器审计日志备份 FTP服务器 bcompare Beyond Compare Mac内存不够用怎么办 多个客户端访问 IO多路复用 TCP相关API 代理 cocoapods 软考 yaml Ultralytics 可视化 SenseVoice 版本 C++软件实战问题排查经验分享 0xfeeefeee 0xcdcdcdcd 动态库加载失败 程序启动失败 程序运行权限 标准用户权限与管理员权限 CDN 宝塔 EMUI 回退 降级 安防软件 端口测试 DeepSeek r1 Open WebUI Kali 渗透 linux安装配置 cd 目录切换 tailscale derp derper 中转 OpenManus triton 模型分析 线性代数 电商平台 大文件分片上传断点续传及进度条 如何批量上传超大文件并显示进度 axios大文件切片上传详细教 node服务器合并切片 vue3大文件上传报错提示错误 大文件秒传跨域报错cors 嵌入式Linux IPC NLP模型 NLP 自学笔记 小米 澎湃OS Android apt DOIT 四博智联 Unity Dedicated Server Host Client 无头主机 miniapp 真机调试 调试 debug 断点 网络API请求调试方法 国内源 flink cursor MCP server C/S windows日志 trea idea 常用命令 文本命令 目录命令 阿里云ECS Jellyfin LORA composer bat TrueLicense H3C Ubuntu22.04 开发人员主页 强化学习 iventoy VmWare OpenEuler css3 word 开发 元服务 应用上架 deep learning docker部署翻译组件 docker部署deepl docker搭建deepl java对接deepl 翻译组件使用 换源 Debian Unity插件 镜像 nlp deekseek 代理服务器 UDP 嵌入式系统开发 vu大文件秒传跨域报错cors fast 极限编程 Xinference db 带外管理 金仓数据库 2025 征文 数据库平替用金仓 服务器时间 大模型推理 ShenTong perl 李心怡 大模型部署 查看显卡进程 fuser ArtTS 云耀服务器 Python基础 Python教程 Python技巧 WLAN 上传视频文件到服务器 uniApp本地上传视频并预览 uniapp移动端h5网页 uniapp微信小程序上传视频 uniapp app端视频上传 uniapp uview组件库 个人博客 KylinV10 麒麟操作系统 Vmware zip unzip nvm whistle 软链接 硬链接 网络爬虫 CentOS 物联网开发 Deepseek-R1 私有化部署 推理模型 HAProxy SEO 增强现实 沉浸式体验 应用场景 技术实现 案例分析 AR 树莓派 VNC 域名服务 DHCP 符号链接 配置 虚幻引擎 virtualbox 信创 信创终端 中科方德 问题解决 ubuntu24 vivado24 kotlin ssh远程登录 企业网络规划 华为eNSP 浏览器开发 AI浏览器 hosts 合成模型 扩散模型 图像生成 Sealos rancher 烟花代码 烟花 元旦 OpenHarmony 性能调优 安全代理 加解密 Yakit yaklang 鸿蒙开发 移动开发 程序 性能分析 推荐算法 SVN Server tortoise svn 流量运营 沙盒 sequoiaDB dity make ip协议 HarmonyOS NEXT 原生鸿蒙 prometheus数据采集 prometheus数据模型 prometheus特点 searxng PPI String Cytoscape CytoHubba 计算生物学 生物信息 基因组 TCP协议 抗锯齿 英语 fstab 拓扑图 llama.cpp 知识图谱 firewall top Linux top top命令详解 top命令重点 top常用参数 milvus 本地化部署 onlyoffice 在线office ubuntu20.04 ros1 Noetic 20.04 apt 安装 react native chrome 浏览器下载 chrome 下载安装 谷歌浏览器下载 私有化 MVS 海康威视相机 autodl 状态模式 玩机技巧 软件分享 软件图标 IMM rtsp服务器 rtsp server android rtsp服务 安卓rtsp服务器 移动端rtsp服务 大牛直播SDK CentOS Stream proxy模式 MySql figma Qwen2.5-VL vllm grub 版本升级 扩容 磁盘清理 hexo harmonyOS面试题 wpf 容器技术 SSH 密钥生成 SSH 公钥 私钥 生成 deepseek r1 win服务器架设 windows server docker搭建pg docker搭建pgsql pg授权 postgresql使用 postgresql搭建 iperf3 带宽测试 java-rocketmq 对比 工具 meld DiffMerge 内网环境 h.264 QT 5.12.12 QT开发环境 Ubuntu18.04 粘包问题 环境配置 lio-sam SLAM 软件卸载 系统清理 网卡的名称修改 eth0 ens33 项目部署 ABAP navicat 存储维护 NetApp存储 EMC存储 deployment daemonset statefulset cronjob Attention AI Agent 字节智能运维 办公自动化 自动化生成 pdf教程 visual studio 健康医疗 互联网医院 MDK 嵌入式开发工具 论文笔记 sublime text rpa PX4 MacOS ranger MySQL8.0 k8s二次开发 集群管理 智能音箱 智能家居 Linux find grep 钉钉 端口聚合 windows11 System V共享内存 进程通信 gnu 抓包工具 kernel 网络建设与运维 jina 防火墙 NAT转发 NAT Server 中兴光猫 换光猫 网络桥接 自己换光猫 HiCar CarLife+ CarPlay QT RK3588 wsgiref Web 服务器网关接口 Node-Red 编程工具 流编程 docker desktop image 多端开发 智慧分发 应用生态 鸿蒙OS xpath定位元素 ardunio BLE 端口 查看 ss 云桌面 微软 AD域控 证书服务器 su sudo 浏览器自动化 解决方案 安装MySQL 通信工程 毕业 conda配置 conda镜像源 风扇控制软件 搭建个人相关服务器 docker部署Python vnc yum换源 开机黑屏 热榜 欧标 OCPP macOS 电视剧收视率分析与可视化平台 数字证书 签署证书 mm-wiki搭建 linux搭建mm-wiki mm-wiki搭建与使用 mm-wiki使用 mm-wiki详解 js 离线部署dify 论文阅读 智能电视 网络搭建 神州数码 神州数码云平台 云平台 服务器正确解析请求体 AI员工 IO 搜狗输入法 中文输入法 dock 加速 nosql yolov5 MobaXterm 区块链 Linux权限 权限命令 特殊权限 多路转接 WSL2 上安装 Ubuntu 像素流送api 像素流送UE4 像素流送卡顿 像素流送并发支持 西门子PLC 通讯 进程优先级 调度队列 进程切换 达梦 DM8 接口优化 MAVROS 四旋翼无人机