Uinx线程详解

目录

 

一.什么是线程?

并发(Concurrency)

并行(Parallelism)

  1.1 线程的概念

  1.2 线程的基本函数

 1.3 线程的基本使用例子:

 二.线程的属性

2.1线程属性使用例子

三.线程互斥

 3.1互斥锁

3.2互斥锁常用函数 

 3.3互斥锁使用例子


一.什么是线程?

    在Linux中,线程是操作系统能够进行运算调度的最小单位,它被包含在进程之中,是进程中的实际运作单位。同一进程中的多条线程将共享该进程所拥有的全部资源,但每条线程都有各自的调用栈和程序计数器,能够独立运行。

   那么什么是并发与并行呢?        

并发(Concurrency)

并发是指系统能够处理多个同时发生的任务。在并发编程中,多个任务在同一时间段内执行,但它们可能不是同时执行。并发可以通过时间分片(time slicing)来实现,操作系统快速地在多个任务之间切换,给人一种同时执行的错觉。

并发的主要目的是提高资源的利用率和系统的响应速度。例如,在一个Web服务器中,可以同时处理多个用户的请求,这样就可以更有效地利用服务器的资源,并缩短用户的等待时间。

并行(Parallelism)

并行是指系统能够在同一时刻执行多个任务。并行编程需要多个处理器或者多个核心,每个处理器或核心可以独立地执行一个任务。并行是真正的“同时”执行,它通过增加计算资源来提高性能。

并行的主要目的是通过同时执行多个任务来提高性能和缩短任务的完成时间。例如,在一个多核处理器上,可以同时运行多个线程或者进程,这样可以显著减少执行复杂计算或者数据处理任务所需的时间。

  1.1 线程的概念

  • 轻量级进程(Lightweight Process, LWP):线程又被称为轻量级进程,因为线程的创建、撤销和切换比进程更快。
  • 线程的属性:线程具有就绪、阻塞和运行三种基本状态,以及创建和终止两种辅助状态。
  • 线程的执行:线程可以并发执行,这意味着它们似乎是在同时运行,但实际上是在CPU时间片轮转机制下快速切换执行。

如何使用线程:

在Linux中,线程可以通过 POSIX 线程(pthread)库来使用。这个库提供了一系列函数来创建、同步和管理线程。

线程共享资源

1.文件描述符表
2.每种信号的处理方式
3.当前工作目录
4.用户ID和组ID
5.内存地址空间 (.text/.data/.bss/heap/共享库)


线程非共享资源

1.线程id
2.处理器现场和栈指针(内核栈)
3.独立的栈空间(用户空间栈)
4.errno变量
5.信号屏蔽字
6.调度优先级

线程优、缺点

  • 优点: 1. 提高程序并发性 2. 开销小 3. 数据通信、共享数据方便
  • 缺点: 1. 库函数,不稳定 2. 调试、编写困难、gdb不支持 3. 对信号支持不好
  • 优点相对突出,缺点均不是硬伤。Linux下由于实现方法导致进程、线程差别不是很大

  1.2 线程的基本函数

基本函数介绍和使用:

  1. 线程创建:pthread_create

    int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine) (void *), void *arg);
    
    • thread:指向线程标识符的指针。
    • attr:线程属性,通常设为NULL表示使用默认属性。
    • start_routine:线程运行函数的起始地址。
    • arg:运行函数的参数。
  2. 线程终止:pthread_exit

    void pthread_exit(void *retval);
    
    • retval:线程的返回值。
  3. 等待线程终止:pthread_join

    int pthread_join(pthread_t thread, void **retval);
    
    • thread:等待终止的线程标识符。
    • retval:存储线程返回值的指针。
  4. 线程取消:pthread_cancel

    int pthread_cancel(pthread_t thread);
    
    • thread:要取消的线程标识符。
  5. 线程互斥锁:pthread_mutex_lock/unlock

    int pthread_mutex_lock(pthread_mutex_t *mutex);
    int pthread_mutex_unlock(pthread_mutex_t *mutex);
    
    • mutex:互斥锁变量,用于同步线程对共享资源的访问。

 1.3 线程的基本使用例子:

