【Linux】线程同步与互斥 - 2(线程同步/条件变量/基于阻塞/环形队列的cp模型/线程池/线程安全/读写锁)

同步的概念

同步是保证数据安全的情况下(互斥的前提下),让线程访问资源具有一定的顺序性。比如多个线程访问同一个资源,如果该线程申请锁失败,那么它只能在该资源的申请队列的末尾排队等待。既然线程都在排队等待了,那为什么还需要锁呢?这是因为线程不是乖乖的在队列的末尾排队等待,而是直接去申请锁,失败了再排队。

同步与互斥的关系

互斥可以用互斥锁实现,但是互斥也有互斥的问题,比如调度不均衡,竞争不均衡。避免这些问题可以采用一些策略,比如同步。即:**互斥是同步的基础,同步是互斥的扩展。**互斥可以看作是最简单的同步

在纯互斥的场景下,哪个线程能申请到锁是不可预见的,而在互斥 + 同步的场景下,线程是按照一定顺序申请锁,执行临界区代码的线程的顺序是可以预见的。

复制代码
并发控制
    ├── 互斥 (Mutual Exclusion)
    │   ├── 解决:同时访问的问题
    │   ├── 工具:互斥锁、读写锁、信号量
    │   └── 结果:数据一致性
    │
    └── 同步 (Synchronization)
        ├── 解决:先后顺序的问题
        ├── 工具:条件变量、信号量、屏障
        └── 结果:执行顺序可控

条件变量

当一个线程在访问完共享资源时,在它释放锁之后,要敲一下"铃铛",目的是让下一个准备访问该共享资源的线程知道可以访问了。这个"铃铛"和等待队列,就叫做"条件变量"

条件变量的相关函数

条件变量初始化/销毁 - pthread_cond_init/destroy

cpp 复制代码
int pthread_cond_init(pthread_cond_t *restrict cond,const pthread_condattr_t *restrict
attr);
参数:
cond:要初始化的条件变量
attr:NULL

与互斥锁相同,静态全局的条件变量不需要使用 pthread_cond_init/destroy 来初始化/销毁
pthread_cond_t cond = PTHREAD_COND_INITIALIZER;

int pthread_cond_destroy(pthread_cond_t *cond)

线程加入等待队列 - pthread_cond_wait

cpp 复制代码
int pthread_cond_wait(pthread_cond_t *restrict cond,pthread_mutex_t *restrict mutex);
参数:
cond:要在这个条件变量上等待
mutex:互斥锁

线程被唤醒后,会从 pthread_cond_wait 函数中返回,并继续执行后续的代码

注意:

在使用时,我们通常是这样使用的:

cpp 复制代码
// ... 
pthread_mutex_lock(&mutex); // 先申请锁

// 在锁的保护下

while(资源未就绪) pthread_cond_wait(&cond,&mutex); // 让线程去等待队列等待

// 条件满足、被唤醒
{
    // ... 临界区
}

pthread_mutex_unlock(&mutex);
// ... 

为什么 pthread_cond_wait 需要传递互斥锁

  • 在 if 判断中,可能也会访问临界资源,所以 while 循环判断必须在 pthread_mutex_lock 之后,如果条件不满足,线程在等待之前又必须释放锁。线程在被唤醒之后,在执行临界区代码之前,又必须持有锁。这是该函数需要传递互斥锁的一方面。
  • 另一方面,假如pthread_cond_wait 不需要传递互斥锁,在 while 循环中 调用pthread_mutex_unlock:
cpp 复制代码
pthread_mutex_lock(&mutex);
while(资源未就绪) {
    pthread_mutex_unlock(&mutex);  // 解锁

    // 问题:解锁和等待之间有时间窗口    
    // 假如此时该线程被切走了
    // 另一个线程拿到锁,修改共享数据,使条件满足,调用 pthread_cond_signal
    // 该线程继续执行
    
    // 调用 pthread_cond_wait
    pthread_cond_wait(&cond);       // 然后等待
    // 问题:signal 已经发送过了,线程A永远收不到信号!
}
  • 那么整个 while 循环判断就不会是原子的,会有线程安全问题。pthread_cond_wait 函数内部用某种方式将 1、把当前线程加入cond 的等待队列 2 、释放mutex 3、标记线程为等待状态 设计为原子的

唤醒等待队列的线程 - pthread_cond_signal/broadcast

