【Linux】多线程编程基础

news2024/11/15 13:48:34
在这里插入图片描述

💻文章目录

  • 📄前言
  • 🌺linux线程基础
    • 线程的概念
      • 线程的优缺点
      • 线程与进程的区别
    • 线程的创建
  • 🌻linux线程冲突
    • 概念
    • 互斥锁函数介绍
    • 加锁的缺点
  • 📓总结


📄前言

无论你是否为程序员,相信多线程这个词汇应该都有所耳闻,像是在某个优化很差的游戏中听闻这游戏甚至是单线程的,如果你对多线程感兴趣,不妨点进本文来学习多线程编程,即使没有深厚的C/C++编程基础,你也能到本文学习到如何编写多线程程序。

🌺linux线程基础

线程的概念

线程指的是系统中的执行路径,每个线程都线程系统中的一切进程都至少有一个线程,它们共享同一个进程.

其实在linux中,实际并没有真正的线程,线程通常被称为轻量级进程(LWP),这是因为在linux的实现中,线程和进程并没有什么本质的区别,只是线程被设计得更加轻量,以便更高效实现并发执行。


   线程pcb
   
  task_struc   -+                        +-------------------+
  +--------+    |                        |    内核映射区域     |
  |        |    |                        +-------------------+
  +--------+    |                        ||
                |                        +-------------------+
  task_struc    |                        |                   |
  +--------+    |                        |      共享库        |
  |        |    |                        |                   |
  +--------+    |                        +-------------------+
                |                        ||
  task_struc    |    指向同一地址空间      +-------------------+
  +--------+    |-------------------+>   |      数据段        |
  |        |    |                        +-------------------+
  +--------+    |                        |   未初始化数据区    |
                |                        +-------------------+
  task_struc    |                        |   已初始化数据区    |
  +--------+    |                        +-------------------+
  |        |    |                        |      代码段        |
  +--------+   -+                        +-------------------+

线程的优缺点

  • 优点:
  1. 共享资源:在同一线程的线程共享着大部分内存空间,如:代码段、数据段、文件描述符、堆、共享内存区等。这使得线程间通信非常地高效,无需IPC机制开销。
  2. 独立调度:虽然线程中大部分地址空间都与主线程共享,但线程也有自己的一部分数据,如:栈与寄存器状态,这使得他们可以独立于其他线程运行。
  3. 响应性:在多线程程序中,一个进程的阻塞不会影响到其他进程。
  4. 资源利用率:多线程可以提高在多核处理器上运行的效率,实现并行执行。
  • 缺点:
  1. 编程困难:因为多线程需要考虑到临界区、互斥、同步等问题,所以对程序员的代码能力要求较高。
  2. 同步复杂性:多线程的资源共享需要谨慎处理,否则会出现数据二义性问题。
  3. 调试困难: 多线程调试一直都是令人头疼的问题,因为bug可能会难以复现,并且不是所有调试工具都支持多线程调试。
  4. 健壮性:如果任意一个线程触发了异常,则整个程序都会终止。

线程与进程的区别

  1. 定义:进程是资源分配的最小单位,线程则是cpu调度执行的最小单位。
  2. 资源共享:进程之间资源独立,同一进程内的线程共享进程资源。
  3. 创建开销:线程的创建和切换开销都小于进程,因为线程之间资源共享。

线程的创建

  • 函数接口介绍:头文件:<pthread.h>
  1. 创建线程
// 创建线程
int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine) (void *), void *arg);

// pthread_t 是 POSIX 线程(Pthreads)库中定义的一个数据类型,用于唯一标识一个线程
  • 参数
    • thread: 线程
    • attr: 指定线程属性的指针,可设为NULL
    • start_routine:线程开始执行的函数
    • arg: start_routine 函数的参数
  1. 回收线程
// 等待线程结束并回收线程的资源,防止类似“僵尸进程”的情况
int pthread_join(pthread_t thread, void** retval);
  • 参数:
    • thread:用于回收的线程id;
    • retval:用于存储线程的返回值。
  1. 退出线程:
// 用于终止当前的线程,因为exit会终止整个进程,所以有了这个函数
void pthread_exit(void* retval);
  • 参数:
    • retval:退出线程时返回的值
  1. 分离线程:
// 如果觉得join操作是一种负担的时候,可以使用pthread_detach
// 用于分离线程,当线程结束时,自动回收线程资源。
int pthread_detach(pthread_t thread);
  • 参数:
    • thread:分离的线程id

介绍完了函数接口,就到实践的时间啦。

  • 使用函数
#include <pthread.h>
#include <iostream>

