【Linux】多线程的相关知识点

一、线程安全

1.1 可重入 VS 线程安全

1.1.1 概念

  • 线程安全:多个线程并发执行同一段代码时,不会出现不同的结果。常见对全局变量或者静态变量进行操作,并且没有锁的保护的情况下,会出现问题。
  • 重入:同一个函数被不同的执行流调用,当前一个流程还没有执行完,就有其他的执行力再次进入,一个函数在重入的情况下,运行结果不会出现任何不同或者任何问题,则该函称之为可重入函数,否则为不可重入函数。
  • 线程安全是线程在执行中的相互关系,重入是函数的特点
  • 引起线程安全有很多种情况,重入是其中的一种

1.1.2 常见的线程不安全的情况

  • 不保护共享变量的函数
  • 函数状态随着被调用,状态发生变化的函数
  • 返回指向静态变量指针的函数
  • 调用线程不安全函数的函数

1.1.3 对函数状态随着被调用,状态发生变化进行解释

class A
{
public:
    void fun()
    {
        std::cout << "fun" << std::endl;
    }
}

class B : public class A
{
    int count = 0;
public:
    void test()
    {
        fun();
        count++;
        std::cout << count << std::endl;
    }
}

1.2 常见的线程安全的情况

  • 每个线程对全局变量或者静态变量只有读取的权限,而没有写入的权限,一般来说,这些线程是安全的
  • 类或者接口对于线程来说都是原子操作
  • 多个线程之间的切换不会导致该接口的执行结果存在二义性 

1.3 常见不可重入的情况

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

1.4 常见的可重入的情况

  • 不使用全局变量或静态变量
  • 不使用malloc/free开辟的空间
  • 不调用不可重入函数
  • 不返回静态或去全局数据,所有数据都有函数的调用者提供
  • 使用本地数据,或者通过制作全局数据的本地拷贝来保护全局数据

1.5 可重入与线程安全的联系

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

1.6 可重入与线程安全的区别

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

二、常见锁的概念

2.1 死锁的概念:

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

2.2 出现死锁的场景:

  1. 在加锁之后,又进行了一次加锁操作
  2. 现在有两个线程:线程A和线程B。两个线程都要互相申请两个锁才能进行继续访问,但是由于访问的顺序不同,会造成死锁的现象

2.3 死锁的四个必要条件:(?????)

  1. 互斥条件:一个资源每次只能被一个执行流使用
  2. 请求与保持条件:一个执行流因请求资源而阻塞时,对已获得的资源保持不放
  3. 不剥夺条件:一个执行流已经获得的资源,在未使用完之前,不能强行剥夺
  4. 循环等待条件:若干个执行流之间形成一种头尾相接的循环等待资源的关系

2.4 避免死锁:

  • 破坏死锁的四个必要条件
  • 加锁顺序一致
  • 避免锁未释放的场景
  • 资源一次性分配

三、Linux线程同步

3.1 条件变量

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

3.2 同步概念与竞态条件

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

四、STL、智能指针和线程安全

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

       STL中的容器不是线程安全的,因为STL的设计初衷是将性能挖掘到极致,而一旦涉及到加锁保证线程安全,会对性能造成巨大的影响,而且对于不同的容器,加锁方式的不同,性能也可能不同(例如hash表的锁表和锁桶)

       因此STL默认不是线程安全的,如果需要在多线程环境下使用,往往需要调用者自行保证线程安全。

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

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

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

五、线程安全的单例模式(有待学习)

5.1 什么是单例模式

       单例模式,属于创建类型的一种常用的软件设计模式。通过单例模式的方法创建的类在当前进程中只有一个实例。

5.2 单例模式的特点

       某些类,只应该具有一个对象(实例),就称之为单例。例如一个男人只能有一个媳妇。

       在很多服务器开发场景中,经常需要让服务器加载很多的数据到内存中,往往需要用一个单例的类来管理这些数据。

5.3 饿汉实现方式和懒汉实现方式