假如所有线程都加入等待队列了,如果没有人唤醒,那么它们将一直阻塞等待。通常可以让主线程来控制唤醒线程的逻辑。

cpp 复制代码
int pthread_cond_broadcast(pthread_cond_t *cond);
int pthread_cond_signal(pthread_cond_t *cond);
特性 pthread_cond_signal pthread_cond_broadcast
唤醒数量 唤醒至少一个等待线程 唤醒所有等待线程
具体唤醒谁 由调度策略决定,通常是优先级最高的(对头) 所有线程都被唤醒
适用场景 资源可用,只需一个消费者 状态变化,所有等待者都需处理
竞争激烈程度 低(只有一个线程被唤醒) 高(所有线程被唤醒,争夺锁)

使用实例:让 5 个线程按顺序的对一个全局变量 cnt 进行 ++ 操作。

cpp 复制代码
#include <iostream>
#include <unistd.h>
#include <pthread.h>

int cnt = 0;
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
pthread_cond_t cond = PTHREAD_COND_INITIALIZER;

void *Count(void * args)
{
    pthread_detach(pthread_self());
    uint64_t number = (uint64_t)args;
    std::cout << "pthread: " << number << " create success" << std::endl;

    while(true)
    {
        pthread_mutex_lock(&mutex);
        pthread_cond_wait(&cond, &mutex); //pthread_cond_wait让线程等待的时候会自动释放锁
        std::cout << "pthread: " << number << " , cnt: " << cnt++ << std::endl;
        pthread_mutex_unlock(&mutex);
    }
}

int main()
{
    // uint64_t:unsigned long long,防止等下强制转换为 64 位指针时出现警告
    for(uint64_t i = 0; i < 5; i++)
    {
        pthread_t tid;
        pthread_create(&tid, nullptr, Count, (void*)i);
        usleep(1000); // 保证等待队列的顺序就是创建顺序
    }

    sleep(3); // 确保都加入了等待队列

    std::cout << "main thread ctrl begin: " << std::endl;

    while(true) 
    {
        sleep(1);
        // pthread_cond_signal(&cond); //唤醒在cond的等待队列中等待的一个线程,默认都是第一个
        pthread_cond_broadcast(&cond); //唤醒所有
        std::cout << "signal one thread..." << std::endl;
    }

    return 0;
}

生产者-消费者模型

模型简介

生产者-消费者模型(consumer - producter model,简称 cp 模型)是并发编程中最经典的同步问题,它描述了两类线程(生产者和消费者)如何共享一个固定大小的缓冲区

生产者和消费者的核心约束:

  1. 缓冲区满:生产者不能向满的缓冲区放入数据

  2. 缓冲区空:消费者不能从空的缓冲区取出数据

  3. 互斥访问:同一时刻只能有一个线程操作缓冲区

缓冲区存在的意义(cp 模型的优点):

1、支持生产和消费的速率不同(生产者不用关心消费者的消费速率,只需关心缓冲区是否有空间,消费者也不关心生产者的生产速率,只关心缓冲区是否有数据)

2、实现生产和消费的解耦(生产者生产完数据之后不用等待消费者处理,直接把数据放入缓冲区,消费者直接找生产者要数据,而是从缓冲区里取)

生产者和消费者的同步与互斥:

因为缓冲区其实是共享资源,多个生产者和消费者同时访问缓冲区必然存在并发问题。所以生产者和生产者之间是互斥关系,消费者与消费者之间是互斥关系 。生产者不能一直向缓冲区生产,导致消费者饥饿,所以生产者和消费者之间是同步关系

生产者-消费者模型为什么是高效的

  • 生产者生产的数据一定是从外部获取的(比如网络、用户、其他线程),而消费者在缓冲区拿到数据之后要对数据做加工处理。生产者-消费者模型之所以是高效的是因为:生产者从外部获取数据时,消费者可能正在从缓冲区拿数据并作加工处理;消费者在作加工处理时,生产者可能正在从从外部获取数据并把数据放入缓冲区。
  • 既然在某一时刻只能由一个生产者生产数据或一个消费者消费数据,那为什么还要多生产者和多消费者呢?该模型高效的地方不在于对缓冲区拿放的过程,而在于:在某一时刻可以有多个生产者同时从外部获取数据,也可以有多个消费者在消费数据

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

