Linux 自旋锁,互斥量(互斥锁),读写锁
自旋锁(Spin Lock)
自旋锁类似于互斥量,不过自旋锁不是通过休眠阻塞进程,而是在取得锁之前一直处于忙等待的阻塞状态。这个忙等的阻塞状态,也叫做自旋。
自旋锁通常作为底层原语实现其他类型的锁。
适用场景:
1)锁被持有的时间短,而且线程不希望在重新调度上花费太多的成本;
2)在非抢占式内核中,会阻塞中断,这样中断处理程序不会让系统陷入死锁状态。因为中断处理程序无法休眠,只能使用这种锁;
缺点:
1)线程自旋等待锁变成可用时,CPU不能做其他事情,会浪费CPU资源;
伪代码
S = 1
线程P:
// 进入区
while (S <= 0) ; // 自旋
S--; // P操作
... // 临界区
// 退出区
S++; // V操作
自旋锁接口
自旋锁接口与互斥量类似,容易相互替换。
#include <pthread.h>
int pthread_spin_init(pthread_spinlock_t *lock, int pshared);
int pthread_spin_destroy(pthread_spinlock_t *lock);
int pthread_spin_lock(pthread_spinlock_t *lock);
int pthread_spin_trylock(pthread_spinlock_t *lock);
int pthread_spin_unlock(pthread_spinlock_t *lock);
注意:
1)如果自旋锁当前在解锁状态,pthread_spin_lock不用自旋,就可以对它加锁;
2)如果自旋锁当前在加锁状态,再获得锁的结果是未定义的。如果调用pthread_spin_lock,会返回EDEADLK错误或其他错误,或者调用者可能会永久自旋。取决于具体实现。
3)试图对没有加锁的自旋锁解锁,结果也是未定义的。
示例
自旋锁使用
#include <pthread.h>
#include <stdio.h>
#define THREAD_NUM 100
pthread_spinlock_t  spinlock;
void *thread_main(void *arg)
{
    int id = (int)arg;
    pthread_spin_lock(&spinlock); // 获得锁
    printf("thread main %d get the lock begin\n", id);
    printf("thread main %d get the lock end\n", id);
    pthread_spin_unlock(&spinlock); // 释放锁
    return NULL;
}
int main()
{
    pthread_spin_init(&spinlock, 0); /* PTHREAD_PROCESS_PRIVATE == 0*/
    int x = PTHREAD_PROCESS_PRIVATE;
    printf("x = %d\n", x);
    int i;
    pthread_t tids[THREAD_NUM];
    for (i = 0; i < THREAD_NUM; i++) {
        pthread_create(&tids[i], NULL, thread_main, i); // 创建线程
    }
    for (i = 0; i < THREAD_NUM; i++) {
        pthread_join(tids[i], NULL); // 连接线程
    }
    return 0;
}
互斥量(互斥锁, Mutex)
互斥量(Mutex)通过休眠阻塞进程/线程,确保同一时间只有一个线程访问数据。休眠,也就意味着会放弃CPU资源。
加锁
对互斥量加锁后,任何其他试图再次对互斥量加锁的线程,都会被阻塞,直到当前线程释放该互斥锁。
解锁
如果阻塞在该互斥锁上的线程有多个,当锁可用时,所有线程都会变成可运行状态,第一个变为运行的线程,就可以对互斥量加锁,其他线程则再次等待锁而进入休眠。
适用场景
多线程或多进程运行环境,需要对临界区资源进行保护时。
缺点
1)使用不当,任意导致死锁;
2)无法表示临界区资源可用数量(由信号量解决);
接口
#include <pthread.h>
int pthread_mutex_destroy(pthread_mutex_t *mutex);
int pthread_mutex_init(pthread_mutex_t *restrict mutex, const pthread_mutexattr_t *restrict attr); // 函数方式初始化,attr是线程属性
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER; // 直接赋值方式初始化
int pthread_mutex_lock(pthread_mutex_t *mutex); // 加锁
int pthread_mutex_trylock(pthread_mutex_t *mutex); // 尝试加锁,不会阻塞
int pthread_mutex_unlock(pthread_mutex_t *mutex); // 解锁
使用示例
#define THREAD_NUM 100
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
void *thread_main(void *arg)
{
    int id = (int)arg;
    pthread_mutex_lock(&mutex);
    printf("thread main %d get the lock begin\n", id);
    printf("thread main %d get the lock end\n", id);
    pthread_mutex_unlock(&mutex);
    return NULL;
}
int main()
{
    int i;
    pthread_t tids[THREAD_NUM];
    for (i = 0; i < THREAD_NUM; i++) {
        pthread_create(&tids[i], NULL, thread_main, i);
    }
    for (i = 0; i < THREAD_NUM; i++) {
        pthread_join(tids[i], NULL);
    }
    return 0;
}
读写锁(Read-Write Lock)
读写锁类似于互斥量,不过读写锁允许更高的并行性。读写锁,也叫共享互斥锁(shared-exclusive lock)。
当读写锁以读模式锁住时,可以说成是以共享模式锁住的。当以写模式锁住时,可以说成是以互斥模式锁住的。
读写锁与互斥锁的区别
读写锁与互斥锁的区别在于:
互斥锁 要么是加锁状态,要么是不加锁状态,而且一次只有一个线程能取得锁、对其加锁;
读写锁 可以有3种状态:读模式加锁,写模式加锁,不加锁。一次只有一个线程能占有写模式的读写锁,不过多个线程可以同时占有读模式的读写锁。
1)当读写锁是写加锁状态时,在被解锁前,所有试图对其加锁的线程都会被阻塞。
2)当读写锁是读加锁状态时,在被解锁前,所有以读模式加锁的线程都可以得到访问权,以写模式加锁的线程会被阻塞。
简而言之,读写锁是读状态与读状态之间共享,与写状态之间互斥,写状态是与任何状态互斥。
互斥锁是只有加锁和解锁状态,加锁状态之间互斥。
适用场景
读写锁非常适合对数据结构进行读操作的次数 远大于写的情况。
使用接口
初始化销毁:
#include <pthread.h>
int pthread_rwlock_destroy(pthread_rwlock_t *rwlock);
int pthread_rwlock_init(pthread_rwlock_t *restrict rwlock,
      const pthread_rwlockattr_t *restrict attr);
