• 【Linux】一文带你理解清楚 同步与互斥(附大量C++理解代码和理解图片,逻辑清晰-通俗易懂)

【Linux】一文带你理解清楚 同步与互斥(附大量C++理解代码和理解图片,逻辑清晰-通俗易懂)

2025-04-24 19:34:21 1 阅读


每日激励:“不设限和自我肯定的心态:I can do all things。 — Stephen Curry”

绪论​:
本章是Linux线程中非常重要的概念,它不仅在Linux中非常重要同样也是在日常工作项目中常用的两种方法,通过互斥防止多个线程访问同一个资源时导致数据的问题,以及再次通过同步的关系让多个线程之间的互斥更加的有序,本章将通过知识 + 实例 的方式带你轻松认识清楚到底什么是常见的互斥和同步。
————————
早关注不迷路,话不多说安全带系好,发车啦(建议电脑观看)。

1.线程的互斥

背景

在程序中部分资源是共享的,如全局的东西所有线程都能访问得到。
对此当多个线程同时访问这种共享资源时,就可能导致数据不一致

解决方法

  1. 任何一个时刻,只允许一个线程正在访问共享资源—临界资源(也就共享的数据)
  2. 我们把访问进程中访问临界资源的代码—临界区(被保护的区域)

也就有了概念:

  • 互斥:如何时刻,互斥保证有且只有一个执行流进入临界区
  • 原子性:不会被任何调度机制打断的操作,该操作只有两态,要么完成,要么未完成

1.1 互斥的案例(了解)

对于c/c++代码会经过编译变成汇编,而某些代码的底层可能不止一条汇编语句就肯定不是原子的(因为要同时考虑三个语句的真假)如下面的变量a++的底层:

因为时间片切换,可能在一个代码的底层的多个语句中的第二个语句时间片就到了,导致未执行完就切走了,走之前会保存好上下文数据。轮到执行其他线程,假设其他线程也同样访问相同的内存资源,并修改了该资源,当后面时间片到了。又轮到一开始的线程,他会继续从未完成的部分开始,而非先读取内存数据,这样就会导致前面的线程的任务白做(因为一开始的线程的数据时之前的数据,他修改后就会导致数据回到之前)!

多线程并发访问全部整形的汇编,若不是原子的,就会有数据不一致的并发问题。

其中认识到CPU中执行过程:

  1. 算:算术运算
  2. 逻:逻辑运算(真假)
  3. 中:处理内外中断
  4. 控:控制单元(时钟控制)

其中 if 判断语句同样不是原子的(本质也和上面的++一样需要到CPU中去处理判断,这样汇编语句就有多个)!数据在内存中,本质是被线程共享的,数据被读取到寄存器汇总,本质变成了线程的上下文,它是属于线程私有数据数据。

1.2 互斥锁 函数(了解)

  1. 定义锁

    1. 全局锁:pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER
      此时定义的全局锁就能直接使用,并且不用销毁。
    2. 初始化锁:int pthread_mutex_init(pthread_mutex_t *restrict mutex, const pthread_mutexattr_t *restrict attr)
      attr是锁的属性,暂不考虑写成nullptr
  2. 销毁锁

    1. int pthread_mutex_destroy(pthread_mutex_t *mutex)

当在局部创建一把锁时,需要对其进行初始化操作:

  1. pthread_mutex_init
  2. pthread_mutex_destory,也就代表使用完后需要摧毁

因为自定义的函数会使用到锁所以需要把它当参数传递进去

  1. 加锁pthread_mutex_lock(pthread_mutex_t*mutex)

  2. 解锁pthread_mutex_unlock(pthread_mutex_t*mutex)

申请成功返回0,失败返回错误码

注意事项:

  1. 加锁时我们要尽可能的少给代码块加锁
  2. 一般加锁是给临界区加锁
  3. 申请锁本身是安全,原子的
  4. 一旦有了公共资源,程序员就需要保证是加锁的!

根据互斥的定义:

任何时刻只允许一个线程申请锁成功!其他线程申请失败时,会在mutex锁上进行阻塞,本质也就等待。

也就推出了不阻塞的锁:

  1. 不阻塞的申请锁:pthread_mutex_trylock(pthread_mutex_t*mutex)

线程在临界区中访问加锁的临界资源的时候是可能发生切换,但即便切换了别人也仍然不能访问。(理解成,一个房间只要有钥匙才能进入,而当我们进入后我们出去时是拿着钥匙走的,别人同样还 是进不去)