生产者-消费者模型中,缓冲区可以是特定的数据结构,常见的是阻塞队列(Blocking Queue),其与普通的队列区别在于,当队列为空时,从队列获取元素的操作将会被阻塞,直到队列中被放入了元素;当队列满时,往队列里存放元素的操作也会被阻塞,直到有元素被从队列中取出。下面采用C++ queue模拟阻塞队列的生产消费模型。

BlockQueue.hpp 阻塞队列的实现

cpp 复制代码
#pragma once

#include <iostream>
#include <queue>
#include <pthread.h>

template <class T>
class BlockQueue
{
    static const int defalutnum = 20;
public:
    BlockQueue(int maxcap = defalutnum) :maxcap_(maxcap)
    {
        pthread_mutex_init(&mutex_, nullptr);
        pthread_cond_init(&c_cond_, nullptr);
        pthread_cond_init(&p_cond_, nullptr);
        // low_water_ = maxcap_/3;
        // high_water_ = (maxcap_*2)/3;
    }

    // 谁来唤醒呢?
    T pop()
    {
        pthread_mutex_lock(&mutex_);
        while (q_.size() == 0) 
        {
            // 如果线程wait时,被误唤醒了呢??
            pthread_cond_wait(&c_cond_, &mutex_); 
        }

        T out = q_.front(); 
        q_.pop();

        // if(q_.size()<low_water_) pthread_cond_signal(&p_cond_);
        pthread_cond_signal(&p_cond_); // pthread_cond_broadcast
        pthread_mutex_unlock(&mutex_);

        return out;
    }

    void push(const T& in)
    {
        pthread_mutex_lock(&mutex_);
        while (q_.size() == maxcap_) { // 做到防止线程被伪唤醒的情况
            // 伪唤醒情况
            pthread_cond_wait(&p_cond_, &mutex_); //1. 调用的时候,自动释放锁 2.?
        }
        // 1. 队列没满 2.被唤醒 
        q_.push(in);                    
        // if(q_.size() > high_water_) pthread_cond_signal(&c_cond_);
        pthread_cond_signal(&c_cond_);
        pthread_mutex_unlock(&mutex_);
    }

    ~BlockQueue()
    {
        pthread_mutex_destroy(&mutex_);
        pthread_cond_destroy(&c_cond_);
        pthread_cond_destroy(&p_cond_);
    }
private:
    std::queue<T> q_; 
    //int mincap_;
    int maxcap_;      // 极值
    pthread_mutex_t mutex_;
    pthread_cond_t c_cond_;
    pthread_cond_t p_cond_;

    // int low_water_;
    // int high_water_;
};
  • 在多生产者和多消费者的情况下,可能出现伪唤醒的情况

情景再现:假设现在阻塞队列以及满了,消费者在消费一个数据之后,阻塞队列只有一个空位,但是由于某种原因,消费者唤醒了多个生产者,多个生产者开始同时竞争申请锁,一个生产者竞争成功并向阻塞队列放入数据并释放锁之后,另一个生产者不小心又申请到了锁,又向阻塞队列放入数据,这种情况就叫做伪唤醒。解决方法就是判断阻塞队列是否有空位或为空采用 while 循环判断。


生产者生产的数据和消费者消费的数据用一个类来模拟,一个类对象代表一个简单的运算任务。生产者随机生产数据,模拟从外部获取的不同数据。

tesk.hpp:

cpp 复制代码
#pragma once
#include <iostream>
#include <string>

std::string opers = "+-*/%";

// 错误码
enum {
    DivZero = 1,
    ModZero,
    Unknown
};

class Task
{
public:
    Task(int x, int y, char op) : data1_(x), data2_(y), oper_(op), result_(0), exitcode_(0)
    {
    }
    void run()
    {
        switch (oper_)
        {
        case '+':
            result_ = data1_ + data2_;
            break;
        case '-':
            result_ = data1_ - data2_;
            break;
        case '*':
            result_ = data1_ * data2_;
            break;
        case '/':
        {
            if (data2_ == 0) exitcode_ = DivZero;
            else result_ = data1_ / data2_;
        }
        break;
        case '%':
        {
            if (data2_ == 0) exitcode_ = ModZero;
            else result_ = data1_ % data2_;
        }            break;
        default:
            exitcode_ = Unknown;
            break;
        }
    }
    // 方便消费者调用任务:t() <-> t.run()
    void operator ()()
    {
        run();
    }

    // 获取任务结果
    std::string GetResult()
    {
        std::string r = std::to_string(data1_);
        r += oper_;
        r += std::to_string(data2_);
        r += "=";
        r += std::to_string(result_);
        r += "[code: ";
        r += std::to_string(exitcode_);
        r += "]";

        return r;
    }

