这一章,我们要来验证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. C. Magic Ship (思维+二分)

    https://codeforces.com/contest/1117/problem/C 你是一个船长.最初你在点 (x1,y1) (显然,大海上的所有点都可以用平面直角坐标描述),你想去点 (x2 ...

  2. mongodb的初步使用

    一.mongodb简介 MongoDB是一个基于分布式文件存储的数据库.由C++语言编写.旨在为WEB应用提供可扩展的高性能数据存储解决方案. MongoDB是一个介于关系数据库和非关系数据库之间的产 ...

  3. Scrapy错误-no active project Unknown command: crawl

    在运行别人的scrapy项目时,使用命令行 scrapy crawl douban(douban是该项目里爬虫的名字,烂大街的小项目---抓取豆瓣电影). 执行之后,出现报错如下: 上网搜寻无果. 大 ...

  4. C++ GUI Qt4编程(09)-3.3spreadsheet-toolbar

    1. C++ GUI Qt4编程第三章,增加工具栏.状态栏和快捷键. 2. mainwindow.h /**/ #ifndef MAINWINDOW_H #define MAINWINDOW_H #i ...

  5. http statusCode(状态码) 200、300、400、500序列详解

    201-206都表示服务器成功处理了请求的状态代码,说明网页可以正常访问.200(成功) 服务器已成功处理了请求.通常,这表示服务器提供了请求的网页.201(已创建) 请求成功且服务器已创建了新的资源 ...

  6. oracle 备份恢复篇(三)---rman spfile的丢失

    一,环境准备 1, 拥有全备 数据 2, 查看spfile文件位置 SQL> SQL> SELECT NAME, VALUE, DISPLAY_VALUE FROM V$PARAMETER ...

  7. jdk监控tomcat

    一, tomcat配置文件 在tomcat的配置文件中添加被监控的项 #在tomcat配置文件中开启监控功能 vim /application/tomcat/bin/catalina.sh +97 C ...

  8. 日志收集之filebeat

    一,软件介绍 Filebeat是一个轻量级日志传输Agent,可以将指定日志转发到Logstash.Elasticsearch.Kafka.Redis等中.Filebeat占用资源少,而且安装配置也比 ...

  9. java String类型转 java.sql.time类型

    String[] timePhase = reservationRuleInDTO.getTimePhase().split(",");List<ReservationTim ...

  10. DP Intro - Tree POJ2342 Anniversary party

    POJ 2342 Anniversary party (树形dp 入门题) Anniversary party Time Limit: 1000MS   Memory Limit: 65536K To ...