Java多线程之非线程安全
在Java多线程中我会重点总结五个如下的技术点:
1、非线程安全是如何出现的
2、synchronized对象监视器为Objec时的使用
3、synchronized对象监视器为Class时的使用
4、关键字volatile的主要作用
今天我先说一说第一个问题,非线程安全是如何出现的。“非线程安全”会在多个线程对同一个对象中的实例变量进行并发访问的时候发生,产生的后果就是“脏读”,也就是读取到的数其实是已经被更改过的了。下面我分别实现“线程安全”和“非线程安全”的两个例子,让大家来体会一下对实例变量更改是怎么发生的。但是多线程的结果会有很多输出结果,希望大家还是亲自敲代码才会理解的更深一些,下面先说一下“线程安全”的例子。如下:
public static void main(String[] args) {
PrivateNum num = new PrivateNum();
MyThreadA a = new MyThreadA(num);
MyThreadB b = new MyThreadB(num);
a.start();
b.start();
}
public static class PrivateNum {
public void addnum(String name) {
//重点是num变量是addnum这个方法的局部变量
int num = 0;
try {
if ("a".equals(name)) {
num = 100;
System.out.println("a ser over!");
Thread.sleep(2000);
} else {
num = 200;
System.out.println("b set over!");
}
System.out.println("线程" + name + "的num=" + num);
} catch (Exception e) {
e.printStackTrace();
}
}
}
public static class MyThreadA extends Thread {
private PrivateNum privateNum;
public MyThreadA(PrivateNum privateNum) {
super();
this.privateNum = privateNum;
}
@Override
public void run() {
super.run();
privateNum.addnum("a");
}
}
public static class MyThreadB extends Thread {
private PrivateNum privateNum;
public MyThreadB(PrivateNum privateNum) {
super();
this.privateNum = privateNum;
}
@Override
public void run() {
super.run();
privateNum.addnum("b");
}
}
运行结果如下:无论输出顺序是怎样的,但是这段代码执行的结果一定是线程安全的,原因就在于num的这个变量是addnum方法私有的局部变量,PrivateNum的对象访问不到num这个变量,所以就没有多num这个变量改变的这个说,自然就不会出现“非线程安全”的状况。

下面在说一个“非线程安全”的例子,如 public static void main(String[] args) { GetNum num = new GetNum();
ThreadA a = new ThreadA(num);
a.start();
ThreadB b = new ThreadB(num);
b.start(); } public static class GetNum {
//之所以会出现“非线程安全”的现象,是因为num这个变量是GetNUm类的全局变量
private int num = 0; public void getNum(String name) {
try { if ("a".equals(name)) {
num = 100;
System.out.println("a set over"); 执行到这里的时候,线程a休眠了,此时num=100,接着线程b执行了,执行的结果把num重新赋值为num=200了,线程b执行完了之后,休眠的2000毫秒也过了,
线程a会接着执行,而不是重新执行,所以这个时候的num仍然是200,发生了“非线程安全问题”
Thread.sleep(2000); } else {
num = 200;
System.out.println("b set over");
} System.out.println("线程" + name + "的num=" + num);
} catch (Exception e) {
e.printStackTrace();
}
}
} public static class ThreadA extends Thread { private GetNum num; public ThreadA(GetNum num) {
super();
this.num = num;
} @Override
public void run() { super.run();
num.getNum("a"); }
} public static class ThreadB extends Thread { private GetNum num; public ThreadB(GetNum num) {
super();
this.num = num;
} @Override
public void run() { super.run();
num.getNum("b"); }
}
运行结果如下:线程a的num=200,出现了“非线程安全”的现象了,出现的原因就是线程a和线程b都对PrivateNum类下的num对象作出改变了,具体的改变过程在程序中已经分析过了。而解决“非线程安全”的问题就是同步,下面的几节会说。

