Java多线程:线程间通信之volatile与sychronized这篇文章中我们了解了synchronized的基本特性,知道了一旦有一个线程访问某个对象的synchronized修饰的方法或代码区域时,该线程则获取这个对象的锁,其他线程不能再调用该对象被synchronized影响的任何方法。那么,如果这个线程自己调用该对象的其他synchronized方法,Java是如何判定的?这就涉及到了Java中锁的重要特性:可重入性,也就是今天的主题。

1. 线程安全与可重入性

在回答引言的问题前,我们先讲解一下可重入性。在线程这块知识中,可重入性常常和线程安全进行对比。

1.1. 线程安全

线程安全函数的概念比较直观,众所周知,同一进程的不同线程会共享同一主内存,线程的私有栈中只包括PC,栈,操作数栈,局部变量数组和动态链接。对共享内存进行读写时,若要保证线程安全,则必须通过加锁的方式。

1.2. 可重入

1.2.1. 定义

关于可重入这一概念,我们需要参考维基百科。

若一个程序或子程序可以“在任意时刻被中断然后操作系统调度执行另外一段代码,这段代码又调用了该子程序不会出错”,则称其为可重入(reentrant或re-entrant)的。即当该子程序正在运行时,执行线程可以再次进入并执行它,仍然获得符合设计时预期的结果。与多线程并发执行的线程安全不同,可重入强调对单个线程执行时重新进入同一个子程序仍然是安全的。

1.2.2. 可重入的条件

  • 不在函数内使用静态或全局数据。
  • 不返回静态或全局数据,所有数据都由函数的调用者提供。
  • 使用本地数据(工作内存),或者通过制作全局数据的本地拷贝来保护全局数据。
  • 不调用不可重入函数。

1.3. 可重入与线程安全

一般而言,可重入的函数一定是线程安全的,反之则不一定成立。在不加锁的前提下,如果一个函数用到了全局或静态变量,那么它不是线程安全的,也不是可重入的。如果我们加以改进,对全局变量的访问加锁,此时它是线程安全的但不是可重入的,因为通常的枷锁方式是针对不同线程的访问(如Java的synchronized),当同一个线程多次访问就会出现问题。只有当函数满足可重入的四条条件时,才是可重入的。

2. synchronized的可重入性

2.1. synchronized是可重入锁

回到引言里的问题,如果一个获取锁的线程调用其它的synchronized修饰的方法,会发生什么?

从设计上讲,当一个线程请求一个由其他线程持有的对象锁时,该线程会阻塞。当线程请求自己持有的对象锁时,如果该线程是重入锁,请求就会成功,否则阻塞。

我们回来看synchronized,synchronized拥有强制原子性的内部锁机制,是一个可重入锁。因此,在一个线程使用synchronized方法时调用该对象另一个synchronized方法,即一个线程得到一个对象锁后再次请求该对象锁,是永远可以拿到锁的

在Java内部,同一个线程调用自己类中其他synchronized方法/块时不会阻碍该线程的执行,同一个线程对同一个对象锁是可重入的,同一个线程可以获取同一把锁多次,也就是可以多次重入。原因是Java中线程获得对象锁的操作是以线程为单位的,而不是以调用为单位的。

2.2. synchronized可重入锁的实现

之前谈到过,每个锁关联一个线程持有者和一个计数器。当计数器为0时表示该锁没有被任何线程持有,那么任何线程都都可能获得该锁而调用相应方法。当一个线程请求成功后,JVM会记下持有锁的线程,并将计数器计为1。此时其他线程请求该锁,则必须等待。而该持有锁的线程如果再次请求这个锁,就可以再次拿到这个锁,同时计数器会递增。当线程退出一个synchronized方法/块时,计数器会递减,如果计数器为0则释放该锁。

3. 参考文章

可重入函数与线程安全函数

可重入

java内置锁synchronized的可重入性

最新文章

  1. 注意 AppResLib.dll.*.mui 的生成操作应该为 Content
  2. Red Gate(SQLToolbelt)SQL Server的安装与注册(破解)
  3. <Oracle Database>诊断文件
  4. javascript技术大全
  5. 如何让vim编辑器永久显示行号
  6. 事务四大特征:原子性,一致性,隔离性和持久性(ACID)
  7. Dell vsotro 14 3000系列从win10重装win7
  8. SQL Server中timestamp(时间戳)
  9. WPF-控件-ListView
  10. SWFUpload 中文乱码问题
  11. Java---网络蜘蛛-网页邮箱抓取器~源码
  12. MAC上使用maven打android的包,报错:No Android SDK path could be found. 解决办法
  13. Qt之QComboBox定制(二)
  14. AI五子棋需求规格说明书
  15. HotSpot 的垃圾收集器
  16. js截取url地址后面的文件名
  17. kolla单节点部署openstack
  18. uva-704-暴力枚举-双向bfs
  19. [leetcode]Permutations @ Python
  20. leetcode404

热门文章

  1. Android进阶(十四)Android Adapter详解
  2. python检测变量是否有定义(即使用前检查是否定义好)
  3. Linux环境编程导引
  4. SpriteBuilder中粒子发射器的reset on visibility toggle选项解释
  5. [FreeRadius2]遇到问题记录
  6. 【面试笔试算法】牛客网一站通Offer编程题2016.4.19
  7. 怎样使用projectproperty sheet(.vsprops)来管理工程
  8. 股票K线图
  9. 【linux学习笔记之一】linux系统目录结构以及常用系统命令
  10. 新版MATERIAL DESIGN 官方动效指南(一)