    // 打印任务内容
    std::string GetTask()
    {
        std::string r = std::to_string(data1_);
        r += oper_;
        r += std::to_string(data2_);
        r += "=?";
        return r;
    }
    ~Task()
    {
    }

private:
    int data1_;
    int data2_;
    char oper_;

    int result_;
    int exitcode_;
};

main 函数(主线程)用来创建多生产者和多消费者,让生产者能够生产数据,消费者能够消费数据

main.cpp:

cpp 复制代码
#define _CRT_SECURE_NO_WARNINGS 1
#include "BlockQueue.hpp"
#include "Task.hpp"
#include <unistd.h>
#include <ctime>

void* Consumer(void* args)
{
    BlockQueue<Task>* bq = static_cast<BlockQueue<Task> *>(args);

    while (true)
    {
        // 消费
        Task t = bq->pop();

        // 计算
        // t.run();
        t();

        std::cout << "处理任务: " << t.GetTask() << " 运算结果是: " << t.GetResult() << " thread id: " << pthread_self() << std::endl;
        // t.run();
        // sleep(1);
    }
}

void* Productor(void* args)
{
    int len = opers.size();
    BlockQueue<Task>* bq = static_cast<BlockQueue<Task> *>(args);
    
    while (true)
    {
        // 模拟生产者生产数据
        int data1 = rand() % 10 + 1; // [1,10]
        usleep(10);
        int data2 = rand() % 10;
        char op = opers[rand() % len];
        Task t(data1, data2, op);

        // 生产
        bq->push(t);
        std::cout << "生产了一个任务: " << t.GetTask() << " thread id: " << pthread_self() << std::endl;
        sleep(1);
    }
}

int main()
{
    srand(time(nullptr));

    BlockQueue<Task>* bq = new BlockQueue<Task>();
    pthread_t c[3], p[5];
    for (int i = 0; i < 3; i++)
    {
        pthread_create(c + i, nullptr, Consumer, bq);
    }

    for (int i = 0; i < 5; i++)
    {
        pthread_create(p + i, nullptr, Productor, bq);
    }

    for (int i = 0; i < 3; i++)
    {
        pthread_join(c[i], nullptr);
    }
    for (int i = 0; i < 5; i++)
    {
        pthread_join(p[i], nullptr);
    }
    delete bq;
    return 0;
}

信号量

  • 上面基于BlockingQueue的生产者消费者模型中,将阻塞队列看成一个整体,在某时刻只允许最多 1 个生产者或消费者访问阻塞队列。现在将阻塞队列看成多份,每个生产者/消费者都只能访问该阻塞队列的特定部分。
  • 使用信号量对阻塞队列的各部分做管理,记录阻塞队列的空闲部分数量。信号量会保证 PV 操作是原子的。在 P 操作和 V 操作之间,不需要像条件变量那样对资源是否就绪做判断,而是直接使用资源,因为 P 操作本身就在做判断:如果仍有空闲部分,往下执行,否则在信号量的等待队列等待。

相关函数

初始化/销毁信号量 - sem_init/destroy

cpp 复制代码
#include <semaphore.h>
int sem_init(sem_t *sem, int pshared, unsigned int value);
参数:
sem_t *sem:定义的信号量的地址
pshared:0表示线程间共享,非零表示进程间共享,默认为 0 即可
value:信号量初始值

int sem_destroy(sem_t *sem);

等待/发布信号量 sem_wait/post

cpp 复制代码
功能:等待信号量,会将信号量的值减1
int sem_wait(sem_t *sem); //P()

功能:发布信号量,表示资源使用完毕,可以归还资源了。将信号量值加1。
int sem_post(sem_t *sem);//V()

基于环形队列的生产消费模型

生产者-消费者模型中,缓冲区可以是特定的数据结构,现在采用环形队列作为缓冲区。环形队列如何判空和判满呢?有两种方法:

1、用一个计数器记录空闲位置。

2、牺牲一个位置,如果 head == tail 表示空,如果 (head + 1)%capacity == tail ,表示已满。