举个例子:

  • 吃完饭,立刻洗碗,这种就是饿汉方式。因为下一顿吃的时候可以立刻拿着碗就能吃饭
  • 吃完饭,先把碗放下,然后下一顿饭用到了这个碗再洗这个碗,这就是懒汉方式。

懒汉方式最核心的思想是:延时加载,从而能够优化服务器的启动速度。

5.3.1 饿汉方式实现单例模式

template<typename T>
class Singleton{
    static T date;
public:
    static T* GetTnstance()    
    {
        return &date;
    }
}
// 只要通过Singleton这个包装类来使用T对象,则一个进程中只有一个T对象的实例

5.3.2 懒汉方式实现单例模式

template<typename T>
class Singleton
{
    static T* inst;
public:
    static T* GetInstance()
    {
        if(inst == nullptr)
        {
            inst = new T();
        }
        return inst;
    }
};

       存在一个严重的问题,线程不安全。如果在第一次调用GetInstance的时候, 两个线程同时调用,可能会创建出两份T对象的实例,但是后续再次调用,就没有问题了。

5.4 将线程池改为懒汉方式实现单例模式(线程安全版本)

// 添加单例
static ThreadPool<T> *_instance;
static pthread_mutex_t _lock;


template <class T>
ThreadPool<T> *ThreadPool<T>::_instance;

template <class T>
pthread_mutex_t ThreadPool<T>::_lock = PTHREAD_MUTEX_INITIALIZER;
static ThreadPool<T> *GetInstance()
{
    // 如果是多线程调用以下的代码就会有问题
    // 所以我们需要进行加锁
    // 利用双判断的方式,可以有效减少获取单例的加锁成本,而且保证线程安全
    // 保证第二次之后,所有的线程不用在加锁,直接返回
    if (nullptr == _instance)
    {
        LockGuard lockguard(&_lock);
        if (nullptr == _instance)
        {
            _instance = new ThreadPool<T>;
            _instance->InitThreadPool();
            _instance->Start();
            LOG(DEBUG, "创建线程池单例");
            return _instance;
        }
     }

     LOG(DEBUG, "获取线程池单例");
     return _instance;
}
// 赋值拷贝警用
ThreadPool<T> &operator=(const ThreadPool<T> &) = delete;
ThreadPool(const ThreadPool<T> &) = delete;

六、其他常见的各种锁

6.1 悲观锁

       在每次读取数据时,总是担心数据会被其他线程修改,所以会在取数据前先加锁(读锁,写锁,行锁等),当其他线程想要访问数据时,被阻塞挂起

6.2 乐观锁

       每次取数据的时候,总是乐观的认为数据不会被其他线程修改,因此不上锁,但是在更新数据前,会判断其他数据在更新前有没有对数据进行修改,主要采用两种方式:版本号机制和CAS操作。

6.2.1 版本号机制

       一般是在数据表中加上一个数据版本号version字段,表示数据被修改的次数,当数据被修改时,version值会加一。当线程A要更新数据值时,在读取数据的同时也会读取version值,在提交更新时,若刚才读取到的version值为当前数据库中的version值相等时才更新,否则重试更新操作,直到更新成功。

6.2.2 CAS操作

       当需要更新数据时,判断当前内存值和之前取得的值是否相等,如果相等,则用新值更新;如果不相等,则失败。失败之后,需要进行重试,一般是一个自旋过程,即不断重试。

6.3 自旋锁

       在之前的学习中,我们从来没有讨论过在临界区里线程执行的时长问题:如果时间比较久:推荐其他线程阻塞挂起等待;如果时间比较短:推荐其他线程不要休眠阻塞挂起,而是不断一直抢占锁,直到申请成功(自旋)。

       自旋的过程中,用户会发现自旋锁和之前学习的互斥锁在行为上是相似的,都是阻塞在那里。

七、读者写者问题

7.1 引入读者写者问题

读者写者问题的例子:写文章,打印报纸、杂志,出黑板报

  • 读者总多,写者较少——读者写者问题最常见的情况
  • 有线程向公共资源中写入,其他线程从公共资源中读取数据——读者写者问题

