(原创)确保JAVA线程安全的4种常用方法
在Java中可以有很多方法来保证线程安全,比如使用同步方法、同步块,使用原子类(atomic concurrent classes),实现并发锁,使用volatile关键字,使用不变类和线程安全类。
这里是最基础的线程安全教程
实际上在volatile的使用上很容易有误解,以为volatile就可以做原子操作,实际不然。Volatile 变量具有 synchronized 的可见性特性,但是不具备原子特性。这就是说线程能够自动发现 volatile 变量的最新值。Volatile 变量可用于提供线程安全,但是只能应用于非常有限的一组用例:多个变量之间或者某个变量的当前值与修改后值之间没有约束。因此,单独使用 volatile 还不足以实现计数器、互斥锁或任何具有与多个变量相关的不变式(Invariants)的类(例如 “start <=end”)。
对于volatile修饰的变量,jvm虚拟机只是保证从主内存加载到线程工作内存的值是最新的。
直接上代码:
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; public class Counter { public static int count = ; //对于值引用来说,多线程操作的是变量的副本,操作完后刷新到主存中.所以不具有原子性。
//错误的volatile使用方法,volatil只是直接进行内存地址操作,但并不能保证线程安全.volatile很容易被误用,用来进行原子性操作,
public volatile static int volatileCount = ;
static Object obj =new Object();
public static AtomicInteger atomicCount;// 正确的方法1,使用原子操作 static class MyObject{// 正确的方法4,使用地址引用,多线程是通过地址操作。值的改变是同一个变量(地址)
static int mycount=;
} public static void inc1() {
MyObject.mycount++;
} public static void inc() {
//这里延迟1毫秒,使得结果明显
try {
Thread.sleep();
} catch (InterruptedException e) {
} //典型错误1:在资源对象加锁显然是不对的,实际上毫无意义
//Lock lock =new ReentrantLock();
//lock.lock();
//synchronized (obj) // 正确的方法2,可重人的同步块操作。这也是最常用的办法
{
count++;
volatileCount++;
atomicCount.incrementAndGet();
}
//lock.unlock();
} public static void main(String[] args) { //同时启动100个线程,去进行i++计算,看看实际结果
atomicCount =new AtomicInteger();
Lock lock =new ReentrantLock(); // 正确的方法3,可重人锁 ReentrantLock
Thread threads[]=new Thread[]; for (int i = ; i < ; i++) {
threads[i]=new Thread(new Runnable() {
@Override
public void run() {
//lock.lock();// 正确的方法3,可重人锁 ReentrantLock
Counter.inc();
//lock.unlock();
inc1();
}
});
threads[i].start();
} //保障线程全部结束
for(int i=;i<;i++){
try {
threads[i].join();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
} //如果没有同步锁.值有可能不同。
System.out.println("运行结果1:Counter.count=" + Counter.count);
//atomicCount值都应该是一致的
System.out.println("运行结果2:Counter.atomicCount=" + Counter.atomicCount);
//atomicCount值。如果没有同步锁.值有可能不同。
System.out.println("运行结果3:Counter.volatileCount=" + Counter.volatileCount);
//使用地址引用,多线程是通过地址操作。值的改变是同一个变量(地址)。值都应该是一致的
System.out.println("运行结果4:Counter.mycount=" +MyObject.mycount);
} }
运行之后,结果可能会这样
运行结果1:Counter.count=96
运行结果2:Counter.atomicCount=100
运行结果3:Counter.volatileCount=97
运行结果4:Counter.mycount=100
如果在52行和54行取消注释(或者取消32行的注释),结果必然如下:
运行结果1:Counter.count=100
运行结果2:Counter.atomicCount=100
运行结果3:Counter.volatileCount=100
运行结果4:Counter.mycount=100
(原创)确保JAVA线程安全的4种常用方法的更多相关文章
- java线程池和五种常用线程池的策略使用与解析
java线程池和五种常用线程池策略使用与解析 一.线程池 关于为什么要使用线程池久不赘述了,首先看一下java中作为线程池Executor底层实现类的ThredPoolExecutor的构造函数 pu ...
- java线程池与五种常用线程池策略使用与解析
背景:面试中会要求对5中线程池作分析.所以要熟知线程池的运行细节,如CachedThreadPool会引发oom吗? java线程池与五种常用线程池策略使用与解析 可选择的阻塞队列BlockingQu ...
- Java线程同步的四种方式详解(建议收藏)
Java线程同步属于Java多线程与并发编程的核心点,需要重点掌握,下面我就来详解Java线程同步的4种主要的实现方式@mikechen 目录 什么是线程同步 线程同步的几种方式 1.使用sync ...
- Java线程池的几种实现 及 常见问题讲解
工作中,经常会涉及到线程.比如有些任务,经常会交与线程去异步执行.抑或服务端程序为每个请求单独建立一个线程处理任务.线程之外的,比如我们用的数据库连接.这些创建销毁或者打开关闭的操作,非常影响系统性能 ...
- Java线程创建的两种方式
java多线程总结一:线程的两种创建方式及优劣比较 (一)---之创建线程的两种方式 java实现多线程的两种方法的比较
- Java线程池的四种创建方式
Java通过Executors提供四种线程池,分别为:newCachedThreadPool创建一个可缓存线程池,如果线程池长度超过处理需要,可灵活回收空闲线程,若无可回收,则新建线程. newFix ...
- Java线程--interrupt join yield setDaemon常用方法的使用
概念: 操作系统可以有多个进程,一个线程可以有一个或多个线程.进程与进程之间不共享内存,都在各自的空间中运行.而线程不仅可以共享内存,还可以用有一个自己的内存空间,叫做线程栈. 线程又称轻量级进程.J ...
- Java更新XML的四种常用方法简介
本文简要的讨论了Java语言编程中更新XML文档的四种常用方法,并且分析这四种方法的优劣.其次,本文还对如何控制Java程序输出的XML文档的格式做了展开论述. JAXP是Java API for X ...
- java线程实现的四种方式
java多线程的实现可以通过以下四种方式 1.继承Thread类,重写run方法 2.实现Runnable接口,重写run方法 3.通过Callable和FutureTask创建线程 4.通过线程池创 ...
随机推荐
- RSA生成、加密、解密、签名。
首先,要会生成RSA密码对. https://app.alipay.com/market/document.htm?name=saomazhifu#page-23 (事例中的密钥对好像有问题,最 ...
- group by 和 distinct 的区别
SELECT fs.card_id, fs. NAME, fs.email, fs.phone_num, fs.weixin_num, fs.permission, fs.open_id FROM f ...
- MYSQL 问题小总结
mysql 问题小总结 1.MySQL远程连接ERROR 2003(HY000):Can't connect to MySQL server on ‘ip’(111)的问题 通常是mysql配置文件中 ...
- shell的基本语法
一 赋值运算符 1 += :使用方法是,((x+=需要增加的数字))算和值. 2 *= :使用方法是,((x*=需要怎加的倍数))算乘值. 3 %= :使用方法是,((x%=需要除以的数字))算余数 ...
- 使用eclipse创建android项目的时候为什么会生成两个项目
使用eclipse创建android项目的时候为什么会生成两个项目 问题描述: 使用eclipse创建一个Android项目时,发现project列表中会多创建出一个appcompat_v7项目,再创 ...
- 2018.09.14 洛谷P3567 [POI2014]KUR-Couriers(主席树)
传送门 简单主席树啊. 但听说有随机算法可以秒掉%%%(本蒟蒻并不会) 直接维护值域内所有数的出现次数之和. 当这个值不大于区间总长度的一半时显然不存在合法的数. 这样在主席树上二分查值就行了. 代码 ...
- 2018.09.11 bzoj3629: [JLOI2014]聪明的燕姿(搜索)
传送门 一道神奇的搜索. 直接枚举每个质因数的次数,然后搜索就行了. 显然质因数k次数不超过logkn" role="presentation" style=" ...
- 2018.07.10NOIP模拟 Draw(容斥原理)
Draw 题目背景 SOURCE:NOIP2016-RZZ-4 T3 题目描述 给定笛卡尔坐标系上 n 个不重复的点. 定义一个 L 形为: 一个形如 (x,y),(x+1,y)-(x+a,y),(x ...
- js限制上传图片类型和大小
<script type="text/javascript"> function checkFile(brandLogo){ var file=brandLogo.va ...
- 关于iOS的自动弹出键盘问题
-(void)viewWillAppear:(BOOL)animated { [super viewWillAppear:animated]; [self.textField beco ...