pthread_rwlock_t rwlock = PTHREAD_RWLOCK_INITIALIZER; // 直接赋值方式初始化读写锁
attr = NULL,表示使用默认的读写锁属性。
读、写模式获得锁,解锁:
#include <pthread.h>
int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock); // 读模式取得锁
int pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock); // 读模式取得锁的条件版本
int pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock); // 写模式取得锁
int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock); // 写模式取得锁的条件版本
int pthread_rwlock_unlock(pthread_rwlock_t *rwlock); // 解锁
注意:
1)不论是处于写模式,还是读模式,都可以用pthread_rwlock_unlock解锁。
2)条件版本不会阻塞线程。
Linux 自旋锁,互斥量(互斥锁),读写锁的更多相关文章
- Linux的线程同步对象:互斥量Mutex,读写锁,条件变量
		
进程是Linux资源分配的对象,Linux会为进程分配虚拟内存(4G)和文件句柄等 资源,是一个静态的概念.线程是CPU调度的对象,是一个动态的概念.一个进程之中至少包含有一个或者多个线程.这 ...
 - pthread中互斥量,锁和条件变量
		
互斥量 #include <pthread.h> pthread_mutex_t mutex=PTHREAD_MUTEX_INTIIALIZER; int pthread_mutex_in ...
 - go Mutex (互斥锁)和RWMutex(读写锁)
		
转载自: https://blog.csdn.net/skh2015java/article/details/60334437 golang中sync包实现了两种锁Mutex (互斥锁)和RWMute ...
 - Linux环境编程之同步(三):读写锁
		
概述 相互排斥锁把试图进入我们称之为临界区的全部其它线程都堵塞住.该临界区通常涉及对由这些线程共享一个或多个数据的訪问或更新.读写锁在获取读写锁用于读某个数据和获取读写锁用于写直接作差别. 读写锁的分 ...
 - Linux 多线程互斥量互斥
		
同步 同一个进程中的多个线程共享所在进程的内存资源,当多个线程在同一时刻同时访问同一种共享资源时,需要相互协调,以避免出现数据的不一致和覆盖等问题,线程之间的协调和通信的就叫做线程的同步问题, 线程同 ...
 - 深刨显式锁ReentrantLock原理及其与内置锁的区别,以及读写锁ReentrantReadWriteLock使用场景
		
