- 1、本文档共4页,可阅读全部内容。
- 2、有哪些信誉好的足球投注网站(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
- 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载。
- 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
查看更多
Java可见性机制的原理
Java可见性机制的原理今天主要介绍Java可见性机制的原理。作者:佚名来源:/Linux/2016-11/136808.htmLinux社区|2016-11-11 00:39javascript:favorBox(open);?收藏 javascript:;?分享 基本概念1.可见性当一个线程修改了共享变量时,另一个线程可以读取到这个修改后的值。2.内存屏障(Memory Barriers)处理器的一组指令,用于实现对内存操作的顺序限制。3.缓冲行CPU告诉缓存中可以分配的最小存储单位,处理器填写缓存行时,会加载整个缓存行。4.Lock前缀的指令Lock前缀的指令在多核处理器下会发生两件事情:1)将当前处理器的缓存行的数据协会到系统内存。2)这个写回内存的操作会使其他CPU缓存了该内存的地址的数据无效。5.缓存一致性协议在多处理器下,为零保证各个处理器的缓存是一致的,每个处理器都会通过嗅探在总线上传播的数据来检查自己缓存的值是不是过期了。当处理器发现自己缓存行对应的地址被修改,就会将当前处理器的缓存行设置为无效状态。当处理器对这个数据进行读写的时候,会重新把数据从内存中读取到处理器缓存中。6.CASCompareAndSwap 比较并交换CAS操作需要输入两个值,一个旧值(执行CAS操作前的值,期望值)和一个新值,只有当当前值等于旧值时,才可以将当前值设置为新值,否则不设置。这是一个原子操作,由硬件保证。7.重排序规则从根本上来所,JMM 对编译器和处理器的重排序限制只有一条,只要不改变程序执行的结果(指的是单线程或者正确同步的多线程环境下),那么编译器和处理器怎么优化都可以。Volatile从上面的Lock前缀指令和缓存一致性协议可以看出来,这就是volatile的实现原理了。实际上,valatile变量被写入时,确实加了一个Lock前缀的指定,以此来达到可见性的目的。finalFinal域只能被显示地赋值一次,但是这并不代表final域不能被多次初始化。比如:final int i ;i在构造函数中被赋值之前,就会被初始化为默认的值:0.通过调试代码可以证明这一点。为了保证final域的值不会在为初始化的情况下被访问到,程序员只需要保证一点即可:即,在构造函数中,正在被构造的对象(this)没有“逸出”,那么不需要任何同步手段,就能保证任意线程看到的final域,包括基本类型和引用类型,都是已经被正确地通过构造函数初始化过了的。一个会是正在被构造的对象逸出的例子:public?class?FinalTest{??final?int?i;??static?FinalTest?obj;???public?FinalTest(){??i?=1;??/**??*这里会使正在被构造的对象逸出,如果和上一句做了重排序,那么其他线程就可以通过obj访问到还为被初始化的final域。??**/??obj?=?this;???}?}?Happens-Before规则happens-before的含义Happen-Before规则用来描述两个操作之间的顺序关系,这两个操作可以再一个线程内,也可以不再一个线程内。此顺序并不严格意味着执行时间上的顺序,而是至前一个操作的结果要对后一个操作可见。Happens-Before关系的定义如下:如果一个happens-before另一个操作,那么第一个操作的执行结果对第二个操作可见,而且第一个操作的执行顺序排在第二个操作之前两个操作之间存在happens-before关系,并不意味着Java平台的具体实现必须按照happens-before关系指定的顺序来执行。如果重排序之后的执行结果,与按照happens-before关系来执行的结果一致,那么这种重排序并不非法。举例来说,如果在程序执行顺序上,A先于B,并且A修改了共享变量,而B正好使用该共享变量,那么A需要happen-before B,再直白一点,就是A对共享变量的修改,需要在B执行时,对B可见。happens-before规则程序顺序规则:一个线程中的每个操作,happens-before于该线程中的任意后续操作。监视器锁规则:对一个锁的解锁,happens-before于随后对这个锁的加锁。volatile规则:对一个volatile域的写,happens-before于任意后续对这个volatile域的读。传递性:如果Ahappens-before B,并且B happens-before C,那么A happens-before C。start()规则:如果线程A执行操作ThreadB.start(),那么A线程的ThreadB.start()操作happens-before于线程B中的任意操作。join()规则:如果线程A执行操作Thr
文档评论(0)