初识Volatile关键字

序言:*牢骚*

   相信大家也了解过OKhttp以及AsyncTask源码,他们在线程池以及一些共享变量的操作都使用了一个关键字--》Volatile,对此我也是一知半解,我们在之前所了解的它是一个对线程保持可见性的功能定义,却不能深刻了解其含义与使用场景。作为一直在公司忙于项目进度的我,也是甚是羞愧,所以今天也是参考了一些大神的博客了解了一下。

为什么要使用Volatile关键字?可能有人会说别人那么用,照搬喽,WTF,其实很简单就是为了保证并发编程的安全性,那问题又来了,怎么保证并发编程访问共享变量的安全性呢,那么我们就要从Java内存模型来了解。

Java内存模型:

1、所有的变量都存储在主内存中。

2、不同的线程都有自己的工作内存。

3、不同线程对变量的操作必须在工作内存中进行。

4、不同线程间无法访问对方的变量,所有线程间的数据传递必须在主内存中进行。


举个简单的小例子:

线程A

int x = 10;

x=12;

线程B

x++;

y=x;

大家认为如果多个线程操作共享变量x,线程A和B输出x和y的值是多少,对A输出的x=12,而B输出的y=10;那这是为什么呢?因为多线程操作x的时候A从主内存中Copy了x的值,然后执行赋值,然而这个时候并没有将数据写入到主内存中,B线程重新再主内存Copy了一份原始值接着进行赋值操作,在这里只有int x =10是具有原子性,就是基本数据直接赋值的属于原子性操作,变量赋值变量就不具备原子性。基于此种内存模型,便产生了多线程中的数据“脏读”问题。这也是著名的缓存一致性问题又称为共享变量。

如何保证共享变量安全性呢,这里就涉及到了并发编程的三大概念:原子性、有序性、可见性。

原子性:一个或多个操作,要么全部成功,要么全部失败,类似于我们数据库事务

可见性:多个线程访问一个共享变量,当一个线程访问这个变量时,其他的线程应该是立刻能得知这个变量最新修改的值。

      ①由于可见性对共享变量带来的数据安全问题,Java官方也推荐使用Volatile修饰共享变量,保证被修改的值能立即刷新到主内存。

      ☛ 需要注意的是:普通变量 无法保证可见性

有序性:

     指令重排序??什么鬼,不要慌,其实就是系统为了优化代码而进行的顺序排序

      定义:处理器为了提高程序运行效率,可能会对输入代码进行优化,它不保证程序中各个语句的执行先后顺序同代码中的顺序一致,但是他会保证最终执行结果和代码顺序执行的结果是一致的。

核心:处理器在进行重哦爱须是会考虑指令之间的数据依赖性。

指令重排序不会影响单个线程的执行,但是会影响并发线程执行的正确性。

并发线程必须保证原子性,可见性以及有序性。还要保证happens-before原则(先行发生原则)大家可以自己去了解一下,很简单的。

延伸:Volatile、Synchronized和ReentrantLock

Synchronized:同步代码块和同步方法《Synchronized的用法

ReentrantLock:是一个接口

viod lock():线程执行此方法,如果锁空闲,当前线程就会获取锁,反之就会禁用该线程一直处于等待状态,直到锁释放。

void tryLock():只是尝试获取锁,并不会导致当前线程被禁用。

void unLock():必须由持有者释放,反之如果线程不持有执行该方法可能导致异常。

概念:重入锁与公平锁

所谓的重入锁就是自己获取自己的锁以后再次获取自己内部的锁。

公平锁就是cpu在调度线程是在等待队列里随机挑选一个线程去执行,那么优先级低的可能就会一直无法获取就会发生饥饿现象。

ReentrantLock通过构造器传入true或者false区分公不公平

Synchronized与ReentrantLock区别:

Lock是一个接口,而Synchronized是Java关键字


②Synchronized发生异常时会自动释放线程占有锁不会死锁,而lock发生异常时,如果用户没有主动通过Unlock去释放锁,很可能造成死锁。

③lock可以知道有没有成功获取锁,而Synchronized不可以。

Java1.5时,Synchronized是性能低耗的,操作都需要转入内核去操作。

Java1.6时,Synchronized优化了大量的性能,而官方也提倡在Synchronized能实现需求下,优先使用Synchronized。

最后感谢本文博文作者Ruheng的文章你真的了解Volatile吗

大家可以参考一下,谢谢,轻喷

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Java8张图 11、字符串不变性 12、equals()方法、hashCode()方法的区别 13、...
    Miley_MOJIE阅读 3,759评论 0 11
  • volatile这个关键字可能很多朋友都听说过,或许也都用过。在Java 5之前,它是一个备受争议的关键字,因为在...
    抓兔子的猫阅读 366评论 4 5
  • 此文章出自:<a href="http://www.cnblogs.com/dolphin0520/p/39203...
    zlb阅读 650评论 0 6
  • OKHttp牛逼之处 1.支持HTTP2/SPDY黑科技 --->okHttp中分包就分为Http1 和Http2...
    河里的枇杷树阅读 242评论 0 0
  • 余禾来到余付东家里的时候,眼见着二大爷的脸刷的一下白了。她笑着跟二娘张娟说,来给三弟弟补补课。 张娟给余付东生了三...
    白小科阅读 245评论 0 1