7.1.1 321 原则

  • 3种关系:读者与读者(没有关系),写者与写者(互斥),读者与写者(互斥和同步)
  • 2种角色:读者,写者
  • 1种场景:公共资源

7.1.2 生产者消费者模型与读者写者问题的本质区别

  • 读者和消费者的本质区别:消费者会把数据拿走,而读者不会把数据拿走,只会进行拷贝

7.2 模拟实现一下读者写者的加锁逻辑

        对于公共资源来说,创建一个全局变量,读者锁和写者锁。但是在实际中,只要一个读者锁。

int reader_count = 0;
pthread_mutex_t wlock;
pthread_mutex_t rlock;

对于读者来说:

lock(&rlock); // 先将读者加锁
if(reader_count == 0)
{
    lock(&wlock); // 变量为空,说明第一次读,将写者加锁
        // 这种操作只会进行一次,否则就有死锁
    //如果申请成功,继续运行,不会有任何读者进来
    //如果申请失败,阻塞
}
++ reader_count;
unlock(&rlock);


// 开始进行常规的read

lcok(&rlock);
--read_count;
if(read_count == 0) // 如果读者数量为0,则可以唤醒写者
{
   unlock(&wlock); 
}
unlock(&rlock);

对于写者来说:

lock(&wlock);

// 写入操作

unlock(&wlock);

7.3 了解一下系统中读写锁的接口

7.3.1 初始化读写锁

函数的原型:

#include <pthread.h>

int pthread_rwlock_init(pthread_rwlock_t *restrict rwlock, 
                                 const pthread_rwlockattr_t* restrict attr);

函数的功能:

        进行初始化读写锁
函数的参数:

  • rwlock:指向创建的读写锁对象
  • attr:属性,一般置为nullptr

函数的返回值:

  • 成功返回 0, 失败直接返回错误号

7.3.2 销毁读写锁

函数的原型:

#include <pthread.h>

int pthread_rwlock_destroy(pthread_rwlock_t *rwlock);

函数的功能:

       将所创建好的读写锁进行销毁
函数的参数:

  • rwlock:执行所要销毁的读写锁的指针

函数的返回值:

  • 成功返回 0, 失败直接返回错误号

7.3.3 给读者锁加锁

函数的原型:

#include <pthread.h>

int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock);

7.3.4 给写者锁加锁

函数的原型:

#include <pthread.h>

int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock);

7.3.5 给读者锁和写者锁解锁

函数的原型:

#include <pthread.h>

int pthread_rwlock_unlock(pthread_rwlock_t *rwlock);

7.3.6 代码部分

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

// 读写锁的概念

int count = 0;           // 共享资源
pthread_rwlock_t rwlock; // 创建一个读写锁

#define NUM 5

// 读者
void *reader(void *arg)
{
    pthread_rwlock_rdlock(&rwlock); // 给读者加锁
    std::cout << "Reader conut:" << count << std::endl;
    pthread_rwlock_unlock(&rwlock); // 进行解锁
    return nullptr;
}

// 写者
void *writer(void *arg)
{
    pthread_rwlock_wrlock(&rwlock); // 给写者加锁
    count++;
    pthread_rwlock_unlock(&rwlock); // 给读者解锁
    return nullptr;
}

