这一章,我们要来验证volatile关键字不是原子性的,OK,还是用代码来说话。

①.线程类,操作i++ 500次

  1. package com.multiThread.thread;
  2. publicclassNumberThreadimplementsRunnable{
  3. privatevolatileint num =0;
  4. @Override
  5. publicvoid run(){
  6. for(int i =0;i<500;i++){
  7. num++;
  8. }
  9. }
  10. publicint getNum(){
  11. return num;
  12. }
  13. publicvoid setNum(int num){
  14. this.num = num;
  15. }
  16. }
②.测试类,创建5个线程同时执行
  1. package com.multiThread.test.automic;
  2. import com.multiThread.thread.NumberThread;
  3. publicclassAutomicTest{
  4. publicstaticvoid main(String[] args){
  5. NumberThread numThread =newNumberThread();
  6. for(int i =0;i<5;i++){
  7. Thread t =newThread(numThread);
  8. t.start();
  9. }
  10. try{
  11. Thread.sleep(1000);
  12. }catch(InterruptedException e){
  13. e.printStackTrace();
  14. }
  15. System.out.println(numThread.getNum());
  16. }
  17. }
多次运行结果:(正常结果2500居多)
  1. 2500
  2. 2282
  3. 2458
 
为什么造成这种现象?
首先我们要知道i++不是原子操作,是线程不安全的,它分为以下3步:
1.获取i的值
2.执行i+1的操作
3.将结果赋值给i
其次就算变量已经使用volatile关键字来修饰,只能保证读取全局变量num值的时候从主存拿到的是最新的值。
但是当多个线程同时操作自加的时候,如果之前取到的是同一个值,再自加后得到的值是相同的。
 
解决的方式如下两种:
1.可以使用synchronize关键字进行同步操作,这种情况需要把volatile去掉。因为synchronize本身就会操作工作内存和主内存直接的数据同步,此方式不再赘述。
2.将i++更改为线程安全的原子性操作,使用AtomicInteger替代i++
 
①.线程类
  1. package com.multiThread.thread;
  2. import java.util.concurrent.atomic.AtomicInteger;
  3. publicclassNumberThread2implementsRunnable{
  4. privateAtomicInteger num =newAtomicInteger();
  5. @Override
  6. publicvoid run(){
  7. for(int i =0;i<500;i++){
  8. num.incrementAndGet();
  9. }
  10. }
  11. publicAtomicInteger getNum(){
  12. return num;
  13. }
  14. publicvoid setNum(AtomicInteger num){
  15. this.num = num;
  16. }
  17. }
②.测试类
  1. package com.multiThread.test.automic;
  2. import com.multiThread.thread.NumberThread;
  3. import com.multiThread.thread.NumberThread2;
  4. publicclassAutomicTest{
  5. publicstaticvoid main(String[] args){
  6. NumberThread2 numThread2 =newNumberThread2();
  7. for(int i =0;i<5;i++){
  8. Thread t =newThread(numThread2);
  9. t.start();
  10. }
  11. try{
  12. Thread.sleep(1000);
  13. }catch(InterruptedException e){
  14. e.printStackTrace();
  15. }
  16. System.out.println(numThread2.getNum());
  17. }
  18. }
运行结果
  1. 2500
 
 
 
 
 
 
 

