Linux多线程(8.3 线程同步与互斥)
3. 线程的同步与互斥
为什么需要同步与互斥
一个进程运行时,数据存储在内存中。如果一个数据要进行运算,必须先将数据拷贝到寄存器中。比如要对栈上的一个int i进行“++”操作,需要将i的值拷贝到寄存器中,将该值自加后再拷贝到原来的内存。
如果此时有两个线程均进行的是这样的操作,可能出现两个线程都拷贝了i原来的值到寄存器,然后各种加一,再拷贝到i对应内存的情况,最终导致i这个变量只自加了一次。
(个人理解:写数据的过程为:
- 把内存中的i拷贝进寄存器;
- i++;
- 读取寄存器中的i到内存;
两个线程同时写产生的问题:
- A线程把i拷贝进寄存器,i++;
- 此时B线程又把内存中的i拷贝进了寄存器,覆写了原本寄存器中已经+1的i,B线程执行i++操作;
- A、B线程分别读取寄存器中的i;
此时的i就只自加了一次。
)
线程的同步与互斥的三种机制
- 互斥量
- 信号量
- 条件变量
3.1 互斥量
(1)原子性
如果一个线程锁定了一个互斥量,那么临界区内的操作要么全部完成,要么一个也不执行。
(2)唯一性

需要注意的问题:有可能上锁后忘记解锁;或者在解锁语句前程序抛出异常,导致无法解锁。

在c++中,lock()之后容易忘记unlock(),与忘记释放指针所指堆内存空间导致内存泄露的情况类似。于是与智能指针类似,也有了lock_guard,用来防止开发人员忘了解锁。
lock_guard:

(3)非繁忙等待

3.2 互斥量使用
互斥量使用过程如下:
(1)互斥量声明与初始化

(2)互斥量加锁
C语言中,通过系统调用pthread_mutex_lock对互斥量加锁。

(3)互斥量判断是否加锁
pthread_mutex_trylock()用于判断线程是否加锁,语义与pthread_mutex_lock类似,不同的是在锁被占用时返回EBUSY错误而不是挂起等待。
(4)访问共享资源
(5)互斥量解锁
pthread_mutex_unlock调用对指定的互斥量解锁。

(6)互斥量销毁
pthread_mutex_destroy()用于注销一个互斥量。

3.3 信号量
信号量可以实现线程的同步与互斥(设置不同的初始值,例如同步设为0,互斥设为1),其本质就是P/V操作,
也就是wait(S)和signal(S)这两个原语。
1. 记录型信号量
value:代表资源数目的整型变量;
L:进程链表,用于链接所有等待该资源的进程。
记录型信号量可描述为:
typedef struct{
int value;
struct process *L;
}semaphore;
相应的wait(S)和signal(S)操作如下:
void wait(semaphore S) //相当于申请资源
{
S.value--;
if(S.value < 0)
{
add this process to S.L;
block(S.L); //block原语,进行自我阻塞,放弃处理机
}
}
void signal(semaphore S) //相当于释放资源
{
S.value++; //释放一个进程,使可供分配的资源数+1
if(S.value <= 0) //若+1后仍是S.value <= 0,则表示在S.L中仍有等待该资源的进程被阻塞
{
remove a process P from S.L;
wakeup(P); //wakeup原语,唤醒S.L中的第一个等待进程
}
}
2. 利用信号量实现同步
设S为进程P1,P2的公共信号量,初值为0。
进程P2中的y语句要使用P1中x语句的运行结果,所以只有当x执行完毕后y才可以执行。
实现进程同步的算法如下:
semaphore S = 0; //初始化信号量
P1()
{
x; //语句x
V(S); //告诉进程P2,语句x已经完成
...
}
P2()
{
...
P(S); //检查语句x是否执行完成
y; //检查无误,运行y语句
...
}
3. 利用信号量实现进程互斥
设S为进程P1,P2的公共信号量,初值为1。
实现进程互斥的算法如下:
semaphore S = 1; //初始化信号量
P1()
{
...
P(S); //准备开始访问临界区,加锁
进程P1的临界区;
V(S); //访问结束,解锁
...
}
P2()
{
...
P(S); //准备开始访问临界区,加锁
进程P2的临界区;
V(S); //访问结束,解锁
...
}
3.4 信号量的使用方法

(1)信号量初始化


(2)信号量的P操作
如果信号量的value值为0,则阻塞当前线程;若不为0,则将value减1。

(3)信号量的V操作

(4)信号量的销毁

(5)信号量的其它调用

