了解了并发编程中锁的基本原理之后,接下来看看Java是如何利用这些原理来实现各种锁,原子变量,同步组件的。在开始分析java.util.concurrent的源代码直接,首先要了解的就是sun.misc.Unsafe类,这个类可以说的java并发包的基础,基本上所有的组件都是依赖Unsafe来做底层的同步操作。

Unsafe类有100+个方法,大部分是native方法,可以理解为Java平台和底层操作系统直接的桥梁。它封装了大量的底层操作,比如直接操作内存的方法,低级同步的方法,CAS方法,直接操作Class的方法,直接操作Object的方法等等。有了Unsafe类,就可以像C, C++一样精确地操作内存。当然Java的一大优点就是可以安全的操作内存,所以不提倡开发者直接使用Unsafe类。JDK本身的类很多都利用了Unsafe来进行底层操作。

Unsafe和并发编程相关的有几类方法:

1. CAS方法

2. 操作条件队列的方法,比如park()让线程进入等待,unpark()唤醒线程

3. 存取volatile变量的方法,比如getBooleanVolatile, putBooleanVolatile

首先看看CAS方法,主要是3个comAndSwapXXX方法

1. compareAndSwapObject提供了对一个对象引用进行CAS的能力

2. compareAndSwapInt提供了对一个32位整数进行CAS操作的能力

3. compareAndSwapLong提供了对64位整数进行CAS操作的能力

关于CAS的概念请看这篇 聊聊高并发(十二)分析java.util.concurrent.atomic.AtomicStampedReference源码来看如何解决CAS的ABA问题

 
  1. public final class Unsafe{

  2. public final native boolean compareAndSwapObject(Object o, long offset,

  3. Object expected,

  4. Object x);

  5. /**

  6. * Atomically update Java variable to <tt>x</tt> if it is currently

  7. * holding <tt>expected</tt>.

  8. * @return <tt>true</tt> if successful

  9. */

  10. public final native boolean compareAndSwapInt(Object o, long offset,

  11. int expected,

  12. int x);

  13. /**

  14. * Atomically update Java variable to <tt>x</tt> if it is currently

  15. * holding <tt>expected</tt>.

  16. * @return <tt>true</tt> if successful

  17. */

  18. public final native boolean compareAndSwapLong(Object o, long offset,

  19. long expected,

  20. long x);

  21. ..............

  22. }

操作条件队列的方法有两个:

1. park(boolean isAbsolute, long time), 这个方法会让当前线程进入等待,并释放锁。Java并发包里的Condition接口的底层实现就是利用了Unsafe的park方法来实现的。第一个参数isAbsolute是表示用绝对时间还是相对事件,如果是绝对时间,就等待直到time,比如Condition接口的awaitUntil(Date deadline)。isAbsolute为false时,等待一个时间间隔

2. unpark(Object thread)唤醒等待的线程,Condition的signal方法就是基于unpark实现的

 
  1. /**

  2. * Unblock the given thread blocked on <tt>park</tt>, or, if it is

  3. * not blocked, cause the subsequent call to <tt>park</tt> not to

  4. * block. Note: this operation is "unsafe" solely because the

  5. * caller must somehow ensure that the thread has not been

  6. * destroyed. Nothing special is usually required to ensure this

  7. * when called from Java (in which there will ordinarily be a live

  8. * reference to the thread) but this is not nearly-automatically

  9. * so when calling from native code.

  10. * @param thread the thread to unpark.

  11. *

  12. */

  13. public native void unpark(Object thread);

  14. /**

  15. * Block current thread, returning when a balancing

  16. * <tt>unpark</tt> occurs, or a balancing <tt>unpark</tt> has

  17. * already occurred, or the thread is interrupted, or, if not

  18. * absolute and time is not zero, the given time nanoseconds have

  19. * elapsed, or if absolute, the given deadline in milliseconds

  20. * since Epoch has passed, or spuriously (i.e., returning for no

  21. * "reason"). Note: This operation is in the Unsafe class only

  22. * because <tt>unpark</tt> is, so it would be strange to place it

  23. * elsewhere.

  24. */

  25. public native void park(boolean isAbsolute, long time);

