Android中线程同步之Mutex与Condtion的用法

2024-02-18 22:08

本文主要是介绍Android中线程同步之Mutex与Condtion的用法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Android封装的同步类主要有MUTEX(AutoMutex)与Condition。本文分析如何使用他们,具体的实现参见《 Android中线程同步之MUTEX与Condtion的实现 》与《 pthread之MUTEX与Condition用法 》。
Keywords: Mutex, Condition Variable, AutoLock/AutoMutex, Android
关键字:互斥体、条件变量、自动锁/自动互斥体、Android

一、MUTEX


看MUTEX的外部接口定义(声明在frameworks/native/include/utils/Mutex.h中)
<图1 TODO>

1.1 构造/析构

[cpp]  view plain copy
  1. Mutex();  
  2. Mutex(const char* name);  
  3. Mutex(int type, const char* name = NULL);  
  4. ~Mutex();  

构造函数的区别在于参数:
- name 是为MUTEX指定名字,如不指定,缺省是NULL;
- type 是指定MUTEX的类型,有
[cpp]  view plain copy
  1. enum {  
  2.     PRIVATE = 0,  
  3.     SHARED = 1  
  4. };  
两种类型:PRIVATE是进程内部使用的;SHARED是适用于跨进程共享的。
如不指定,缺省是PRIVATE的类型。

1.2 功能函数

MUTEX有下面三个主要的功能函数:
[cpp]  view plain copy
  1. status_t    lock();  
  2. void        unlock();  
  3. status_t    tryLock();  

lock()    获取锁。如果获取就返回,否则挂起等待;
unlock()  释放锁;
tryLock() 如果当前锁可被获取(未被别的线程获取)就lock,否则也直接返回。返回值:0代表成功;其它值失败。与lock()的区别在于不论成功与否都会及时返回,而不是挂起等待。

线程在进入MUTEX保护的临界区之前通过lock()获取锁,获取锁之后可以执行临界区内的代码,退出临界区之后通过unlock释放锁。
某一时刻临界区内至多只有一个线程在执行,如果已有线程T1在执行,要进入临界区的其它线程T2在执行lock()之后就会被挂起等待;直到线程T1释放掉锁之后,线程T2才能获得锁进入临界区执行。

lock()/unlock()必须配合使用,tryLock()则要根据执行的结果有无获得锁而选择是否unlock()。

1.3 典型场景

[cpp]  view plain copy
  1. m_mutex.lock();  
  2. // CRITICAL AREA,需要保护的内容  
  3. m_mutex.unlock();  

二、Condition

看Condition的外部接口定义(声明在frameworks/native/include/utils/Condition.h中)
<图2 TODO>

2.1 构造/析构

[cpp]  view plain copy
  1. Condition();  
  2. Condition(int type);  
  3. ~Condition();  

构造函数的区别在于type参数:
- type 是指定Condition的类型,有
[cpp]  view plain copy
  1. enum {  
  2.     PRIVATE = 0,  
  3.     SHARED = 1  
  4. };  
两种类型:PRIVATE是进程内部使用的;SHARED是适用于跨进程共享的。
如不指定,缺省是PRIVATE的类型。

2.2 功能函数

Condition有下面四个主要的功能函数:
[cpp]  view plain copy
  1. status_t wait(Mutex& mutex);  
  2. status_t waitRelative(Mutex& mutex, nsecs_t reltime);  
  3. void signal();  
  4. void broadcast();  

wait() 等待条件变量(Condition Variable)
    Mutex作为参数,调用该函数之前该MUTEX必须已经被lock住。
    执行该函数,会unlock该MUTEX,并等待条件变量。如果不能获得该条件变量,就被挂起等待;获得了该条件变量,就重新lock住MUTEX并返回。而这些操作都是原子操作的。
    该函数执行之后,Mutex被重新lock住,所以执行函数之后,必须有Mutex的unlock操作。
waitRelative()与和wait()的区别是,会有一个等待超时时间,到了时间没有获得该条件变量也会返回,可通过返回值判断结果。
signal()和broadcast() 触发条件变量(Condition Variable)
    signal()和broadcast() 的区别是,signal()只允许等待该条件变量的一个线程获得;broadcast()允许等待该条件变量的所有线程获得并继续执行。
    signal()和broadcast() 执行之前也必须lock住Mutex,执行之后unlock Mutex。

2.3 典型场景

