- 1、有哪些信誉好的足球投注网站(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。。
- 2、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载。
- 3、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
- 4、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
- 5、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们。
- 6、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
- 7、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
Java线程安全问题.pdf
Java 线程安全问题 浅谈java 内存模型 不同的平台,内存模型是不一样的,但是jvm 的内存模型规范是统一的。其实java 的多线程并发问题最终都会反映在 java 的内存模型上,所谓线程安全无非是要控制多个线程对某个资源的有序访问或修改。总结java 的内存模型,要解决两个 主要的问题:可见性和有序性。我们都知道计算机有高速缓存的存在,处理器并不是每次处理数据都是取内存的。JVM 定义 了自己的内存模型,屏蔽了底层平台内存管理细节,对于java 开发人员,要清楚在jvm 内存模型的基础上,如果解决多线程 的可见性和有序性。 那么,何谓可见性? 多个线程之间是不能互相传递数据通信的,它们之间的沟通只能通过共享变量来进行。Java 内存模 型(JMM )规定了jvm 有主内存,主内存是多个线程共享的。当 new 一个对象的时候,也是被分配在主内存中,每个线程 都有自己的工作内存,工作内存存储了主存的某些对象的副本,当然线程的工作内存大小是有限制的。当线程操作某个对象 时,执行顺序如下: (1) 从主存复制变量到当前工作内存 (read and load) (2) 执行代码,改变共享变量值 (use and assign) (3) 用工作内存数据刷新主存相关内容 (store and write) JVM 规范定义了线程对主存的操作指令:read ,load ,use ,assign ,store ,write。当一个共享变量在多个线程的工作内 存中都有副本时,如果一个线程修改了这个共享变量,那么其他线程应该能够看到这个被修改后的值,这就是多线程的可见 性问题。 那么,什么是有序性呢 ?线程在引用变量时不能直接从主内存中引用,如果线程工作内存中没有该变量,则会从主内存中 拷贝一个副本到工作内存中,这个过程为 read-load,完成后线程会引用该副本。当同一线程再度引用该字段时,有可能重新从主 存中获取变量副本(read-load-use),也有可能直接引用原来的副本(use),也就是说 read,load,use 顺序可以由 JVM 实现系统 决定。 线程不能直接为主存中中字段赋值,它会将值指定给工作内存中的变量副本(assign),完成后这个变量副本会同步到主 存储区(store-write) ,至于何时同步过去,根据JVM 实现系统决定.有该字段,则会从主内存中将该字段赋值到工作内存中,这 个过程为 read-load,完成后线程会引用该变量副本,当同一线程多次重复对字段赋值时,比如: Java 代码 1. for(int i=0;i10;i++) 2. a++; 线程有可能只对工作内存中的副本进行赋值,只到最后一次赋值后才同步到主存储区,所以 assign,store,weite 顺序可以由 JVM 实现系统决定。假设有一个共享变量 x ,线程a 执行 x=x+1。从上面的描述中可以知道x=x+1 并不是一个原子操作, 它的执行过程如下: 1 从主存中读取变量 x 副本到工作内存 2 给 x 加 1 3 将 x 加 1 后的值写回主存 如果另外一个线程 b 执行 x=x-1 ,执行过程如下: 1 从主存中读取变量 x 副本到工作内存 2 给 x 减 1 3 将 x 减 1 后的值写回主存 那么显然,最终的 x 的值是不可靠的。假设x 现在为 10 ,线程a 加 1 ,线程b 减 1 ,从表面上看,似乎最终x 还是为 10 , 但是多线程情况下会有这种情况发生: 1 / 9 1 :线程a 从主存读取 x 副本到工作内存,工作内存中 x 值为 10 2 :线程b 从主存读取 x 副本到工作内存,工作内存中 x 值为 10 3 :线程a 将工作内存中 x 加 1 ,工作内存中x 值为 11 4 :线程a 将 x 提交主存中,主存中 x 为 11 5 :线程b 将工作内存中 x 值减 1 ,工作内存中x 值为 9 6 :线程b 将 x 提交到中主存中,主存中 x 为 9 同样,x 有可能为 11 ,如果x 是一个银行账户,线程 a 存款,线程 b 扣款,显然这样是有严重问题的,要解决这个问题,必 须保证线程 a 和线程 b 是有序执行的,并且每个线程执行的加 1 或减 1 是一个原子操作。看看下面代码: Java 代码 1. public class Account { 2. 3. pr
文档评论(0)