void *thread_func(void *arg)
{
    // 获取当前线程的tid
    std::cout << "Thread" << (char *)arg << " id:" << gettid() << " started" << std::endl;
    int cnt = 10;
    while (cnt >= 0)
    {
        std::cout << "Thread" << (char *)arg << " id:" << gettid() << " is running, cnt = " << cnt << std::endl;
        cnt--;
        sleep(1);
    }
    // 子线程退出
    pthread_exit(nullptr); // 可有可无
}

void *func_test(void* args)
{
	printf("I LOVE LINUX\n");
	pthread_detach(pthread_self());	// 使用pthread_self()可以使子线程自己分离。
    return nullptr;
}

int main()
{
    pthread_t thread, thread2;
    pthread_create(&thread, nullptr, thread_func, (void *)"-1");
    pthread_create(&thread2, nullptr, func_test, nullptr);
    // 主线程等待子线程结束
    pthread_join(thread, NULL); // 回收线程
    return 0;
}

🌻linux线程冲突

概念

多线程的高效率也是存在着代价的,当多个线程同时访问一份资源时,就会发生线程冲突(数据二义性),我们一般将这些多个线程都要访问的资源称为临界区

要探讨数据二义性问题,就得从汇编代码开始讲解

; 例如一个简单的++操作,看似只做了一个操作,但在汇编中却并不是这样。

MOV EAX, [x]   ; 将x的值加载到EAX寄存器
INC EAX        ; 将EAX寄存器的值增加1		
MOV [x], EAX   ; 将修改后的值存回内存位置x

; 多个线程同时访问这个资源(x),当线程1在将x放入寄存器EAX时,线程2可能就已经将x++,并改变了内存的数值
; 线程1将寄存器的值++后,又放回了x的内存。建议使用vs2022 进行反汇编调试来观看现象。
  • 线程冲突演示
#include <pthread.h>
#include <iostream>

int x = 0;

void *func(void *args)
{
    for (int i = 0; i < 100000000; i++)	//数值越大,冲突概率越大
        ++x;
        
    pthread_exit(nullptr);
}

int main()
{
    // 线程冲突演示
    pthread_t pid1, pid2;
		
    pthread_create(&pid1, nullptr, func, nullptr);
    pthread_create(&pid2, nullptr, func, nullptr);

    pthread_join(pid1, nullptr);	//回收线程
    pthread_join(pid2, nullptr);

    cout << "x = " << x << endl;

    return 0;
}
// 结果:
// x = 154698688

解决方案: 为了解决这种情况,就得当线程访问临界区资源时限制为一个线程访问,也就是说,需要给线程加锁。

互斥锁函数介绍

  1. 创建锁
// 初始化锁 pthread_mutex_t 用于声明互斥量(mutex)对象。

// 静态加锁 (全局变量或静态进行初始化)
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

// 动态初始化
pthread_mutex_t mutex;
pthread_mutex_init(&mutex, NULL);
  1. 线程加锁
// 给线程加锁
int pthread_mutex_lock(pthread_mutex_t *mutex);
  • 参数:
    • mutex:指向互斥锁对象
  1. 互斥锁解锁
int pthread_mutex_unlock(pthread_mutex_t *mutex);
  • 参数:
    • mutex:指向需要解锁的互斥锁对象的指针。

注意:加锁操作本身时原子性的,所以不用担心锁的二义性。

  • 互斥锁的使用
int x = 0;
// 初始化锁对象
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

void *func(void *args)
{
    for (int i = 0; i < 100000000; i++)
    {
        pthread_mutex_lock(&mutex);	//加锁
        ++x;
        pthread_mutex_unlock(&mutex);	//解锁
    }
    pthread_exit(nullptr);
}

int main()
{
    // 线程冲突演示
    pthread_t pid1, pid2;

    pthread_create(&pid1, nullptr, func, nullptr);
    pthread_create(&pid2, nullptr, func, nullptr);

    pthread_join(pid1, nullptr);
    pthread_join(pid2, nullptr);

    cout << "x = " << x << endl;

    return 0;
}

加锁的缺点

如果我们尝试运行程序,会发现加锁后的运行速度明显慢了不少。锁的使用会增加性能的开销,而且线程可能会变成串行执行,为了避免多余的性能开销,每次使用锁都应该避免将非临界区的资源加锁。

在一些特殊的情况下,可能会

  • 死锁演示:
pthread_mutex_t mtx = PTHREAD_MUTEX_INITIALIZER;

void *func(void *args)
{
    int* cnt = (int*)args;
	if(*cnt <= 0)	return nullptr;

 	pthread_mutex_lock(&mtx);		// 第二次递归时等待着线程解锁

    std::cout << "func()" << std::endl;	
  	--(*cnt);
    func(args);		// 递归进入下一层,但锁还没解锁。

    pthread_mutex_unlock(&mtx);	// 程序永远走不到这里。
		
    return nullptr;
}