#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>// 线程运行函数
void *print_message_function(void *ptr) {char *message;message = (char *) ptr;printf("%s \n", message);return NULL;
}int main() {pthread_t thread1, thread2;char *message1 = "Thread 1";char *message2 = "Thread 2";// 创建线程1if(pthread_create(&thread1, NULL, print_message_function, (void*) message1)) {printf("Error creating thread 1\n");return 1;}// 创建线程2if(pthread_create(&thread2, NULL, print_message_function, (void*) message2)) {printf("Error creating thread 2\n");return 1;}// 等待线程终止if(pthread_join(thread1, NULL)) {printf("Error joining thread 1\n");return 2;}if(pthread_join(thread2, NULL)) {printf("Error joining thread 2\n");return 2;}return 0;
}

在这个例子中:

  • 定义了一个print_message_function函数,它接收一个指向字符数组的指针,并打印出来。
  • main函数中,创建了两个线程thread1thread2,它们都执行print_message_function函数,但分别传递了不同的消息。
  • 使用pthread_create创建线程,使用pthread_join等待线程结束。

程序执行时,会创建两个线程,它们可能会并发地运行,打印出各自的消息。使用pthread_join确保了主程序会等待这两个线程完成它们的任务后再退出。

 二.线程的属性

线程属性(Thread Attributes)是可以用来定制线程的各种特性的集合。在创建线程时,可以指定线程的属性,如果不指定,线程会使用默认属性。线程属性通过pthread_attr_t结构体来表示,并且在使用前需要初始化。

常见的线程属性包括:

  • detachstate:线程的分离状态属性,可以设置为PTHREAD_CREATE_JOINABLE(默认值,可以调用pthread_join来等待线程结束)或PTHREAD_CREATE_DETACHED(线程一旦结束,其资源立即被回收,不能被等待)。
  • schedpolicy:线程的调度策略,可以是SCHED_OTHER(默认值,普通轮转调度),SCHED_FIFO(先来先服务调度),或SCHED_RR(时间片轮转调度)。
  • inheritsched:线程的继承调度策略,可以设置为PTHREAD_EXPLICIT_SCHED(显式指定线程的调度属性)或PTHREAD_INHERIT_SCHED(继承创建它的线程的调度属性,默认值)。
  • scope:线程的作用域,可以设置为PTHREAD_SCOPE_SYSTEM(与系统范围内的线程竞争CPU时间)或PTHREAD_SCOPE_PROCESS(仅与同一进程内的线程竞争CPU时间,通常是默认值)。

2.1线程属性使用例子

#include <pthread.h>int main() {pthread_attr_t attr;pthread_attr_init(&attr); // 初始化线程属性// 设置线程为分离状态pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);// 使用线程属性创建线程pthread_t thread;pthread_create(&thread, &attr, thread_function, NULL);// 清理线程属性pthread_attr_destroy(&attr);// ... 其他代码 ...return 0;
}

三.线程互斥

 3.1互斥锁

互斥锁(Mutex):

互斥锁是一种同步机制,用于防止多个线程同时访问共享资源。在多线程环境中,当多个线程尝试同时访问同一资源时,互斥锁确保任何时刻只有一个线程能够访问该资源。

3.2互斥锁常用函数 

常用的互斥锁函数:

  • 初始化互斥锁:pthread_mutex_init

    c

    复制

    int pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t *attr);
    
    • mutex:指向互斥锁变量的指针。
    • attr:互斥锁属性,通常设为NULL表示使用默认属性。
  • 锁定互斥锁:pthread_mutex_lock

    c

    复制

    int pthread_mutex_lock(pthread_mutex_t *mutex);
    
    • mutex:指向互斥锁变量的指针。
  • 尝试锁定互斥锁:pthread_mutex_trylock

    c

    复制

    int pthread_mutex_trylock(pthread_mutex_t *mutex);
    
    • mutex:指向互斥锁变量的指针。如果互斥锁已经被锁定,函数不会阻塞,而是返回错误码。
  • 解锁互斥锁:pthread_mutex_unlock

    c

    复制

    int pthread_mutex_unlock(pthread_mutex_t *mutex);
    
    • mutex:指向互斥锁变量的指针。
  • 销毁互斥锁:pthread_mutex_destroy

    c

    复制

    int pthread_mutex_destroy(pthread_mutex_t *mutex);
    
    • mutex:指向互斥锁变量的指针。

 3.3互斥锁使用例子