多线程(四)~数据操作的原子性,使用原子性操作AutomicInteger替换非原子性的i++的操作的更多相关文章

  1. Java多线程学习---------超详细总结(java 多线程 同步 数据传递 )

    目录(?)[-] 一扩展javalangThread类 二实现javalangRunnable接口 三Thread和Runnable的区别 四线程状态转换 五线程调度 六常用函数说明 使用方式 为什么 ...

  2. volatile非原子性的示例

    volatile非原子性的示例 package com.stono.thread2.page124; public class MyThread extends Thread { volatile p ...

  3. volatile非原子性示例

    volatile非原子性示例 学习了:<Java多线程编程核心技术>高洪岩 著 Page124 package com.stono.thread2.page124_2; public cl ...

  4. JAVA笔记14__多线程共享数据(同步)/ 线程死锁 / 生产者与消费者应用案例 / 线程池

    /** * 多线程共享数据 * 线程同步:多个线程在同一个时间段只能有一个线程执行其指定代码,其他线程要等待此线程完成之后才可以继续执行. * 多线程共享数据的安全问题,使用同步解决. * 线程同步两 ...

  5. iOS开发——实战篇Swift篇&UItableView结合网络请求,多线程,数据解析,MVC实战

    UItableView结合网络请求,多线程,数据解析,MVC实战 学了这么久的swift都没有做过什么东西,今天就以自己的一个小小的联系,讲一下,怎么使用swift在实战中应用MVC,并且结合后面的高 ...

  6. java 多线程四

    java 多线程一 java 多线程二 java 多线程三 java 多线程四 一个生产者,消费者的例子: import java.util.Stack; /** * Created by root ...

  7. 原生js实现类的添加和删除,以及对数据的add和update、view ,ajax请求 ,页面离开的操作

    1 类操作 function hasClass(cla, element) { if(element.className.trim().length === 0) return false; var ...

  8. Unix环境高级编程(四)数据系统文件和信息

    本章主要介绍了Unix系统的正常运行要使用的与系统有关的数据文件和信息.如:口令文件,阴影文件.组文件.附加组.系统标识.时间和日期历程. 口令文件,即Unix系统用户数据库,存储在/etc/pass ...

  9. Java笔记16:多线程共享数据

    一.Thread实现 public class ThreadDemo4 { publicstaticvoid main(String[] args) { new ThreadTest4().start ...

随机推荐

  1. Olesya and Rodion (思维)

    Olesya loves numbers consisting of n digits, and Rodion only likes numbers that are divisible by t. ...

  2. 高精度的N进制转换模板(转K神)

    /* 高精度进制转换 把oldBase 进制的数转化为newBase 进制的数输出. 调用方法,输入str, oldBase newBase. change(); solve(); output(); ...

  3. scrapy框架初识

    一.scrapy简介 Scrapy是一个为了爬取网站数据,提取结构性数据而编写的应用框架,非常出名,非常强悍.所谓的框架就是一个已经被集成了各种功能(高性能异步下载,队列,分布式,解析,持久化等)的具 ...

  4. day_01 python基础 基本数据类型 if条件

    1. python简介 解释型,弱类型,高级开发语言2. 第一个python程序编写   1. 标点符号(英文)   2. 编码格式(utf-8)3.变量   把程序运行产生的中间值储存起来,方便后面 ...

  5. (转载) win10生成SSH keys

    (转载) win10生成 SSH keys:   SSH key 可以让你在你的电脑和Code服务器之间建立安全的加密连接.  先执行以下语句来判断是否已经存在本地公钥: cat ~/.ssh/id_ ...

  6. mapper mysl实现批量插入 更新

    1.批量插入 <insert id="insertConfirm" parameterType="java.util.List"> insert i ...

  7. ListView与ArrayAdapter的搭配使用

    在android中,ListView是一种很重要的控件,一般的使用中,常建立一个所需类型的ArrayList,再通过ArrayAdapter把ListView绑定到ArrayList上,通过Array ...

  8. Java关键字final、static使用总结 (final static在容器中不可以改变容器但可以改变存放)

    一.final        根据程序上下文环境,Java关键字final有“这是无法改变的”或者“终态的”含义,它可以修饰非抽象类.非抽象类成员方法和变量.你可能出于两种理解而需要阻止改变:设计或效 ...

  9. mac上k8s学习踩坑

    本文学习k8s参考内容:http://docs.kubernetes.org.cn/126.html,学习过程中遇到一些坑,记录如下: -------------------------------- ...

  10. Eclipse error: “The import XXX cannot be resolved”

    解决 Eclipse error: “The import XXX cannot be resolved” eclipse中修改: 1. 项目-->Properties-->java bu ...