int main()
{
    // 线程冲突演示
    pthread_t pid;
    int* cnt = new int(10);
    pthread_create(&pid, nullptr, func, (void*)cnt);

    pthread_join(pid, nullptr);
    return 0;
}

📓总结

多线程编程
优点缺点
资源共享线程间共享进程资源(如代码段、数据段、文件描述符等),使得线程间通信非常高效,无需通过IPC机制开销。多线程的资源共享需要通过同步机制(如互斥锁)来管理,否则可能导致数据不一致或竞争条件的问题。
独立调度线程可以独立于其他线程运行,拥有自己的执行路径。这增加了应用程序的响应性和处理效率。线程调度引入了上下文切换的开销,尤其是在高度竞争的环境中,可能降低整体性能。
效率提升在多核处理器上,多线程能够利用额外的核心执行更多的任务,提高了程序的执行效率和资源利用率。编写高效的多线程程序需要深入理解并发、同步等概念,增加了开发的复杂度。

多线程编程是一把双刃剑,使用多线程能够显著提升程序的性能,但它也为程序带来了许多潜在的风险,在处理器核心数越来越多的当今,学习多线程也变得越发重要,希望本文对你的学习有所帮助。

📜博客主页:主页
📫我的专栏:C++
📱我的github:github

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

小白也能在3分钟完成短剧解说的剪辑,这是真的!

3分钟的解说视频&#xff0c;真的需要1小时的手工剪辑吗&#xff1f; 生成解说视频需要经过素材准备、解说词创作、声音录制、视频剪辑和视频合成等多个步骤&#xff0c;每个步骤都需要投入一定的时间和精力&#xff0c;因此整个过程较为耗时耗力。 1. 素材准备&#xff1a; 需…

【LINUX笔记】驱动开发框架

应用程序调动驱动程序 驱动模块运行模式 模块加载-卸载 加载卸载注册函数 加载 驱动编译完成以后扩展名为.ko&#xff0c;有两种命令可以加载驱动模块&#xff1a; insmod和modprobe 驱动卸载 驱动注册注销 //查看当前已经被使用掉的设备号 cat /proc/devices 实现设备的具…

AI系统性学习06—开源中文语言大模型

1、ChatGLM ChatGLM-6B的github地址&#xff1a;https://github.com/THUDM/ChatGLM-6B ChatGLM-6B 是一个开源的、支持中英双语的对话语言模型&#xff0c;基于 General Language Model (GLM) 架构&#xff0c;具有 62 亿参数。结合模型量化技术&#xff0c;用户可以在消费级…

【Java Web基础】一些网页设计基础(二)

文章目录 1. Bootstrap导航栏设计1.1 代码copy与删减效果1.2 居中属性与底色设置1.3 占不满问题分析1.4 字体颜色、字体大小、字体间距设置1.5 修改超链接hover颜色&#xff0c;网站首页字体颜色 1. Bootstrap导航栏设计 1.1 代码copy与删减效果 今天设计导航栏&#xff0c;直…

第4关:创建工程项目表J,并插入数据

任务描述 工程项目表J由工程项目代码(JNO)、工程项目名(JNAME)、工程项目所在城市(CITY)组成。创建工程项目表J(JNO,JNAME,CITY)&#xff0c;并在J表中插入下图数据。 相关知识 1、MySQL创建表的基本语法如下&#xff1a; 其中&#xff0c;table_name 是要创建的表的名称&…

Hololens 2应用开发系列(4)——MRTK基础知识及配置文件配置(下)

Hololens 2应用开发系列&#xff08;4&#xff09;——MRTK基础知识及配置文件配置&#xff08;下&#xff09; 一、前言二、边界系统&#xff08;Boundary&#xff09;三、传送系统&#xff08;Teleport&#xff09;四、空间感知系统&#xff08;Spatial Awareness&#xff09…

Pytorch神经网络-元组/列表如何喂到神经网络中

&#x1f4da;博客主页&#xff1a;knighthood2001 ✨公众号&#xff1a;认知up吧 &#xff08;目前正在带领大家一起提升认知&#xff0c;感兴趣可以来围观一下&#xff09; &#x1f383;知识星球&#xff1a;【认知up吧|成长|副业】介绍 ❤️感谢大家点赞&#x1f44d;&…

设计编程网站集:生活部分:饮食+农业,植物(暂记)

这里写目录标题 植物相关综合教程**大型植物&#xff1a;****高大乔木&#xff08;Trees&#xff09;&#xff1a;** 具有坚硬的木质茎&#xff0c;通常高度超过6米。例如&#xff0c;橡树、松树、榉树等。松树梧桐 **灌木&#xff08;Shrubs&#xff09;&#xff1a;** 比乔木…

基于Jenkins + Argo 实现多集群的持续交付