#include <pthread.h>// 全局变量
pthread_mutex_t lock; // 互斥锁
int shared_data = 0; // 共享数据// 线程函数
void *thread_function(void *arg) {pthread_mutex_lock(&lock); // 加锁shared_data++; // 访问共享数据pthread_mutex_unlock(&lock); // 解锁return NULL;
}int main() {pthread_mutex_init(&lock, NULL); // 初始化互斥锁pthread_t threads[10];for(int i = 0; i < 10; i++) {pthread_create(&threads[i], NULL, thread_function, NULL); // 创建线程}for(int i = 0; i < 10; i++) {pthread_join(threads[i], NULL); // 等待线程结束}pthread_mutex_destroy(&lock); // 销毁互斥锁printf("Shared data: %d\n", shared_data); // 输出共享数据的最终值return 0;
}

在这个例子中,我们创建了一个互斥锁lock来保护对shared_data的访问。每个线程在访问shared_data之前都会尝试锁定互斥锁,完成访问后解锁。这样可以确保在任一时刻只有一个。

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

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

相关文章

C语言笔试题之求解X的平方根

求解X的平方根 一、实例要求 1、给定一个非负整数 x &#xff0c;计算并返回 x 的算术平方根 &#xff1b;2、由于返回类型是整数&#xff0c;结果只保留整数部分 &#xff0c;小数部分将被舍去&#xff1b;3、不允许使用任何内置指数函数、运算符&#xff1b; 二、实例分析…

python作业

1.找出10000以内能被5或6整除&#xff0c;但不能被两者同时整除的数(函数) 2.写一个方法&#xff0c;计算列表所有偶数下标元素的和(注意返回值) 3.根据完整的路径从路径中分离文件路径、文件名及扩展名。 4.根据标点符号对字符串进行分行 5.去掉字符串数组中每个字符串的空格 …

江协STM32:定时器定时中断和定时器定时闹钟

定时器中断 新建文件 按这个图来编写程序 第一步&#xff1a;RCC开启时钟&#xff0c;定时器到基准时钟和整个外设到工作时钟就会同时打开 第二步&#xff1a;选择时基单元的时钟源&#xff0c;对于定时中断选择内部时钟源 第三步&#xff1a;配置时基单元&#xff0c;ARR,P…

Golang Channel底层实现原理

1、本文讨论Channel的底层实现原理 首先&#xff0c;我们看Channel的结构体 简要介绍管道结构体中&#xff0c;几个关键字段 在Golang中&#xff0c;管道是分为有缓冲区的管道和无缓冲区的管道。 这里简单提一下&#xff0c;缓冲区大小为1的管道和无缓冲区的管道的区别&…

维基百科推广方法及注意事项解析-华媒舍

1. 维基百科 维基百科是一个自由而开放的在线百科全书&#xff0c;由志愿者共同创建和编辑。它是全球最大的百科全书&#xff0c;包含了广泛的主题和知识。作为一个公共平台&#xff0c;维基百科是广告和宣传的禁区&#xff0c;但它可以是一个有效的推广工具&#xff0c;帮助您…

ENSP华为防火墙WEB登录操作指南

ENSP华为防火墙WEB登录操作指南 华为防火墙登录WEB 1、华为防火墙配置&#xff1a;&#xff08;需要在互联接口下放通https和ping&#xff09; int g0/0/0 service-manage https permit service-manage ping permit 2、电脑需要配置虚拟网卡 3、虚拟网卡与云和防火墙配置的IP地…

【学习心得】Numpy学习指南或复习手册

本文是自己在学习Numpy过后总是遗忘的很快&#xff0c;反思后发现主要是两个原因&#xff1a; numpy的知识点很多&#xff0c;很杂乱。练习不足&#xff0c;学习过后一段时间不敲代码就会忘记。 针对这两个问题&#xff0c;我写了这篇文章。希望将numpy的知识点织成一张网&…

【Vmware】 debian 12 安装教程

1.前提说明 VMware 17.5.1 (自行安装)&#xff0c;参考Debian 12maven 3.8.7git 2.39.2jdk 1.8 / 11 / 17 1.1.Debian 下载 访问(https://www.debian.org/download) 下载 Debian 这是 Debian 12&#xff0c;代号为 bookworm&#xff0c;网络安装&#xff0c;用于 64 位 PC&a…

机器学习算法与应用

机器学习是人工智能领域中的一个重要分支&#xff0c;它通过构建和训练模型来使计算机系统具备从数据中学习并做出预测或决策的能力。在这篇博客中&#xff0c;我们将深入探讨机器学习算法的基本原理、常见类型以及在实际应用中的案例。 机器学习算法概述 介绍机器学习的基本概…

PLC通过Modbus转Profine网关接温度传感器方案

