最新资讯

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

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

2025-04-27 22:00:48 0 阅读


每日激励:“不设限和自我肯定的心态: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/3916.html

搜索文章

Tags

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