int main()
{
    pthread_t reader_threads[NUM], writer_threads;
    pthread_rwlock_init(&rwlock, nullptr); // 给读写锁进行初始化

    pthread_create(&writer_threads, nullptr, writer, nullptr);
    for (int i = 0; i < NUM; i++)
    {
        pthread_create(&reader_threads[i], nullptr, reader, nullptr);
    }

    pthread_join(writer_threads, nullptr);
    for (int i = 0; i < NUM; i++)
    {
        pthread_join(reader_threads[i], nullptr);
    }

    pthread_rwlock_destroy(&rwlock);

    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/740256.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

学习笔记——动态路由——RIP(距离矢量协议)

一、距离矢量协议 1、距离矢量协议 矢量行为&#xff1a;协议收到一个路由之后&#xff0c;查看是否可以加入到本地的路由表中&#xff0c;如果可以加入&#xff0c;则可以传递&#xff0c;如果不可以加入&#xff0c;则无法传递。 距离矢量路由协议 RIP基于距离矢量算法(又…

百度Agent初体验(制作步骤+感想)

现在AI Agent很火&#xff0c;最近注册了一个百度Agent体验了一下&#xff0c;并做了个小实验&#xff0c;拿它和零一万物&#xff08;Yi Large&#xff09;和文心一言&#xff08;ERNIE-4.0-8K-latest&#xff09;阅读了相同的一篇网页资讯&#xff0c;输出资讯摘要&#xff0…

【前端面试】二叉树递归模板和题解

递归模板和步骤 递归题目的通用步骤递归模板总结1. 树的遍历&#xff08;DFS&#xff09;2. 二叉树的最大深度3. 二叉树的最近公共祖先 递归题目的记忆技巧 递归题目的通用步骤 明确递归函数的功能&#xff1a;确定递归函数的输入参数和返回值&#xff0c;明确函数的功能。基准…

stm32学习-硬件I2C读取MPU6050

配置流程 第一步&#xff1a;配置I2C外设&#xff0c;对I2C外设进行初始化&#xff08;替换上一篇文章的I2C_Init&#xff09; 第二步&#xff1a;控制外设电路&#xff0c;实现指定地址写的时序&#xff08;替换上一篇文章的WriteReg&#xff09; 第三步&#xff1a;控制外…

如何使能PCIe的ASPM?

1. ASPM概述 PCIe总线的电源管理包含ASPM(Active State Power Management)和软件电源管理两方面内容。所谓的ASPM是指PCIe链路在没有系统软件参与的情况下&#xff0c;由PCIe链路自发进行的电源管理方式。如下是PCIe的ASPM的状态机&#xff0c;其L1是强制性的规定&#xff0c;…

手机数据如何恢复?11 款最佳安卓手机恢复软件

媒体可能由于各种原因而从您的设备中删除&#xff0c;可能是意外或病毒攻击。 在这些情况下&#xff0c;照片恢复应用程序是唯一的解决方案。理想的照片恢复应用程序取决于各种因素&#xff0c;例如存储设备的损坏程度、删除照片后的持续时间以及应用程序使用的恢复算法的有效性…

视频融合共享平台LntonCVS视频监控平台在农场果园等场景的使用方案

我国大江南北遍布着各类果园。传统的安全防范方式主要是建立围墙&#xff0c;但这种方式难以彻底阻挡不法分子的入侵和破坏。因此&#xff0c;需要一套先进、科学、实用且稳定的安全防范报警系统&#xff0c;以及时发现并处理潜在问题。 需求分析 由于果园地处偏远且缺乏有效防…

银联支付,你竟然还不知道它怎么工作?

银联支付咱都用过&#xff0c;微信和支付宝没这么“横行”的时侯&#xff0c;我们取款、转账、付款时用的ATM机、POS机&#xff0c;都是银联支付完成的。 今天&#xff0c;就让咱们了解一下银行卡支付的工作原型。 首先&#xff0c;说说中国银联 中国银联&#xff08;China U…

骨干教师的评选条件

作为一名教师&#xff0c;我常在思考&#xff0c;什么样的教师才能被称为"骨干"&#xff1f;"骨干"不仅仅是一个荣誉的标签&#xff0c;更是一份沉甸甸的责任和使命。究竟什么样的条件才能成为评选骨干教师的标准呢&#xff1f; 必须具备扎实的专业知识。在…

Vscode lanuch.json

Intro 使用launch.json 能够方便的运行需要传很多参数的代码文件 如下&#xff1a; import math import argparse # 1、导入argpase包def parse_args():parse argparse.ArgumentParser(descriptionCalculate cylinder volume) # 2、创建参数对象parse.add_argument(--rad…

推荐三款必备软件,个个五星好评,你一定不要错过

WiseCare365 WiseCare365是一款由WiseCleaner推出的综合性Windows系统优化和加速工具。它集成了多种功能&#xff0c;旨在帮助用户清理、优化和维护电脑系统&#xff0c;提升电脑性能和安全性。 WiseCare365的主要功能包括&#xff1a; 系统清理&#xff1a;它可以清理各种缓存…

mysql学习——SQL中的DQL和DCL

SQL中的DQL和DCL DQL基本查询条件查询聚合函数分组查询排序查询分页查询 DCL管理用户权限控制 学习黑马MySQL课程&#xff0c;记录笔记&#xff0c;用于复习。 DQL DQL英文全称是Data Query Language(数据查询语言)&#xff0c;数据查询语言&#xff0c;用来查询数据库中表的记…

求职经验分享(12):找工作什么最重要?——项目篇

找工作什么最重要&#xff1f;——项目篇 找工作什么最重要&#xff1f;从小粉和老白的视角来看&#xff0c;我们认为最重要的是&#xff1a; 学历 & 实习 & 项目 很多同学经常在说背八股&#xff0c;背八股&#xff0c;其实我认为&#xff0c;八股只是在实习和项目不…

Modbus转Profibus网关在汽车行业的应用

一、前言 在当前汽车工业的快速发展中&#xff0c;汽车制造商正通过自动化技术实现生产的自动化&#xff0c;目的是提高生产效率和减少成本。Modbus转Profibus网关&#xff08;XD-MDPB100&#xff09;应用于汽车行业&#xff0c;主要体现在提升自动化水平、优化数据传输以及实…

每日一题——Python代码实现PAT甲级1006 Sign In and Sign Out(举一反三+思想解读+逐步优化)五千字好文

一个认为一切根源都是“自己不够强”的INTJ 个人主页&#xff1a;用哲学编程-CSDN博客专栏&#xff1a;每日一题——举一反三Python编程学习Python内置函数 Python-3.12.0文档解读 目录 我的写法 代码点评 时间复杂度分析 空间复杂度分析 我要更强 优化建议 优化后的…

ONLYOFFICE 8.1:全面升级,PDF编辑与本地化加强版

目录 &#x1f4d8; 前言 &#x1f4df; 一、什么是 ONLYOFFICE 桌面编辑器&#xff1f; &#x1f4df; 二、ONLYOFFICE 8.1版本新增了那些特别的实用模块&#xff1f; 2.1. 轻松编辑器 PDF 文件 2.2. 用幻灯片版式快速修改幻灯片 2.3. 无缝切换文档编辑、审阅和查…

【教程】安装DGL环境

转载请注明出处&#xff1a;小锋学长生活大爆炸[xfxuezhagn.cn] 如果本文帮助到了你&#xff0c;欢迎[点赞、收藏、关注]哦~ 关于cuda的安装&#xff0c;可以看这个&#xff1a; 【教程】保姆级安装NVIDIA CUDA、CUDNN环境全纪录解决SSH一段时间自动断开报Destination Host Un…

小型数据中心是什么?如何建设?

在数字化时代&#xff0c;小型数据中心正成为许多企业和组织加强数据管理和服务扩展的理想选择。与传统大型数据中心相比&#xff0c;小型数据中心以其灵活性、高效性和相对较低的运营成本吸引着越来越多的关注。然而&#xff0c;要成功建设一个小型数据中心&#xff0c;并确保…

2024年数据、自动化与智能计算国际学术会议(ICDAIC 2024)

全称&#xff1a;2024年数据、自动化与智能计算国际学术会议&#xff08;ICDAIC 2024&#xff09; 会议网址:http://www.icdaic.com 会议地点: 厦门 投稿邮箱&#xff1a;icdaicsub-conf.com投稿标题&#xff1a;ArticleTEL。投稿时请在邮件正文备注&#xff1a;学生投稿&#…

测评:【ONLYOFFICE】版本更迭与AI加持下的最新ONLYOFFICE桌面编辑器8.1

你是否还在为没有一款合适的在线桌面编辑器而苦恼&#xff1f;你是否还在因为办公软件的选择过少而只能使用WPS或者office&#xff1f;随着办公需求的不断变化和发展&#xff0c;办公软件也在不断更新和改进。ONLYOFFICE 作为一款全功能办公软件&#xff0c;一直致力于为用户提…