作者&#xff1a;周靖峰&#xff0c;青云科技容器顾问&#xff0c;云原生爱好者&#xff0c;目前专注于 DevOps&#xff0c;云原生领域技术涉及 Kubernetes、KubeSphere、Argo。 前文概述 前面我们已经掌握了如何通过 Jenkins Argo CD 的方式实现单集群的持续交付&#xff0c…

基于Springboot的在线投稿系统+数据库+免费远程调试

项目介绍: Javaee项目&#xff0c;springboot项目。采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构&#xff0c;通过Spring SpringBoot Mybatis VueMavenLayui来实现。MySQL数据库作为系统数据储存平台&a…

Java安全 反序列化(3) CC1链-TransformedMap版

Java安全 反序列化(3) CC1链-TransformedMap版 本文尝试从CC1的挖掘思路出发&#xff0c;理解CC1的实现原理 文章目录 Java安全 反序列化(3) CC1链-TransformedMap版配置jdk版本和源代码配置前记 为什么可以利用一.CC链中的命令执行我们可以尝试一下通过InvokerTransformer.tr…

分布式异步任务框架celery

Celery介绍 github地址&#xff1a;GitHub - celery/celery: Distributed Task Queue (development branch) 文档地址&#xff1a;Celery - Distributed Task Queue — Celery 5.3.6 documentation 1.1 Celery是什么 celery时一个灵活且可靠的处理大量消息的分布式系统&…

数据库关系运算理论:传统的集合运算概念解析

✨✨ 欢迎大家来访Srlua的博文&#xff08;づ&#xffe3;3&#xffe3;&#xff09;づ╭❤&#xff5e;✨✨ &#x1f31f;&#x1f31f; 欢迎各位亲爱的读者&#xff0c;感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢&#xff0c;在这里我会分享我的知识和经验。&am…

如何在wps的excel表格里面使用动态gif图

1、新建excel表格&#xff0c;粘贴gif图到表格里面&#xff0c;鼠标右键选择超链接。 找到源文件&#xff0c; 鼠标放到图片上的时候&#xff0c;待有个小手图标&#xff0c;双击鼠标可以放大看到动态gif图。 这种方式需要确保链接的原始文件位置和名称不能变化&#xff01;&a…

网工内推 | 云计算工程师,HCIE认证优先,最高18k*14薪

01 杭州中港科技有限公司 招聘岗位&#xff1a;云计算工程师 职责描述&#xff1a; 1、承担云计算相关工程交付、业务上云及售前测试&#xff0c;从事虚拟化、桌面云、存储、服务器、数据中心、大数据、相关产品的工程项目交付或协助项目交付。 2、承担云计算维护工程师职责&…

深入理解Mysql索引底层原理(看这一篇文章就够了)

目录 前言 1、Mysql 索引底层数据结构选型 1.1 哈希表&#xff08;Hash&#xff09; 1.2 二叉查找树(BST) 1.3 AVL 树和红黑树 1.4 B 树 1.5 B树 2、Innodb 引擎和 Myisam 引擎的实现 2.1 MyISAM 引擎的底层实现&#xff08;非聚集索引方式&#xff09; 2.2 Innodb 引…

L4 级自动驾驶汽车发展综述

摘要:为了减小交通事故概率、降低运营成本、提高运营效率,实现安全、环保的出行,自动驾驶 技术的发展已成为大势所趋,而搭配有L4 级自动驾驶系统的车辆是将车辆驾驶全部交给系统。据此,介绍了自动驾驶汽车的主流技术解决方案;分析了国内外L4 级自动驾驶汽车的已发布车型、…

Python 安装目录及虚拟环境详解

Python 安装目录 原文链接&#xff1a;https://blog.csdn.net/xhyue_0209/article/details/106661191 Python 虚拟环境 python 虚拟环境图解 python 虚拟环境配置与详情 原文链接&#xff1a;https://www.cnblogs.com/hhaostudy/p/17321646.html

C语言易错知识点:二级指针、数组指针、函数指针

指针在C语言中非常关键&#xff0c;除开一些常见的指针用法&#xff0c;还有一些可能会比较生疏&#xff0c;但有时却也必不可少&#xff0c;本文章整理了一些易错知识点&#xff0c;希望能有所帮助&#xff01; 1.二级指针&#xff1a; parr是一个指针数组&#xff0c;其中每…

std::shared_ptr与std::make_unique在类函数中的使用

在最近学习cartographer算法的时候&#xff0c;发现源码中大量的使用了std::shared_ptr与std::make_unique&#xff0c;对于这些东西之前不是很了解&#xff0c;为了更好的理解源代码&#xff0c;因此简单学习了一下这块内容的使用&#xff0c;在这里简单记个笔记。 std::shar…