线程T1的执行:
[cpp]  view plain copy
  1. m_mutex.lock();     
  2. m_cond.wait(m_mutex);     
  3. m_mutex.unlock();  
线程T1通过条件变量在等待某个条件的满足。

线程T2执行:
[cpp]  view plain copy
  1. m_mutex.lock();     
  2. m_cond.signal();     
  3. m_mutex.unlock();  
线程T2在满足条件之后,通过条件变量通知条件满足。

三、Autolock/AutoMutex

Autolock是为了简化Mutex的使用而定义的,它也定义在frameworks/native/include/utils/Mutex.h中,封装了Mutex,并利用c++的构造与析构机制。
[cpp]  view plain copy
  1. Autolock(Mutex& mutex);  
  2. Autolock(Mutex* mutex);  
  3. ~Autolock();  

用法很简单,定义一个局部临时的AutoMutex变量,在该变量定义的地方,构造函数被自动调用,会执行Mutex的lock()操作;在该变量作用域结束的地方,析构函数会被自动调用,会执行Mutex的unlock操作。
所以,可以在需要Mutex保护的区域开始的地方定义一个AutoMutex变量即可,即可实现用Mutex对该区域的保护。

四、总结

本文简要介绍了Android中常用的同步机制Mutex(AutoMutex)、Condition的使用。后续文章(《 Android中线程同步之MUTEX与Condtion的实现 》与《 pthread之MUTEX与Condition用法 》)看它们是如何实现的。

这篇关于Android中线程同步之Mutex与Condtion的用法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/722609

相关文章

golang panic 函数用法示例详解

《golangpanic函数用法示例详解》在Go语言中,panic用于触发不可恢复的错误,终止函数执行并逐层向上触发defer,最终若未被recover捕获,程序会崩溃,recover用于在def... 目录1. panic 的作用2. 基本用法3. recover 的使用规则4. 错误处理建议5. 常见错

前端知识点之Javascript选择输入框confirm用法

《前端知识点之Javascript选择输入框confirm用法》:本文主要介绍JavaScript中的confirm方法的基本用法、功能特点、注意事项及常见用途,文中通过代码介绍的非常详细,对大家... 目录1. 基本用法2. 功能特点①阻塞行为:confirm 对话框会阻塞脚本的执行,直到用户作出选择。②

Android开发中gradle下载缓慢的问题级解决方法

《Android开发中gradle下载缓慢的问题级解决方法》本文介绍了解决Android开发中Gradle下载缓慢问题的几种方法,本文给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、网络环境优化二、Gradle版本与配置优化三、其他优化措施针对android开发中Gradle下载缓慢的问

Python中多线程和多进程的基本用法详解

《Python中多线程和多进程的基本用法详解》这篇文章介绍了Python中多线程和多进程的相关知识,包括并发编程的优势,多线程和多进程的概念、适用场景、示例代码,线程池和进程池的使用,以及如何选择合适... 目录引言一、并发编程的主要优势二、python的多线程(Threading)1. 什么是多线程?2.

Java中将异步调用转为同步的五种实现方法

《Java中将异步调用转为同步的五种实现方法》本文介绍了将异步调用转为同步阻塞模式的五种方法:wait/notify、ReentrantLock+Condition、Future、CountDownL... 目录异步与同步的核心区别方法一:使用wait/notify + synchronized代码示例关键

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

Python itertools中accumulate函数用法及使用运用详细讲解

《Pythonitertools中accumulate函数用法及使用运用详细讲解》:本文主要介绍Python的itertools库中的accumulate函数,该函数可以计算累积和或通过指定函数... 目录1.1前言:1.2定义:1.3衍生用法:1.3Leetcode的实际运用:总结 1.1前言:本文将详

Android里面的Service种类以及启动方式

《Android里面的Service种类以及启动方式》Android中的Service分为前台服务和后台服务,前台服务需要亮身份牌并显示通知,后台服务则有启动方式选择,包括startService和b... 目录一句话总结:一、Service 的两种类型:1. 前台服务(必须亮身份牌)2. 后台服务(偷偷干

MyBatis-Flex BaseMapper的接口基本用法小结

《MyBatis-FlexBaseMapper的接口基本用法小结》本文主要介绍了MyBatis-FlexBaseMapper的接口基本用法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具... 目录MyBATis-Flex简单介绍特性基础方法INSERT① insert② insertSelec