Modbus转Profinet网关用于实现Modbus协议和Profinet协议之间的数据转换和传输。Modbus转Profinet网关接温度传感器的方案主要涉及将Modbus协议的温度传感器数据转换为Profinet协议&#xff0c;以便与工业自动化系统中的其他设备进行通信和数据交换。 以下是实现此方案的基本步骤…

[StartingPoint][Tier0]Mongod

Task 1 How many TCP ports are open on the machine? (机器上打开了多少个 TCP 端口&#xff1f;) Example: $ sudo nmap -sS -T4 10.129.222.112 -p 27017,22 2 Task 2 Which service is running on port 27017 of the remote host? (哪个服务正在远程主机的端口 270…

系统架构设计师-23年-论文题目

系统架构设计师-23年-论文题目 更多软考知识请访问 https://ruankao.blog.csdn.net/ 摘要字数在400字以内&#xff0c;可以分条叙述&#xff0c;但不允许有图、表、流程图。 正文字数为2000字至300字&#xff0c;文中可以分条叙述&#xff0c;但不要全部用分条叙述的方式。 …

设计模式总结-面向对象设计原则

面向对象设计原则 面向对象设计原则简介单一职责原则单一职责原则定义单一职责原则分析单一职责原则实例 开闭原则开闭原则定义开闭原则分析开闭原则实例 里氏代换原则里氏代换原则定义里氏代换原则分析 依赖倒转原则依赖倒转原则定义依赖倒转原则分析依赖倒转原则实例 接口隔离…

向量旋转操作之分段递归交换

开篇 这是对于之前一维向量左旋操作问题的最后一个解法&#xff0c;也是关于这个问题的最后一篇文章。在之前的文章中&#xff0c;我们分别用求逆法、取模置换法对该问题进行了解答&#xff0c;今天&#xff0c;使用的是分段递归的方式。 问题概要 将一个n元一维向量向左旋转i个…

深入了解JUnit 5:新一代Java单元测试框架

深入了解JUnit 5&#xff1a;新一代Java单元测试框架 近年来&#xff0c;Java领域的单元测试框架发展迅速&#xff0c;而JUnit 5作为JUnit系列的最新版本&#xff0c;为开发人员提供了更多的功能和灵活性。在本文中&#xff0c;我们将介绍JUnit 5&#xff0c;并探讨其与JUnit 4…

探索数据库-------MYSQL故障排除与优化

目录 mysql逻辑架构图 一、MySQL 数据库故障 1.1 MySQL 单实例故障排查 1.1.1故障现象 1 1.1.2故障现象 2 1.1.3故障现象 3 1.1.4故障现象 4 1.1.5故障现象 5 1.1.6故障现象 6 1.1.7故障现象 7 1.1.8故障现象 8 1.2MySQL 主从故障排查 1.2.1故障现象 1 1.2.2故障…

电感与磁珠的区别以及在EMC的作用

电感与磁珠的区别以及在EMC的作用 电感的定义和特性电感的频率特性噪声对策方法电感的直流叠加饱和绕组型电感的特性 电感的定义和特性 电感是能够把电能转化为磁能而存储起来的元器件。电感器具有一定的电感&#xff0c;它只阻碍电流的变化。电感器又称扼流器、自电抗器、动态…

Cocos Creator 定时任务

目录 Javascript 内置定时器 Cocos 计时器 update 函数 参考 Javascript 内置定时器 setTimeout&#xff0c; setInterval Cocos 计时器 先创建一个指向某个组件的变量&#xff0c;变量名为 component 开始一个计时器 component.schedule(function() {// 这里的 this 指…

02 OSI和TCP/IP参考模型

OSI参考模型优点&#xff1a; 1. 分层结构&#xff1a;OSI参考模型将网络通信划分为七个不同的层次&#xff0c;每个层次都有特定的功能和责任。这样的分层结构使得网络通信变得模块化&#xff0c;易于理解、设计和维护。 2. 标准化&#xff1a;OSI参考模型的制定依据了通信领…

NPW(监控片的)的要点精讲

半导体的生产过程已经历经数十年的发展&#xff0c;其中主要有两个大的发展趋势&#xff0c;第一&#xff0c;晶圆尺寸越做越大&#xff0c;到目前已有超过70%的产能是12寸晶圆&#xff0c;不过18寸晶圆产业链推进缓慢&#xff1b;第二&#xff0c;电子器件的关键尺寸越做越小&…