存取volatile变量的方法,这些方法让Unsafe对象有了直接存取volatile变量的能力。

 
  1. public native Object getObjectVolatile(Object obj, long l);

  2. public native void putObjectVolatile(Object obj, long l, Object obj1);

  3. public native int getIntVolatile(Object obj, long l);

  4. public native void putIntVolatile(Object obj, long l, int i);

  5. public native boolean getBooleanVolatile(Object obj, long l);

  6. public native void putBooleanVolatile(Object obj, long l, boolean flag);

  7. public native byte getByteVolatile(Object obj, long l);

  8. public native void putByteVolatile(Object obj, long l, byte byte0);

  9. public native short getShortVolatile(Object obj, long l);

  10. public native void putShortVolatile(Object obj, long l, short word0);

  11. public native char getCharVolatile(Object obj, long l);

  12. public native void putCharVolatile(Object obj, long l, char c);

  13. public native long getLongVolatile(Object obj, long l);

  14. public native void putLongVolatile(Object obj, long l, long l1);

  15.   public native float getFloatVolatile(Object obj, long l);

  16.     public native void putFloatVolatile(Object obj, long l, float f);

  17.     public native double getDoubleVolatile(Object obj, long l);

  18.     public native void putDoubleVolatile(Object obj, long l, double d);

关于Unsafe对象的其他信息,比如如何得到Unsafe对象,比如如何直接操作内存,类似反射机制存取对象属性,请查看这篇Java Magic 4. Part 4: sun.misc,Unsafe

java.util.concurrent包提供了一个LockSupport类来封装了Unsafe对象,来提供操作条件队列的方法。

来看一下LockSupport的实现,有几点比较有意思

1. 利用Unsafe直接操作内存来存取对象的能力来设置blocker

Unsafe.objectFieldOffset可以获得某个字段在对象所在内存的offset,有了这个offset,就可以通过对象的引用来找到字段所在的实际内存地址。这种做法在C++中常见,但是在Java中不推荐上层程序使用。

这段代码的意思是把arg对象设置到Thread的parkBlocker属性上。