1.3 简单互斥锁(源码)

//main.cpp
#include
#include
#include
#include"LockGuard.hpp"
#include"Thread.hpp"
#include
#include
#include
using namespace std;
// 应用方的视角

//为了能同时传递线程名和锁变量
//就构建ThreadData类,让其可以直接一起传递进去
class ThreadData
{
public:
    ThreadData(string& name,pthread_mutex_t* lock)
    :threadname(name),pmutex(lock)
    {}
public:
    string threadname;
    pthread_mutex_t * pmutex;
};

void Print(int num)
{
    while (num)
    {
        std::cout << "hello world: " << num-- << std::endl;
        sleep(1);
    }
}

int ticket = 10000; // 全局的共享资源

// pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER; //全局锁,不用init初始化

void GetTicket(ThreadData *td)//抢票原理:
{
    while (true)
    {

        {   //通过花括号括其来的部分为一个代码块也就是确定了临界区

            //pthread_mutex_lock(mutex);//加锁
            LockGuard lockguard(td->pmutex);
            //LockGuard类会自动的申请锁,并且释放锁(因为当他构造和析构时就会执行加锁解锁操作!)
            if (ticket > 0) // 4. 一个线程在临界区中访问临界资源的时候,可不可能发生切换?可能,完全允许!!
            {
                // 充当抢票花费的时间
                usleep(1000);
                printf("%s get a ticket: %d
",td->threadname.c_str(),ticket);
                ticket--;
                // pthread_mutex_unlock(mutex);//解锁
            }
            else
            {
                // pthread_mutex_unlock(mutex);//解锁
                break;
            }
        }

    }
}

string GetThreadName()
{
    static int number = 1;
    char name[64];
    snprintf(name, sizeof(name), "Thread-%d", number++);
    return name;
}

int main()
{
    pthread_mutex_t mutex;
    pthread_mutex_init(&mutex,nullptr);

    string name1 = GetThreadName();
    ThreadData *td1 = new ThreadData(name1,&mutex);//构造类

    Thread<ThreadData*> t1(td1, GetTicket,name1);
    //把ThreadData这个类做参数传递进去,在内部就能同时拿到

    string name2 = GetThreadName();
    ThreadData *td2 = new ThreadData(name2,&mutex);
    Thread<ThreadData*> t2(td2,GetTicket,name2);

    string name3 = GetThreadName();
    ThreadData *td3 = new ThreadData(name3,&mutex);
    Thread<ThreadData*> t3(td3, GetTicket, name3);

    string name4 = GetThreadName();
    ThreadData *td4 = new ThreadData(name4,&mutex);
    Thread<ThreadData*> t4(td4, GetTicket, name4);

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

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

    pthread_mutex_destroy(&mutex);

    delete td1;
    delete td2;
    delete td3;
    delete td4;

    return 0;
}
//Thread.hpp:
#pragma once
#include
#include
#include
#include
using namespace std;

//typedef function func_t

//注意!!
//此处的修改,因为带有模板所以后面用该类型变成fun_t 
template<class T>
using func_t = function<void(T)>;//返回值void 参数为;

//加上模板,这个类型是给传进来的参数数据data的!
template<class T>
class Thread
{
public:
    Thread(T data, func_t<T> func, const string& name)
    :_tid(0),_name(name),_isrunning(false),_func(func),_data(data)
    {}

//因为在类内的函数默认是有this指针的这样就会导致pthread_create的threadrotine的类型不匹配而导致的无法传参
//所以解决方法就是改成静态函数,但此时又不能使用成员变量了,所哟把参数args改成this传递进来!
    static void *ThreadRotine(void* args)
    {
        Thread *ts = static_cast<Thread*>(args);

        ts->_func(ts->_data);
        return nullptr;
    }

    bool Start()                                  
    {
        int n = pthread_create(&_tid,nullptr,ThreadRotine,this);
        if(n == 0)
        {
            _isrunning = true;
            return true;
        }
        else return false;
    }  

    string Threadname()
    {
        return _name;
    }


    bool Join()
    {
        if(!_isrunning) return true;
        int n = pthread_join(_tid,nullptr);
        if(n==0)
        {
            _isrunning = false;
            return true;
        }
        return false; 

    }

    bool Isrunning()
    {
        return _isrunning;
    }

    ~Thread()
    {}

private:
    string _name;
    func_t<T> _func;

    pthread_t _tid;//创建自动形成
    bool _isrunning;
    T _data;
    //pid_t tid;
};

1.3.附加:封装锁

  • 把锁封装成一个类实现,当构造时加锁,析构时释放。
  • 这样把该类写在一个代码块中当成局部变量,这样就能让其在代码块中进行加锁,出代码块析构解锁

LockGuard.hpp:

#pragma once

#include 

class Mutex
{
public:
    Mutex(pthread_mutex_t *lock) : _lock(lock)
    {}
    
    void Lock()

    {
        pthread_mutex_lock(_lock);
    }
    void Unlock()

    {
        pthread_mutex_unlock(_lock);
    }

    ~Mutex()
    {}

private:
    pthread_mutex_t *_lock; // 不定义锁,默认外面会创建传进来
};

class LockGuard
{
public:
     LockGuard(pthread_mutex_t *lock) : _mutex(lock)

    {
        _mutex.Lock();
        
    }
    ~LockGuard()
    {
         _mutex.Unlock();
        
    }

private:
    Mutex _mutex;
};

上述代码就能实现一个基本的抢票机制,通过互斥锁就不会导致数据错乱

ps:远程拷贝:scp 用户名@IP地址:路径(后输入密码即可)

但有可能一个票被一个人全部都抢了,对其他线程就形成了饥饿问题,他的原理是一个线程在申请完锁,解锁后又立马申请锁资源。


要结局饥饿问题,互斥是无法解决的,也就是同步解决

  • 当一个线程申请完锁后不能再立马申请锁资(也就是有一定的顺序性),请看目录中的同步

1.4 互斥锁的本质

大多数体系结构都提供了swap/exchang指令,该指令的作用是把寄存器和内存单元的数据相交换

底层汇编原理:

exchang eax mem_addr(也就是把寄存器eax的内容和内存数据mem_addr交换的汇编语句),因为只有一条主要语句所以交换的过程是原子的,其他语句执行时即使被切换,也会把自身数据(%al寄存器的数据)带走,也就不会影响其他线程。

互斥锁的实现原理:

解释:

每个线程要执行加锁时首先都会先move 0到%al寄存器中,然后在和mutex的数据进行交换,然后判断寄存器%al中的值>0则表示加锁成功,反之则加锁失败在加锁处挂起等待!
加锁失败的情况,因为有人已经在使用该锁,所以会把内存中的mutex改为了0(他也会执行第一个move操作),当他没有归还锁资源时,mutex中的值为0,当别人一交换则%al寄存器中就会变成0就会挂起等待了。(这样也就实现了互斥锁,加锁的区域只能由一个线程使用!)

所以加锁的本质就是:

  • 寄存器和内存数据的交换(因为只有一条语句所以交换的过程是原子的,xchgb %al mutex)

  • 解锁的原理就是基于加锁的,把内存数据mutex改回1,并唤醒等待挂起的线程即可。

  • 加锁原则:谁加锁,谁解锁


1.5 认识:可重入 与 线程安全

线程安全

  1. 多个线程并发同一段代码时,不会出现不同的结果。
  2. 若出现问题(如:崩溃,数据异常),则是线程不安全的。
  3. 其中线程安全和可重入本质是一样的,只不过可重入是函数的概念,线程安全则是线程的概念

线程不安全的情况:

  1. 不保护共享变量的函数(全局变量)
  2. 函数状态随着被调用,状态发生变化的函数(静态变量在函数中,每当被调用就会发生改变)
  3. 返回指向静态变量指针的函数
  4. 调用线程不安全函数的函数

线程安全的情况:

  1. 每个线程对全局变量或者静态变量只有读取的权限,而没有写入的权限,一般来说这些线程是安全的(可以理解成这个变量即使线程修改了,当线程结束后该值又会变回来相当于没变)
  2. 类或者接口对于线程来说都是原子操作
  3. 多个线程之间的切换不会导致该接口的执行结果存在二义性

常见不可重入的情况:

  1. 调用了malloc/free函数,因为malloc函数是用全局链表来管理堆的
  2. 调用了标准I/O库函数,标准I/O库的很多实现都以不可重入的方式使用全局数据结构
  3. 可重入函数体内使用了静态的数据结构

常见可重入的情况

  1. 不使用全局变量或静态变量
  2. 不使用用malloc或者new开辟出的空间
  3. 不调用不可重入函数
  4. 不返回静态或全局数据,所有数据都有函数的调用者提供
  5. 使用本地数据,或者通过制作全局数据的本地拷贝来保护全局数据

可重入与线程安全联系

  1. 可重入函数一定是线程安全的。
  2. 函数是不可重入的,那就不能由多个线程使用,有可能引发线程安全问题
  3. 如果一个函数中有全局变量,那么这个函数既不是线程安全也不是可重入的。

可重入与线程安全区别

  1. 可重入函数是线程安全函数的一种
  2. 线程安全不一定是可重入的,而可重入函数则一定是线程安全的。
  3. 如果将对临界资源的访问加上锁,则这个函数是线程安全的,但如果这个重入函数若锁还未释放则会产生死锁,因此是不可重入的。

1.6 死锁问题

原理:
死锁是指在一组进程中的各个进程均占有不会释放的资源,但因互相申请被其他进程所站用不会释放的资源而处于的一种永久等待状态。
可以理解成,现在用了两把锁A/B各用了一把锁,但两个线程必须同时持有两把锁才能继续往后运行(如下图

一个线程也可能死锁:在程序内部重复申请已经申请过的锁(此时申请不到就会被永久的挂起了)

1.6.1死锁产生的必要条件:

  1. 互斥条件:一个资源每次只能被一个执行流使用
  2. 请求与保持条件:一个执行流因请求资源而阻塞时,对已获得的资源保持不放(相当于上面两个线程的情况)
  3. 不剥夺条件:一个执行流已获得的资源,在末使用完之前,不能强行剥夺(没有因为一个线程优先级高而强行使用所要的锁)
  4. 循环等待条件:若干执行流之间形成一种头尾相接的循环等待资源的关系(可见上图)

1.6.2避免死锁:

  1. 不加锁(破坏互斥条件,但有些时候难以实现)
  2. 若申请某个锁不成功后,释放自身的锁已有的锁(那么就破坏了请求与保持条件)
  3. 当申请某个锁时发现他是被占用的时,直接把他解锁再加锁到当前自己线程(破坏不剥夺条件)
  4. 尽量的把锁资源按顺序申请给线程 !(破坏循环等待条件)
  5. 避免锁未释放的场景以及资源一次分配(一个资源配一把锁)

避免死锁的算法:死锁检测算法,银行家算法。

2.线程的同步

背景:

当一个线程短时间的不断的申请锁,释放锁,导致其他人长时间得不到资源,也就对其他线程产生饥饿问题,为了解决饥饿问题,归还资源后线程不能立即再次申请,再通过类似队列的结构(先进先出,也就是有顺序性)管理要申请锁资源的线程

对此解决这个问题的方法就是同步:

同步:在临界资源使用安全的前提下,让多线程执行具有一定的顺序性。互斥能保证资源的安全,同步能够较为充分高效的使用资源。

2.1 同步 + 互斥 -> CP生产者消费者模型

(计算机领域非常重要的模型)

日常生活中超市就是典型的生产者消费者模型就有生产者、消费者(如下图):

生产者、消费者的三大关系:

  1. 生产者生产者的关系:互斥(供应商相互竞争)
  2. 消费者消费者的关系:互斥(假如只有一份商品了而都想要这个商品那么就是竞争关系)
  3. 生产者消费者的关系:互斥(生产者放完了商品消费者才能拿商品)、同步(不能让一方持续的处理商品(不断地买/卖)都是有问题的)

生产者消费者模型本质就是处理好上方的三个关系。

321原则:

3.三种关系
2.两种角色(生产线程/消费进程)
1.一个交易场所(内存空间)

2.2.1 生产者消费者模型的优势:

  1. 让多执行流之间的一个执行解耦(把生产者和消费者两个线程,双方在内存空间内写或者拿数据,相当于把内存空间当成一个缓冲区可以存一部分数据,所以生产消费不需要相互等待)
  2. 提高处理数据的效率(通过代码解释)

2.2 条件变量

同步的实现就是通过:条件变量(互斥中有锁一样)

条件变量是类似于铃铛提醒人的工具,这个条件变量是为了避免消费者(线程)在资源还没准备好就不断的去内存申请但也拿不到资源的情况(本质就是同步的工具),因为共享空间是互斥的,这样就会导致生产者也无法放数据(饥饿)。

条件变量:当生产者将资源准备好去提醒消费者

  • 其中消费者并不是在锁上等待资源,而是在条件变量处的一个队列(一个阻塞队列
  • 先在条件变量处的队列中排队,当资源就绪条件变量就会唤醒消费线程
  • 当拿取后若想再拿就必须从后开始排(队列)。

也就相当于条件变量的结构为:

struct cond
{
    //条件是否就绪
    int flag;
    //维护一个线程队列
    tcb_queue;
}

当flag表示就绪就从线程队列中唤醒一个线程。

2.3 条件变量的函数(了解)

  1. 条件变量的定义:
    1. 头文件:#include
    2. 成功返回0,错误返回错误码
    3. 条件变量的使用必须要声明和摧毁
对局部条件变量摧毁:
int pthread_cond_destroy(pthread_cond_t *cond);

对局部条件变量进行初始化:
int pthread_cond_init(pthread_cond_t *restrict cond,
              const pthread_condattr_t *restrict attr);
              
全局的条件变量,和锁一样可以直接使用
pthread_cond_t cond = PTHREAD_COND_INITIALIZER;
  1. 线程等待(等待响铃,或者本质就是申请资源
    1. 头文件:#include
    2. 成功返回0,错误返回错误码
在指定的条件变量cond处等待,并且还要传递一把锁mutex
int pthread_cond_wait(pthread_cond_t *restrict cond,
              pthread_mutex_t *restrict mutex);
  1. 唤醒线程(条件变量成立并):
    1. 头文件:#include
    2. 成功返回0,错误返回错误码
唤醒所有线程:
int pthread_cond_broadcast(pthread_cond_t *cond);
唤醒一个线程:     
int pthread_cond_signal(pthread_cond_t *cond);

2.3.1 条件变量的基本使用:

#include 
#include
#include
using namespace std;

pthread_cond_t cond = PTHREAD_COND_INITIALIZER;
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

void* threadRontine(void* args)
{
    string name = static_cast<const char*>(args);

    while(true)
    {
        // sleep(1);
        pthread_mutex_lock(&mutex);

        pthread_cond_wait(&cond,&mutex);//等待
        cout << "I am a new thread:" <<  name <<endl;
        
        pthread_mutex_unlock(&mutex);
    }
}

//主线程
int main()
{
    pthread_t t1,t2,t3;

    pthread_create(&t1,nullptr,threadRontine,(void*)"thread-1"); 
    pthread_create(&t2,nullptr,threadRontine,(void*)"thread-2"); 
    pthread_create(&t3,nullptr,threadRontine,(void*)"thread-3"); 

    sleep(5);//5s后让条件变量唤醒线程
    while(true)
    {
        pthread_cond_signal(&cond);//唤醒一个线程
        // pthread_cond_broadcast(&cond);//唤醒全部线程
        sleep(1);
    }

    pthread_join(t1,nullptr);
    pthread_join(t2,nullptr);
    pthread_join(t3,nullptr);

}

当只唤醒一个线程(pthread_cond_signal)时他是逐个的:
当唤醒全部线程(pthread_cond_broadcast)时他是所有线程一起的:

从上面的饥饿问题不能发现:单纯的互斥,能保证数据的安全,但不一定合理或高效

pthread_cond_wait函数的细节:

  1. 当让线程在进行等待的时候,要自动释放申请的锁。
  2. 线程被在临界区内唤醒的时候,要重新申请并持有锁。
  3. 多个线程被唤醒的时候,它们都要重新申请并持有锁,所以是要竞争锁的
  4. 调用该函数可能失败,这样就会让线程在不满足使用资源条件的前提下(队列中的资源不够多个线程分配)唤醒生产/消费线程,也称为:伪唤醒也就是可能同时唤醒多个,但只有一个线程能拿到锁资源,其他没拿到锁资源的线程就是伪唤醒状态。(对此解决方法是将if语句换成while语句这样就能)

2.4 实现:阻塞队列(及条件变量的应用)

下面将使用到:

  1. 互斥锁
  2. 条件变量
  3. 锁的封装
  4. 将任务对象化
  5. 阻塞队列(让线程实现顺序性)
  6. 模拟生产者消费者模型的运行!

原理:

生产者将一个任务push到队列中,而消费者再去通过pop得到数据并处理。

阻塞队列:

//BlockQueue.hpp
#pragma once
#include
#include
#include
#include
#include
#include"LockGuard.hpp"

using namespace std;

const int defaultcap = 5;//

template<class T>
class Blockqueue
{
public:
    Blockqueue(int cap = defaultcap):_capacity(cap)
    {
        pthread_mutex_init(&_mutex,nullptr);
        pthread_cond_init(&_c_cond,nullptr);
        pthread_cond_init(&_p_cond,nullptr);
    }

    bool IsFull()
    {
        return _q.size() == _capacity;
    }

    bool IsEmpty()
    {
        return _q.size() == 0;
    }
//生产者
    void Push(const T &in)
    {
        LockGuard lockgaurd(&_mutex);
        // pthread_mutex_lock(&_mutex);
        if(IsFull())
        {
            // 生产线程,阻塞等待
            pthread_cond_wait(&_p_cond,&_mutex);
        }
        _q.push(in);
       
        pthread_cond_signal(&_c_cond); //放到里面被唤醒了会在锁处等待了,而非cond处,只要释放锁后就能立刻拿到锁
        // if(_q.size() > _productor_water_line) pthread_cond_signal(&_c_cond);
        // pthread_mutex_unlock(&_mutex);
    }

    void Pop(T *out)
    {
        LockGuard lockgaurd(&_mutex);
        // pthread_mutex_lock(&_mutex);
        if(IsEmpty())   
        {
            //阻塞等待
            pthread_cond_wait(&_c_cond,&_mutex);
        }
        *out = _q.front();

         pthread_cond_signal(&_p_cond);
        //(_q.size() > _consumer_water_line) pthread_cond_signal(&_p_cond);
        _q.pop();

        // pthread_mutex_unlock(&_mutex);
    }
    ~Blockqueue()
    {
        pthread_mutex_destroy(&_mutex);
        pthread_cond_destroy(&_c_cond);
        pthread_cond_destroy(&_p_cond);
    }

private:
    queue<T> _q;
    size_t _capacity;//q.size == capacity满

    pthread_mutex_t _mutex;
    pthread_cond_t _p_cond;//给生产者的
    pthread_cond_t _c_cond;//消费者

    // int _consumer_water_line;// capacity / 3 * 2
    // int _productor_water_line;//capacity / 3
};

封装锁:

//LockGuard.hpp
#pragma once
#include 

class Mutex
{
public:
    Mutex(pthread_mutex_t *lock) : _lock(lock)
    {}
    
    void Lock()

    {
        pthread_mutex_lock(_lock);
    }
    void Unlock()

    {
        pthread_mutex_unlock(_lock);
    }

    ~Mutex()
    {}

private:
    pthread_mutex_t *_lock; // 不定义锁,默认外面会创建传进来
};

class LockGuard
{
public:
     LockGuard(pthread_mutex_t *lock) : _mutex(lock)

    {
        _mutex.Lock();
        
    }
    ~LockGuard()
    {
         _mutex.Unlock();
        
    }

private:
    Mutex _mutex;
};

任务对象:

//Task.hpp
#pragma once
#include
enum 
{
    ok = 0,
    div_zero,
    mod_zero,
    unknow
};


class Task
{
public:
    Task()
    {}

    Task(int x, int y ,char op)
    :_x(x),_y(y),_oper(op)
    {}

    void Run()
    {
        switch (_oper)
        {
            case '+':
                result = _x + _y;
                break;
            case '-':
                result = _x - _y;
                break;
            case '*':
                result = _x * _y;

                break;
            case '/':
            {
                if(_y == 0) {
                    code = div_zero; 
                    break;
                }
                result = _x / _y;
            }
            break;
            case '%':
            {
                if(_y == 0)  {
                    code = mod_zero; 
                    break;
                }
                result = _x % _y;
            }
            break;
            default:
                code = unknow;
                break;
        }
    }

    string PrintTask()
    {
        string s;
        s += to_string(_x);
        s += _oper;
        s += to_string(_y);
        s += "=?";

        return s;
    }
    void operator()()
    {
        Run();
    }
    string PrintResult()
    {
        string s;

        s += to_string(_x);
        s += _oper;
        s += to_string(_y);
        s += " =";
        s += to_string(result);
        s+= " [";
        s+= to_string(code);
        s+= "]";
       
        return s;
    }

    ~Task()
    {}

private:
    int _x;
    int _y;
    char _oper;

    int result;
    int code;//任务退出码,0结果可信,!0结果不可信
};

主程序

//main.cc
#pragma once
#include
#include
#include
#include
#include
#include"LockGuard.hpp"

using namespace std;

const int defaultcap = 5;//

template<class T>
class Blockqueue
{
public:
    Blockqueue(int cap = defaultcap):_capacity(cap)
    {
        pthread_mutex_init(&_mutex,nullptr);
        pthread_cond_init(&_c_cond,nullptr);
        pthread_cond_init(&_p_cond,nullptr);
    }

    bool IsFull()
    {
        return _q.size() == _capacity;
    }

    bool IsEmpty()
    {
        return _q.size() == 0;
    }
//生产者
    void Push(const T &in)
    {
        LockGuard lockgaurd(&_mutex);
        // pthread_mutex_lock(&_mutex);
        //if(IsFull())
        while(IsFull())//把if改成while这样即使返回来了,也要判断数据是否能再放数据 
        {
            // 生产线程,阻塞等待
            pthread_cond_wait(&_p_cond,&_mutex);
        }
        _q.push(in);
       
        pthread_cond_signal(&_c_cond); //放到里面被唤醒了会在锁处等待了,而非cond处,只要释放锁后就能立刻拿到锁
        // if(_q.size() > _productor_water_line) pthread_cond_signal(&_c_cond);
        // pthread_mutex_unlock(&_mutex);
    }

    void Pop(T *out)
    {
        LockGuard lockgaurd(&_mutex);
        // pthread_mutex_lock(&_mutex);
        //if(IsEmpty())   
        while(IsEmpty())//把if改成while这样即使返回来了,也要判断是否有数据能使用 
        {
            //阻塞等待
            pthread_cond_wait(&_c_cond,&_mutex);
        }
        *out = _q.front();

         pthread_cond_signal(&_p_cond);
        //(_q.size() > _consumer_water_line) pthread_cond_signal(&_p_cond);
        _q.pop();

        // pthread_mutex_unlock(&_mutex);
    }
    ~Blockqueue()
    {
        pthread_mutex_destroy(&_mutex);
        pthread_cond_destroy(&_c_cond);
        pthread_cond_destroy(&_p_cond);
    }

private:
    queue<T> _q;
    size_t _capacity;//q.size == capacity满

    pthread_mutex_t _mutex;
    pthread_cond_t _p_cond;//给生产者的
    pthread_cond_t _c_cond;//消费者

    // int _consumer_water_line;// capacity / 3 * 2
    // int _productor_water_line;//capacity / 3
};

对此为什么生产者消费者模型能提高数据处理的效率?


因为对于生产者消费者模型来说,我们不能只看内部他的生产和消费过程,这里是互斥的并没有效率的提升,但是从整体来看生产者线程获取数据的过程和消费者线程处理数据的过程也是需要时间的,他们在处理这些时间时,其他线程就能同步的去执行生产/消费过程,从而实现每个线程都能高效的执行其作用。

2.4 信号量

该方法同样也是实现同步的工具(只不过常用条件变量,所以这里就简略了)

在之前文章中已经写过信号量的基本概念有:

  1. 信号量的本质是一把计数器
  2. 申请信号的本质就是预定资源
  3. PV操作是原子的!

把公共资源不当做整体,多线程不访问临界资源的同一个区域。
对此信号量为了防止分成的n份公共资源,分给了n+k个线程,
信号量的作用就是:确定线程能否访问资源
线程信号量申请成功后,当线程需要使用时就不需要再判断资源是否就绪,直接就能使用了(申请信号量时已经判断了)

2.4.1信号量的基本函数

  1. 信号量初始化与销毁
    1. 头文件:#include
初始化信号量
  int sem_init(sem_t *sem, int pshared, unsigned int value);
  1. sem:返回定义的信号量(输出型)
  2. pshared:在线程间共享(设置为0),还是进程间
  3. value:信号量初始的值
销毁信号量
  int sem_destroy(sem_t *sem);

信号量的PV操作

  1. 申请信号量,P操作–:
	int sem_wait(sem_t *sem);

申请成功继续,失败则阻塞等待

  1. 释放信号量,V操作++:
	int sem_post(sem_t *sem);

上述函数的返回值都是:成功为0,失败为非0错误码


本章完。预知后事如何,暂听下回分解。

如果有任何问题欢迎讨论哈!

如果觉得这篇文章对你有所帮助的话点点赞吧!

持续更新大量C++细致内容,早关注不迷路。

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

搜索文章

Tags

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