13.显示锁 在Java5.0之前,在协调对共享对象的访问时可以使用的机制只有synchronized和volatile.Java5.0增加了一种新的机制:ReentrantLock.与之前提到过的机 ...
 - Linux多线程实践(6) --Posix读写锁解决读者写者问题
		
Posix读写锁 int pthread_rwlock_init(pthread_rwlock_t *restrict rwlock, const pthread_rwlockattr_t *rest ...
 - UNIX环境高级编程——线程同步之读写锁以及属性
		
读写锁和互斥量(互斥锁)很类似,是另一种线程同步机制,但不属于POSIX标准,可以用来同步同一进程中的各个线程.当然如果一个读写锁存放在多个进程共享的某个内存区中,那么还可以用来进行进程间的同步, 互 ...
 - Java 中15种锁的介绍:公平锁,可重入锁,独享锁,互斥锁,乐观锁,分段锁,自旋锁等等
		
Java 中15种锁的介绍 Java 中15种锁的介绍:公平锁,可重入锁,独享锁,互斥锁,乐观锁,分段锁,自旋锁等等,在读很多并发文章中,会提及各种各样锁如公平锁,乐观锁等等,这篇文章介绍各种锁的分类 ...
 
随机推荐
- 为了让她学画画——熬夜用canvas实现了一个画板
			
前言 大家好,我是Fly, canvas真是个强大的东西,每天沉迷这个无法自拔, 可以做游戏,可以对图片处理,后面会给大家分享一篇,canvas实现两张图片找不同的功能, 听着是不是挺有意思的, 有点 ...
 - GeoServer Rest服务启动匿名认证的配置方法
			
GeoServer Rest服务数据默认需要进行用户名.密码的认证,如不需进行该认证,则启动匿名认证即可,配置方式如下(针对war包发布的GeoServer应用): 在GeoServer war包的解 ...
 - innodb引擎相关参数
			
1.innodb_flush_log_at_trx_commit (双一标准之一) 作用:主要控制了innodb将log buffer中的数据写入日志文件并flush磁盘的时间点,取值分别为0.1.2 ...
 - 【系统学习ES6】第一节:新的声明方式
			
[系统学习ES6] 本专题旨在对ES6的常用技术点进行系统性梳理,帮助大家对其有更好的掌握.计划每周更新1-2篇,希望大家有所收获. 以前用ES5时,声明变量只能用var.ES6的出现,为我们带来了两 ...
 - jvm源码解读--18 Java的start()方法解读 以及 wait 和notify流程图
			
drawwed by 张艳涛 and get info from openjdk8 还有一个图
 - Linux系统进入redis并查询值
			
1.进入redisredis-cli -h ip -p port2.查看具体信息info 3.得到redis中存储的所有key值KEYS *4.获取指定key值的value值get "key ...
 - 只是想虐下春丽,一不当心玩了下serverless...感觉还不错哟!
			
事情是这样的-- 前天下午天太热,我在家看电视,换台突然就看到了正在播<西游记>,窗外蝉声特别响,我一下就有种穿越回小学暑假的感觉.当时,我就特别想把我那台小霸王翻出来,玩两盘街霸--虐一 ...
 - 关于C语言中对数字的扩展和缩短
			
关于对数字的扩展:如果需要在不改变他的类型的情况下去扩展一个数字 有符号数字: 如果最高位为0---向左按位复制0 如果最高位为1---向左按位复制1 无符号数字:向左按位复制0即可 对于数字的缩短: ...
 - Java进阶 | 从整体上观察面向对象
			
一.面向对象 面向对象是Java编程中最核心的思想,基本特征:继承.封装.多态. 1.特征之封装 将结构.数据.操作封装在对象实体中,使用时可以不关注对象内部结构,只能访问开放权限的功能入口,从而降低 ...
 - 卷向字节码-Java异常到底是怎么被处理的?
			
你好呀,我是why,你也可以叫我歪歪. 比如下面这位读者: 他是看了我<神了!异常信息突然就没了?>这篇文章后产生的疑问. 既然是看了我的文章带来的进一步思考,恰巧呢,我又刚好知道. 虽然 ...