Linux多线程(8.3 线程同步与互斥)的更多相关文章
- .NET面试题解析(07)-多线程编程与线程同步
系列文章目录地址: .NET面试题解析(00)-开篇来谈谈面试 & 系列文章索引 关于线程的知识点其实是很多的,比如多线程编程.线程上下文.异步编程.线程同步构造.GUI的跨线程访问等等, ...
- .NET面试题解析(07)-多线程编程与线程同步 (转)
http://www.cnblogs.com/anding/p/5301754.html 系列文章目录地址: .NET面试题解析(00)-开篇来谈谈面试 & 系列文章索引 关于线程的知识点其实 ...
- 线程同步 - POSIX互斥锁
线程同步 - POSIX互斥锁 概括 本文讲解POSIX中互斥量的基本用法,从而能达到简单的线程同步.互斥量是一种特殊的变量,它有两种状态:锁定以及解锁.如果互斥量是锁定的,就有一个特定的线程持有或者 ...
- java核心知识点学习----多线程并发之线程同步
1.什么是线程同步? 多线程编程是很有趣的事情,它很容易出现"错误情况",这种情况不是由编码造成的,它是由系统的线程调度造成的,当使用多个线程来访问同一个数据时,很容易出现&quo ...
- linux线程同步(1)-互斥量
一.概述 互斥量是线程同步的一种机制,用来保护多线程的共享资源.同一时刻,只允许一个线程对临界区进行 ...
- Android多线程研究(3)——线程同步和互斥及死锁
为什么会有线程同步的概念呢?为什么要同步?什么是线程同步?先看一段代码: package com.maso.test; public class ThreadTest2 implements Runn ...
- 细说C#多线程那些事 - 线程同步和多线程优先级
上个文章分享了一些多线程的一些基础的知识,今天我们继续学习. 一.Task类 上次我们说了线程池,线程池的QueueUserWorkItem()方法发起一次异步的线程执行很简单 但是该方法最大的问题是 ...
- Win32多线程编程(3) — 线程同步与通信
一.线程间数据通信 系统从进程的地址空间中分配内存给线程栈使用.新线程与创建它的线程在相同的进程上下文中运行.因此,新线程可以访问进程内核对象的所有句柄.进程中的所有内存以及同一个进程中其他所有线 ...
- UNIX环境高级编程——线程同步之互斥锁、读写锁和条件变量(小结)
一.使用互斥锁 1.初始化互斥量 pthread_mutex_t mutex =PTHREAD_MUTEX_INITIALIZER;//静态初始化互斥量 int pthread_mutex_init( ...
- Linux多线程实践(1) --线程理论
线程概念 在一个程序里的一个执行路线就叫做线程(thread).更准确的定义是:线程是"一个进程内部的控制序列/指令序列"; 一切进程至少有一个执行线程; 进程 VS. 线程 ...
随机推荐
- 保持唯一性,请停止使用【python3 内置hash() 函数】
问题: 如图,用hash() 筛重时竟然出现了重复. 如下图: hash字符串时,同一窗口的是一致的,不同窗口结果竟然不同. 原因: python的字符串hash算法并不是直接遍历字符串每个字符去计算 ...
- 统计数据异常值的处理——R语言
在数据分析工作中,面对收集而来的数据,数据清洗是首要环节.异常值(outlier)是数据清洗的重要环节,异常值可能直接会导致后面的数据分析.建模工作出现偏差,下面就给大家介绍一下如何处理数据中的异常值 ...
- PyCharm解决Git冲突
技术背景 在前面的一篇博客中,我们介绍了Fork到自己名下的本地仓库如何与远程原始仓库创建链接的方法.在这篇文章中,我们将要讲解如何应对在这种异步开发的过程中经常有可能会遇到的Git冲突问题,在Pyc ...
- 靶机渗透【billu_b0x】
ip扫描 访问80端口 目录扫描 逐个访问 上传一个图片马,结果没有回显 显示file参数为空.请在"文件"参数中提供文件路径 打开发现有用户名 ![] 发现数据库连接的配置信息, ...
- TF-IDF定义及实现
TF-IDF定义及实现 定义 TF-IDF的英文全称是:Term Frequency - Inverse Document Frequency,中文名称词频-逆文档频率,常用于文本挖掘,资讯检索等 ...
- Redis的缓存穿透+解决方案
1.缓存穿透现象介绍 缓存穿透 :缓存穿透是指客户端请求的数据在缓存中和数据库中都不存在,这样缓存永远不会生效,这些请求都会打到数据库. 常见的解决方案有两种: 缓存空对象 优点:实现简单,维护方便 ...
- P4055 [JSOI2009] 游戏(博弈论 、最大匹配)
P4055 [JSOI2009] 游戏 目录 P4055 [JSOI2009] 游戏 题目传送门 题目大意 : 思路: code 题目传送门 题目大意 : 小AA和小YY玩游戏,在这个游戏中,同一个格 ...
- [C++基础入门] 2、数据类型
文章目录 2 数据类型 2.1 整型 2.2 sizeof关键字 2.3 实型(浮点型) 2.4 字符型 2.5 转义字符 2.6 字符串型 2.7 布尔类型 bool 2.8 数据的输入 2 数据类 ...
- 《C和指针》第一章
1 第一章 C标准库中几个常用的IO函数 int puts(void *str): 从str中提取字符直到遇到第一个'\0'为止,将这些字符串加上'\n'后发送给stdout. int main(vo ...
- Grafana 系列-统一展示-3-Prometheus 仪表板
系列文章 Grafana 系列文章 知识储备 Prometheus Template Variables 你可以使用变量来代替硬编码的细节,如 server.app 和 pod_name 在 metr ...