在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种常用方法的更多相关文章

  1. java线程池和五种常用线程池的策略使用与解析

    java线程池和五种常用线程池策略使用与解析 一.线程池 关于为什么要使用线程池久不赘述了,首先看一下java中作为线程池Executor底层实现类的ThredPoolExecutor的构造函数 pu ...

  2. java线程池与五种常用线程池策略使用与解析

    背景:面试中会要求对5中线程池作分析.所以要熟知线程池的运行细节,如CachedThreadPool会引发oom吗? java线程池与五种常用线程池策略使用与解析 可选择的阻塞队列BlockingQu ...

  3. Java线程同步的四种方式详解(建议收藏)

    ​ Java线程同步属于Java多线程与并发编程的核心点,需要重点掌握,下面我就来详解Java线程同步的4种主要的实现方式@mikechen 目录 什么是线程同步 线程同步的几种方式 1.使用sync ...

  4. Java线程池的几种实现 及 常见问题讲解

    工作中,经常会涉及到线程.比如有些任务,经常会交与线程去异步执行.抑或服务端程序为每个请求单独建立一个线程处理任务.线程之外的,比如我们用的数据库连接.这些创建销毁或者打开关闭的操作,非常影响系统性能 ...

  5. Java线程创建的两种方式

    java多线程总结一:线程的两种创建方式及优劣比较 (一)---之创建线程的两种方式 java实现多线程的两种方法的比较

  6. Java线程池的四种创建方式

    Java通过Executors提供四种线程池,分别为:newCachedThreadPool创建一个可缓存线程池,如果线程池长度超过处理需要,可灵活回收空闲线程,若无可回收,则新建线程. newFix ...

  7. Java线程--interrupt join yield setDaemon常用方法的使用

    概念: 操作系统可以有多个进程,一个线程可以有一个或多个线程.进程与进程之间不共享内存,都在各自的空间中运行.而线程不仅可以共享内存,还可以用有一个自己的内存空间,叫做线程栈. 线程又称轻量级进程.J ...

  8. Java更新XML的四种常用方法简介

    本文简要的讨论了Java语言编程中更新XML文档的四种常用方法,并且分析这四种方法的优劣.其次,本文还对如何控制Java程序输出的XML文档的格式做了展开论述. JAXP是Java API for X ...

  9. java线程实现的四种方式

    java多线程的实现可以通过以下四种方式 1.继承Thread类,重写run方法 2.实现Runnable接口,重写run方法 3.通过Callable和FutureTask创建线程 4.通过线程池创 ...

随机推荐

  1. 2018.09.08 NOIP模拟trip(最长链计数)

    差不多是原题啊. 求最长链变成了最长链计数,其余没有变化. 这一次考试为了保险起见本蒟蒻还是写了上次没写的辅助数组. 代码: #include<bits/stdc++.h> #define ...

  2. Ansible 笔记 (3) - 编写 playbook

    playbook 相当于多个命令的编排组合然后一起运行,类似写脚本.在学习 playbook 之前需要了解 yaml 格式. 编写playbook的步骤: 定义主机与用户 编写任务列表 执行 play ...

  3. 14)settings.xml

    1. User Level. ${user.home}/.m2/settings.xml 2. Global Level. ${maven.home}/conf/settings.xml <se ...

  4. (二分匹配“匈牙利算法”)无题II --HDU --2236

    链接: http://acm.hdu.edu.cn/showproblem.php?pid=2236 代码: #include<cstdio> #include<cstring> ...

  5. MFC中和定时器使用

    在MFC中和定时器相关的有三个函数: 1.设置定时器(定义一个定时器的属性):         SetTimer( UINT nIDEvent, UINT nElapse, void (CALLBAC ...

  6. EBS Archiving and Purging: You Know you need to

    A number of trends in the IT industry have contributed to the increasing size of ERP application dat ...

  7. Delphi Language Overview

    Delphi is a high-level, compiled, strongly typed language that supports structured and object-orient ...

  8. Android-fragment简介-fragment的简单使用

    1.fragment简介 在Android3.0版本之前Google还没有推出fragment,在Android3.0版本之后Google推出了fragment,由于Android3.0版本是过渡期版 ...

  9. Quartz.net 起步

    安装 Quartz 程序包 使用 nuget 命令行安装 Quartz: Install-Package Quartz 如果使用 JSON 序列化,使用 nuget 安装 Quartz.Seriali ...

  10. Checkpoint--相关问题

    Checkpoint是实例级别还是数据库级别? 答:数据库级别,在SQL Server关闭时,会对所有数据库逐一提交checkpoint 测试代码 USE DB0002 GO CHECKPOINT G ...