现采用第三种方法:使用两个信号量,一个信号量表示环形队列还有有多少空间资源(生产者的信号量,初始为环形队列的总容量),另一个信号量表示环形队列还有有多少数据资源(消费者的信号量,初始为0)。

  • 开始的时候,只有生产者可以执行 P 操作,生产者执行 V 操作之后,消费者才能执行 P 操作。
  • 用两个变量分别记录生产者和消费者的当前位置。
  • 由于生产者和消费者对应的下标只有一个,如果要实现多生产者和多消费者,情况有点复杂,所以先实现单生产者单消费者。如果要实现多生产者和多消费者,由于环形队列可以支持同时有一个生产者生产数据和一个消费者消费数据,所以生产者和消费者应该分别使用两把锁(与阻塞队列的 cp 模型不同,阻塞队列的生产者和消费者共用一把锁,即某时刻只能有一个生产者或消费者在操作阻塞队列)
  • 应该先申请信号量,再申请锁。先申请信号量是资源的预定,对资源的使用是申请锁成功之后。从技术角度讲,信号量内部的实现是原子的,不需要锁保护。从逻辑角度讲,这样做可以支持线程在使用资源的同时,其他线程可以申请信号量提前预定,提高并发度。

RIngQueue.hpp:

cpp 复制代码
#pragma once
#include <iostream>
#include <vector>
#include <semaphore.h>
#include <pthread.h>

const static int defaultcap = 5;

template<class T>
class RingQueue{
private:
    void P(sem_t &sem) // 一定要加引用!!!
    {
        sem_wait(&sem);
    }
    void V(sem_t &sem)// 一定要加引用!!!
    {
        sem_post(&sem);
    }
    void Lock(pthread_mutex_t &mutex) // 一定要加引用!!!
    {
        pthread_mutex_lock(&mutex);
    }
    void Unlock(pthread_mutex_t &mutex) // 一定要加引用!!!
    {
        pthread_mutex_unlock(&mutex);
    }
public:
    RingQueue(int cap = defaultcap)
    :ringqueue_(cap), cap_(cap), c_step_(0), p_step_(0)
    {
        sem_init(&cdata_sem_, 0, 0);
        sem_init(&pspace_sem_, 0, cap);

        pthread_mutex_init(&c_mutex_, nullptr);
        pthread_mutex_init(&p_mutex_, nullptr);
    }
    void Push(const T &in) // 生产
    {
        P(pspace_sem_);

        Lock(p_mutex_); // ?
        ringqueue_[p_step_] = in;
        // 位置后移,维持环形特性
        p_step_++;
        p_step_ %= cap_;
        Unlock(p_mutex_); 

        V(cdata_sem_);

    }
    void Pop(T *out)       // 消费
    {
        P(cdata_sem_);

        Lock(c_mutex_); // ?
        *out = ringqueue_[c_step_];
        // 位置后移,维持环形特性
        c_step_++;
        c_step_ %= cap_;
        Unlock(c_mutex_); 

        V(pspace_sem_);
    }
    ~RingQueue()
    {
        sem_destroy(&cdata_sem_);
        sem_destroy(&pspace_sem_);

        pthread_mutex_destroy(&c_mutex_);
        pthread_mutex_destroy(&p_mutex_);
    }
private:
    std::vector<T> ringqueue_;
    int cap_;

    int c_step_;       // 消费者下标
    int p_step_;       // 生产者下标

    sem_t cdata_sem_;  // 消费者关注的数据资源
    sem_t pspace_sem_; // 生产者关注的空间资源

    pthread_mutex_t c_mutex_;
    pthread_mutex_t p_mutex_;
};

Tesk.hpp、main.cpp 的大致内容与阻塞队列的相同。

线程池

线程池是一种线程使用模式。线程过多会带来调度开销,进而影响缓存局部性和整体性能。而线程池维护着多个线程,等待着监督管理者分配可并发执行的任务。这避免了在处理短时间任务时创建与销毁线程的时间代价。线程池不仅能够保证内核的充分利用,还能防止过分调度。可用线程数量应该取决于可用的并发处理器、处理器内核、内存、网络sockets等的数量。

线程池的应用场景:

  1. 需要大量的线程来完成任务,且完成任务的时间比较短 。 WEB服务器完成网页请求这样的任务,使用线程池技术是非常合适的。因为单个任务小,而任务数量巨大,你可以想象一个热门网站的点击次数。 但对于长时间的任务,比如一个Telnet连接请求,线程池的优点就不明显了。因为Telnet会话时间比线程的创建时间大多了。

  2. 对性能要求苛刻的应用,比如要求服务器迅速响应客户请求

  3. 接受突发性的大量请求,但不至于使服务器因此产生大量线程的应用。突发性大量客户请求,在没有线程池情况下,将产生大量线程,虽然理论上大部分操作系统线程数目最大值不是问题,短时间内产生大量线程可能使内存到达极限,出现错误.

