Linux

关注公众号 jb51net

关闭
首页 > 网站技巧 > 服务器 > Linux > Linux线程安全与线程同步

Linux中的线程安全与线程同步详解

作者:s_little_monster_

这篇文章主要介绍了Linux中的线程安全与线程同步,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

一、线程安全

1、概念

我们这里通过理解重入与线程安全的关系来理解线程安全

线程安全多个线程并发同一段代码时,不会出现不同的结果 

重入同一个函数被不同的执行流调用,当前一个流程还没有执行完,就有其他的执行流再次进入,一个函数在重入的情况下运行结果不会出现任何问题,这样的函数称为可重入函数,否则,就是不可重入函数

2、常见线程情况

常见线程不安全情况

常见线程安全情况

3、常见重入情况

常见不可重入情况

常见可重入情况

4、可重入与线程安全

联系

区别

5、死锁

(一)概念

死锁是指在一组进程或线程中的各个进程或线程均占有不会释放的资源,但因互相申请被其他进程或线程所占用的不会释放的资源而处于的一种永久等待的状态

死锁都是人为产生的,我们可以规避掉的

(二)死锁的四个必要条件

(三)避免死锁的方法

二、线程同步

1、概念

在纯互斥的场景下,由于我们的锁只有少量个,多个线程同时竞争锁,但是得到锁的只有一小部分线程,剩下的线程就会因为等待,产生 “线程饥饿” 问题,线程饥饿本质上就是抢夺不到锁的线程,即抢夺不到资源的线程在等待锁的释放,为了避免这里的饥饿的问题,我们就通过线程同步来在保证数据安全的前提下,让线程按照顺序访问临界资源

2、条件变量

(一)概念

当一个线程互斥的访问某个变量时,它可能在其他线程改变状态之前什么也做不了,比如一个线程访问队列时,发现队列为空,那么它只能等待,直到其他进程将一个节点添加到队列当中,这个时候我们就可以利用条件变量来规避这种情况

(二)调用函数

(1)初始化条件变量
#include <pthread.h>
int pthread_cond_init(pthread_cond_t *restrict cond, const pthread_condattr_t *restrict attr);
(2)销毁条件变量
#include <pthread.h>
int pthread_cond_destroy(pthread_cond_t *cond);
(3)等待条件被满足
#include <pthread.h>
int pthread_cond_wait(pthread_cond_t *restrict cond, pthread_mutex_t *restrict mutex);

调用该函数时,线程会自动释放互斥锁mutex,以便其他线程可以获取锁,当收到信号被唤醒后,线程会重新尝试获取互斥锁

(4)唤醒等待线程
#include <pthread.h>
//唤醒一个等待线程
int pthread_cond_signal(pthread_cond_t *cond);
//唤起所有等待线程
int pthread_cond_broadcast(pthread_cond_t *cond);

如果一个线程执行 pthread_cond_broadcast,它会将所有等待该条件变量的线程全部唤醒,若执行 pthread_cond_signal,则只会唤醒至少一个等待该条件变量的线程,而非只唤醒当前线程

(三)样例

#include <iostream>
#include <pthread.h>
#include <vector>
#include <unistd.h>

using namespace std;

#define NUM 4

int cnt = 0;
//条件变量函数的用法几乎与锁函数的用法完全等同
//定义全局锁和全局条件变量
pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;
pthread_cond_t cond = PTHREAD_COND_INITIALIZER;

void *Count(void *args)
{
    pthread_detach(pthread_self()); // 线程分离,跑完就不管了,不在乎它的返回值
	// Linux是64位机,指针是8字节,uint是unsigned long long int   
    uint64_t num = (uint64_t)args;
    cout << "Thread " << num << " is creat success" << endl;
    usleep(100000);
    while (true)
    {
        pthread_mutex_lock(&lock);
        
        //这里pthread_cond_wait要在临界区的原因是:
        //因为 pthread_cond_wait 是让线程去等待,等待的原因一定是临界资源不就绪
        //而临界资源是否就绪,是通过判断得来的,判断也是访问临界资源,所以判断必须在加锁之后
        pthread_cond_wait(&cond, &lock); 
        //线程在此处进入等待状态,等待条件变量 cond 发出信号
        
        cout << "Thread " << num << " is running... cnt: " << cnt << endl;
        cnt++;
        usleep(10000);
        pthread_mutex_unlock(&lock);
    }
}

int main()
{
    for (uint64_t i = 1; i <= NUM; i++)
    {
        pthread_t tid;
        //这里的第四个参数,如果想要与新线程共享这个参数的话,可以设为(void*)&i,进行传址调用
        //我们这里要传值调用,不能让它用i
        pthread_create(&tid, nullptr, Count, (void *)i);
        usleep(1000);
    }
	//指定唤醒线程来访问临界资源
    while (true)
    {
        sleep(1);
        pthread_cond_signal(&cond); // 唤醒一个线程
        cout << "signal one thread..." << endl;
    }
    return 0;
}

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

您可能感兴趣的文章:
阅读全文