Thread的parkBlocker属性用来指出当前线程是在哪个对象上阻塞

 
  1. public class LockSupport {

  2. private LockSupport() {} // Cannot be instantiated.

  3. // Hotspot implementation via intrinsics API

  4. private static final Unsafe unsafe = Unsafe.getUnsafe();

  5. private static final long parkBlockerOffset;

  6. static {

  7. try {

  8. parkBlockerOffset = unsafe.objectFieldOffset

  9. (java.lang.Thread.class.getDeclaredField("parkBlocker"));

  10. } catch (Exception ex) { throw new Error(ex); }

  11. }

  12. private static void setBlocker(Thread t, Object arg) {

  13. // Even though volatile, hotspot doesn't need a write barrier here.

  14. unsafe.putObject(t, parkBlockerOffset, arg);

  15. }

  16. public class Thread{

  17.  /**

  18.      * The argument supplied to the current call to

  19.      * java.util.concurrent.locks.LockSupport.park.

  20.      * Set by (private) java.util.concurrent.locks.LockSupport.setBlocker

  21.      * Accessed using java.util.concurrent.locks.LockSupport.getBlocker

  22.      */

  23.     volatile Object parkBlocker;

  24. }

2. park方法需要指明锁对象,可以看到,park方法先setBlocker标记当前线程是在哪个锁对象上等待,然后调用Unsafe的park方法,当Unsafe的park方法返回时表示已经退出等待,就把blocker设置为null.

用jstack命令查看过线程状态的同学肯定知道jstack能打印出线程是在哪个对象上block,这个对象就是这里的blocker

 
  1. public static void park(Object blocker) {

  2. Thread t = Thread.currentThread();

  3. setBlocker(t, blocker);

  4. unsafe.park(false, 0L);

  5. setBlocker(t, null);

  6. }

LockSupport的park()和unpark()方法和Object.wait(), notify方法都可以操作线程的等待和唤醒,但是两者主要有两个区别

1. 面向的主体不同,LockSupport的park, unpark面向的是线程,而Object.wait, nofify面向的是对象

2. 底层实现机制不同,可以看到Object的wait, notify方法也是native方法,Unsafe的park和unpark方法也是native方法,底层实现不同,Object.notify不能唤醒Unsafe park的线程。

 
  1. public class Object{

  2. public final native void wait(long timeout) throws InterruptedException;

  3. public final native void notify();

  4. public final native void notifyAll();

  5. }

聊聊高并发(十七)解析java.util.concurrent各个组件(一) 了解sun.misc.Unsafe类相关推荐

  1. 聊聊高并发(二十一)解析java.util.concurrent各个组件(三) 深入理解AQS(一)

    AQS是AbstractQueuedSynchronizer的缩写,AQS是Java并包里大部分同步器的基础构件,利用AQS可以很方便的创建锁和同步器.它封装了一个状态,提供了一系列的获取和释放操作, ...

  2. 聊聊高并发(二十九)解析java.util.concurrent各个组件(十一) 再看看ReentrantReadWriteLock可重入读-写锁

    上一篇聊聊高并发(二十八)解析java.util.concurrent各个组件(十) 理解ReentrantReadWriteLock可重入读-写锁 讲了可重入读写锁的基本情况和主要的方法,显示了如何 ...

  3. 聊聊高并发(二十五)解析java.util.concurrent各个组件(七) 理解Semaphore

    前几篇分析了一下AQS的原理和实现,这篇拿Semaphore信号量做例子看看AQS实际是如何使用的. Semaphore表示了一种可以同时有多个线程进入临界区的同步器,它维护了一个状态表示可用的票据, ...

  4. 聊聊高并发(二十二)解析java.util.concurrent各个组件(四) 深入理解AQS(二)

    上一篇介绍了AQS的基本设计思路以及两个内部类Node和ConditionObject的实现 聊聊高并发(二十一)解析java.util.concurrent各个组件(三) 深入理解AQS(一) 这篇 ...

  5. 聊聊高并发(二十九)解析java.util.concurrent各个组件(十一) 再看看ReentrantReadWriteLock可重入读-写锁...

    上一篇聊聊高并发(二十八)解析java.util.concurrent各个组件(十) 理解ReentrantReadWriteLock可重入读-写锁 讲了可重入读写锁的基本情况和基本的方法,显示了怎样 ...

  6. 聊聊高并发(二十)解析java.util.concurrent各个组件(二) 12个原子变量相关类

    这篇说说java.util.concurrent.atomic包里的类,总共12个,网上有很多文章解析这几个类,这里挑些重点说说. 这12个类可以分为三组: 1. 普通类型的原子变量 2. 数组类型的 ...

  7. 聊聊高并发(二十七)解析java.util.concurrent各个组件(九) 理解ReentrantLock可重入锁

    这篇讲讲ReentrantLock可重入锁,JUC里提供的可重入锁是基于AQS实现的阻塞式可重入锁.这篇 聊聊高并发(十六)实现一个简单的可重入锁 模拟了可重入锁的实现.可重入锁的特点是: 1. 是互 ...

  8. 聊聊高并发(二十八)解析java.util.concurrent各个组件(十) 理解ReentrantReadWriteLock可重入读-写锁

    这篇讲讲ReentrantReadWriteLock可重入读写锁,它不仅是读写锁的实现,并且支持可重入性. 聊聊高并发(十五)实现一个简单的读-写锁(共享-排他锁) 这篇讲了如何模拟一个读写锁. 可重 ...

  9. 聊聊高并发(三十一)解析java.util.concurrent各个组件(十三) 理解Exchanger交换器

    这篇讲讲Exchanger交互器,它是一种比较特殊的两方(Two-Party)栅栏,可以理解成Exchanger是一个栅栏,两边一方是生产者,一方是消费者, 1. 生产者和消费者各自维护了一个容器,生 ...

最新文章

  1. 常用Maven收集以及Maven技巧
  2. KDD 2017 《ReasoNet: Learning to Stop Reading in Machine Comprehension》论文笔记
  3. 【视频】v-html的使用
  4. 3分钟掌握Quartz.net分布式定时任务的姿势
  5. junit 测试 dao_JUnit测试Spring Service和DAO(带有内存数据库)
  6. Linux(15)-C/C++、PHP、JAVA概述
  7. footer.php置底,CSS五种方式实现Footer置底
  8. C# 让程序自动以管理员身份运行
  9. 如何终止Java线程
  10. 【USACO】sprime
  11. 正则表达式案例分析 (二)
  12. 是否可以加唯一约束 有空_一篇文章可以学会保温杯的三维设计吗?真可以
  13. 妄想山海测试服下载for android,妄想山海测试服
  14. 轮盘的基础,简单的实现和Toolbar的实现
  15. 关于outlook不能发送126邮件的问题
  16. 虚拟机安装ubuntu的相关经验总结及常见疑问
  17. Git 命令行其实真的很好用
  18. Windows自带截屏-快捷键截图区域至剪切板
  19. krpano 小行星开场和自动旋转
  20. ceph monitor paxos算法

热门文章

  1. flask执行python脚本_如何在Flask中运行python脚本
  2. java布局管理器的应用总结,GridBagLayout布局管理器的应用
  3. 山东大学linux实验四CSDN,山东大学操作系统实验报告材料4进程同步实验
  4. 程序员必备的 10 大 GitHub 仓库
  5. php php5,初探 PHP5 (一)_PHP
  6. vue项目没有router文件夹_vueRouter没有报错,但是页面渲染空白
  7. c语言简单的模拟坐标,C语言模拟实现简单扫雷游戏
  8. 二次探测再散列_杭州二次元影像测量仪
  9. c语言标准输入输出ppt,c语言输入输出语句PPT
  10. 大连开发区中老年运动微信群_消暑!大连近郊最受欢迎的海滨浴场集合来啦