线程池示例:

  1. 在类内创建固定数量线程池,循环从任务队列中获取任务对象。由于线程要执行的函数是在类内创建的,那么它就是类的成员函数,类的成员函数第一个参数默认是 this 指针,而线程要执行的函数必须参数是 void* ,返回值是 void* 的,所以将线程要执行的函数在类内声明为静态的,静态成员函数没有 this 指针 。但是该静态函数会访问类的成员比如任务队列,所以应该给 pthread_create 函数传递 this 指针

  2. 获取到任务对象后,执行任务对象中的任务接口

1、基于环形队列的生产消费模型 done

2、实现线程池

3、functional

4、封装线程

5、在线程池中使用封装的线程

6、为什么线程不 join 就不 run,因为线程不 join 主线程直接就退出了

7、将线程池改为懒汉模式

STL,智能指针和线程安全

STL中的容器是否是线程安全的?

不是.原因是,STL 的设计初衷是将性能挖掘到极致, 而一旦涉及到加锁保证线程安全, 会对性能造成巨大的影响 .而且对于不同的容器, 加锁方式的不同, 性能可能也不同(例如hash表的锁表和锁桶).因此 STL 默认不是线程安全 . 如果需要在多线程环境下使用, 往往需要调用者自行保证线程安全.

智能指针是否是线程安全的?

对于 unique_ptr, 由于只是在当前代码块范围内生效, 因此不涉及线程安全问题.

对于 shared_ptr, 多个对象需要共用一个引用计数变量, 所以会存在线程安全问题. 但是标准库实现的时候考虑到了这个问题, 基于原子操作(CAS)的方式保证 shared_ptr 能够高效, 原子的操作引用计数.

单例模式的线程安全

懒汉方式实现单例模式

cpp 复制代码
template <typename T>
class Singleton {
	static T* inst;
public:
	static T* GetInstance() {
		if (inst == NULL) {
			inst = new T();
		}
		return inst;
	}
};

存在一个严重的问题, 线程不安全.第一次调用 GetInstance 的时候, 如果两个线程同时调用, 可能会创建出两份 T 对象的实例

cpp 复制代码
// 懒汉模式, 线程安全
template <typename T>
class Singleton {
	volatile static T* inst; // 需要设置 volatile 关键字, 否则可能被编译器优化.
	static std::mutex lock;
public:
	static T* GetInstance() {
		if (inst == NULL) { // 双重判定空指针, 降低锁冲突的概率, 提高性能.
			lock.lock(); // 使用互斥锁, 保证多线程情况下也只调用一次 new.
			if (inst == NULL) {
				inst = new T();
			}
			lock.unlock();
		}
		return inst;
	}
};

注意事项:

  1. 双重 if 判定 , 避免不必要的锁竞争:第一次调用 GetInstance:所有线程都判断 inst == NULL 为真,都竞争申请锁。往后调用 GetInstance:所有线程都判断 inst == NULL 为假,直接返回,不用再重复竞争申请锁。

  2. 加锁解锁的位置,

  3. volatile关键字防止过度优化

相关推荐
雨落在了我的手上1 小时前
C语言之数据结构初见篇(2):顺序表之通讯录的实现(续)
c语言·开发语言·数据结构
小生不才yz1 小时前
【Makefile 专家之路 | 基础篇】02. 初试锋芒:编写第一个 Makefile 与运行机制深度剖析
linux
你这个代码我看不懂1 小时前
JVM栈、方法区和堆内存
java·开发语言·jvm
GIS阵地2 小时前
一场由Qt5 painter的drawRect引起的血雨腥风
开发语言·qt·gis·qgis
学编程就要猛2 小时前
JavaEE初阶:多线程案例
java·开发语言
码不停蹄Zzz2 小时前
对内存堆栈管理的简单理解[C语言]
c语言·开发语言
Xu_youyaxianshen2 小时前
[特殊字符] Docker 小白极速入门笔记
linux·docker
getapi2 小时前
FinalShell 连接 CentOS 7 文件管理失败修复教程
linux·运维·centos
程序员学习随笔2 小时前
ext4 原理篇(三):日志子系统 Journal 深度剖析 —— 如何保障数据一致性?
linux·c++