在上述例子中,是在给num赋值的时候出现没有保持线程同步,出现“脏读”,但是在取值的时候也可能出现“脏读”,发生“脏读”是因为在读取实例变量时,这个值已经被修改过了。下面看一个取值发生“脏读”的例子,如下:
public static void main(String[] args) {
try {
PublicVar publicVar = new PublicVar();
ThreadA threadA = new ThreadA(publicVar);
threadA.start();
// 睡眠时间影响输出结果
Thread.sleep(200);
publicVar.getValue();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public static class PublicVar {
public String name = "A";
public String password = "a";
synchronized public void setValues(String name, String password) {
try {
this.name = name;
Thread.sleep(5000);
this.password = password;
System.out.println("setValue method thread name="
+ Thread.currentThread().getName());
System.out.println("setValue name=" + name);
System.out.println("setValue password=" + password);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public void getValue() {
System.out.println("getValue method thread name="
+ Thread.currentThread().getName());
System.out.println("getValue name=" + name);
System.out.println("getValue password=" + password);
}
}
public static class ThreadA extends Thread {
private PublicVar publicVar;
public ThreadA(PublicVar publicVar) {
super();
this.publicVar = publicVar;
}
@Override
public void run() {
super.run();
publicVar.setValues("B", "b");
}
}
运行结果如下:这里边出现“脏读”是因为setValues()方法没有运行完,但是name已经赋值为B,password还没来得赋值就休眠了5000毫秒,所以是main线程先输出的name="B",password="a"。等到5000毫秒到了,子线程又在原来被打断的地方接着运行了,也就是开始给password赋值为b,所以第二次是Thread-0线程输出name="B",password="b"。这里我说过主线程的休眠时间会影响输出结果,如果主线程的休眠时间比子线程的时间长,能够保证子线程执行完,就不会发生“脏读”的情况了。解决脏读就要通过给getValue()方法添加synchronized关键字。

Java多线程之非线程安全的更多相关文章
- Java多线程系列--“JUC线程池”03之 线程池原理(二)
概要 在前面一章"Java多线程系列--“JUC线程池”02之 线程池原理(一)"中介绍了线程池的数据结构,本章会通过分析线程池的源码,对线程池进行说明.内容包括:线程池示例参考代 ...
- Java多线程并发02——线程的生命周期与常用方法,你都掌握了吗
在上一章,为大家介绍了线程的一些基础知识,线程的创建与终止.本期将为各位带来线程的生命周期与常用方法.关注我的公众号「Java面典」了解更多 Java 相关知识点. 线程生命周期 一个线程不是被创建了 ...
- Java多线程系列--“JUC线程池”06之 Callable和Future
概要 本章介绍线程池中的Callable和Future.Callable 和 Future 简介示例和源码分析(基于JDK1.7.0_40) 转载请注明出处:http://www.cnblogs.co ...
- Java多线程系列--“JUC线程池”02之 线程池原理(一)
概要 在上一章"Java多线程系列--“JUC线程池”01之 线程池架构"中,我们了解了线程池的架构.线程池的实现类是ThreadPoolExecutor类.本章,我们通过分析Th ...
- Java多线程系列--“JUC线程池”04之 线程池原理(三)
转载请注明出处:http://www.cnblogs.com/skywang12345/p/3509960.html 本章介绍线程池的生命周期.在"Java多线程系列--“基础篇”01之 基 ...
- Java多线程系列--“JUC线程池”05之 线程池原理(四)
概要 本章介绍线程池的拒绝策略.内容包括:拒绝策略介绍拒绝策略对比和示例 转载请注明出处:http://www.cnblogs.com/skywang12345/p/3512947.html 拒绝策略 ...
- -1-5 java 多线程 概念 进程 线程区别联系 java创建线程方式 线程组 线程池概念 线程安全 同步 同步代码块 Lock锁 sleep()和wait()方法的区别 为什么wait(),notify(),notifyAll()等方法都定义在Object类中
本文关键词: java 多线程 概念 进程 线程区别联系 java创建线程方式 线程组 线程池概念 线程安全 同步 同步代码块 Lock锁 sleep()和wait()方法的区别 为什么wait( ...
- 转:java多线程CountDownLatch及线程池ThreadPoolExecutor/ExecutorService使用示例
java多线程CountDownLatch及线程池ThreadPoolExecutor/ExecutorService使用示例 1.CountDownLatch:一个同步工具类,它允许一个或多个线程一 ...
- Java多线程——进程和线程
Java多线程——进程和线程 摘要:本文主要解释在Java这门编程语言中,什么是进程,什么是线程,以及二者之间的关系. 部分内容来自以下博客: https://www.cnblogs.com/dolp ...
随机推荐
- 获取新浪天气api显示天气情况(转)
直接上一个html的demo <!doctype html> <html class="no-js fixed-layout"> <head> ...
- Java Web 部署到Tomcat
1.在conf目录中,新建Catalina(注意大小写)\localhost目录,在该目录中新建一个xml文件,名字可以随意取,只要和当前文件中的文件名不重复就行了,该xml文件的内容为: <C ...
- linux中的ps命令用法。
在linux中使用ps命令可以查看有哪些进程在运行和运行的状态.进程是否结束.进程有没有僵尸.哪些进程占用了过多的资源等等. ps命令最常用的是用于监控后台进程的工作情况. 名称:ps 使用权限:所有 ...
- 用Java实现 ,冒泡排序与普通排序的区别
冒泡排序与普通排序的区别 /** *个人网址: http://www.lipengfei2013.tk * 功能:冒泡排序与普通排序的区别 */ package www.csdn ...
- 《火球——UML大战需求分析》(0.1)——开篇废话
说明: <火球——UML大战需求分析>是我撰写的一本关于需求分析及UML方面的书,我将会在CSDN上为大家分享前面几章的内容,总字数在几万以上,图片有数十张.欢迎你按文章的序号顺序阅读,谢 ...
- 一、ThinkPHP的介绍
一.ThinkPHP的介绍 //了解 MVC M - Model 模型 工作:负责数据的操作 V - View 视图(模板) 工作:负责前台页面显示 编写html代码 C - Controller 控 ...
- CCNA实验(9) -- Frame Relay
帧中继的一些特点:1.中小企业常用的广域网线路2.通信费用较低3.配置较为复杂 1.将Cisco路由器配置为帧中继交换机2.帧中继基本配置.帧中继映射3.在帧中继的链路上运行RIPv24.帧中继的多点 ...
- Azure Traffic Manager 现可与 Azure 网站集成!
编辑人员注释:本文章由 WindowsAzure 网站团队高级专家级工程师 Jim Cheshire撰写. AzureTraffic Manager 已经推出有一段时间,这是一种跨多个区域管理网 ...
- C#实现 ffmpeg视频转码、播放
近来公司项目要求实现全景相机的视频截取,但是截取的视频需求转码上传.经过研究采用ffmpeg转码,奉上一个详细介绍的博文: 最简单的基于FFMPEG的转码程序 主要是转码的操作过程,能够实现了从相机获 ...
- 第七届河南省赛A.物资调度(dfs)
10401: A.物资调度 Time Limit: 2 Sec Memory Limit: 128 MB Submit: 95 Solved: 54 [Submit][Status][Web Bo ...