由一个多线程共享Integer类变量问题引起的。。。
最近看到一个多线程面试题,有三个线程分别打印A、B、C,请用多线程编程实现,在屏幕上循环打印10次ABCABC…
看到这个题目,首先想到的是解决方法是定义一个Integer类对象,初始化为0,由3个线程共享,如果Integer对象取余3之后等于0,则打印A,同时进行加1操作;如果Integer对象取3之后等于1,则打印B,同时进行加1操作;如果Integer对象取3之后等于1,则打印C,如果循环打印了10次的话,就退出线程。
/**
* ThreeThread
* 3个线程测试
*/
public class ThreeThread { public static void main(String[] args) throws InterruptedException {
Integer gData = 0;
Thread thread1 = new MyTask(gData, 0, "A");
Thread thread2 = new MyTask(gData, 1, "B");
Thread thread3 = new MyTask(gData, 2, "C"); thread1.start();
thread2.start();
thread3.start(); thread1.join();
thread2.join();
thread3.join();
} } class MyTask extends Thread { private Integer gData;
private int n;
private String info; public MyTask(Integer gData, int n, String info) {
super("thread " + info);
this.gData = gData;
this.n = n;
this.info = info;
} public void run() {
int i = 0; while (true) {
synchronized (gData) {
if (gData % 3 == n) {
System.out.print(info + " ");
gData++;
i++;
}
} if (i == 10) {
break;
}
else {
Thread.yield();
try {
sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
} }
运行程序结果如下:

发现只有A线程打印了"A",并没有发现B线程和C线程打印字符串:(。难道是A线程更改了Integer对象的值,而B线程和C线程并没有“看到”更新后的值?于是,在线程类的run方法的while循环中增加代码如下:
while (true) {
System.out.println(Thread.currentThread().getName() + " " + gData);
synchronized (gData) {
if (gData % 3 == n) {
System.out.print(info + " ");
gData++;
i++;
}
}
...
}
运行程序结果如下:

由运行结果可知,刚开始A、B、C线程都拥有Integer类变量,并且初值为0。当A线程更改Integer类变量为1时,但是B和C线程中的Integer类变量的值仍然为0,因此,结果肯定不会打印出ABCABC....
通过阅读Integer类源码,可知Integer类中存放int值的变量类型是final的:
/**
* The value of the {@code Integer}.
*
* @serial
*/
private final int value;
也就是说,Integer类对象的值每更新一次,就会创建一个新的Integer对象。运行程序结果只打印出了"A",表示刚开始A、B、C线程都拥有同一个Integer类变量,并且初值为0,但是当A线程更新Integer对象的值后,A线程中的Integer对象和B/C线程中的Integer对象已经不是同一个对象了。
为了能够正常打印出ABCABC字符串,可以把Integer对象类型改为AtomicInteger,代码如下:
/**
* ThreeThread
* 3个线程测试
*/
public class ThreeThread { public static void main(String[] args) throws InterruptedException {
AtomicInteger gData = new AtomicInteger(0);
Thread thread1 = new MyTask(gData, 0, "A");
Thread thread2 = new MyTask(gData, 1, "B");
Thread thread3 = new MyTask(gData, 2, "C"); thread1.start();
thread2.start();
thread3.start(); thread1.join();
thread2.join();
thread3.join();
} } class MyTask extends Thread { private AtomicInteger gData;
private int n;
private String info; public MyTask(AtomicInteger gData, int n, String info) {
super("thread " + info);
this.gData = gData;
this.n = n;
this.info = info;
} public void run() {
int i = 0; while (true) {
synchronized (gData) {
if (gData.get() % 3 == n) {
System.out.print(info + " ");
gData.incrementAndGet();
i++;
}
} if (i == 10) {
break;
}
else {
Thread.yield();
try {
sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
} }

第二种打印ABCABC...字符串的解决方法是使用wait/notify函数,示例代码如下:
/**
* ThreeThread2
* 三个线程依次输出A B C,使用线程同步方式
*/
public class ThreeThread2 { public static void main(String[] args) throws InterruptedException {
Object A = new Object();
Object B = new Object();
Object C = new Object(); MyThread myThread1 = new MyThread(C, A, "A");
MyThread myThread2 = new MyThread(A, B, "B");
MyThread myThread3 = new MyThread(B, C, "C"); myThread1.start();
Thread.sleep(10);
myThread2.start();
Thread.sleep(10);
myThread3.start(); try {
myThread1.join();
myThread2.join();
myThread3.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
} } class MyThread extends Thread {
private Object prev;
private Object curr;
private String info; public MyThread(Object prev, Object curr, String info) {
this.prev = prev;
this.curr = curr;
this.info = info;
} public void run() {
int cnt = 10; while (cnt-- > 0) {
synchronized (prev) {
synchronized (curr) {
System.out.print(info + " ");
curr.notify();
} try {
prev.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}

由一个多线程共享Integer类变量问题引起的。。。的更多相关文章
- 并行编程之多线程共享非volatile变量,会不会可能导致线程while死循环
背景 大家都知道线程之间共享变量要用volatilekeyword.可是,假设不用volatile来标识,会不会导致线程死循环?比方以下的伪代码: static int flag = -1; void ...
- 用JAVA写一个多线程程序,写四个线程,其中二个对一个变量加1,另外二个对一个变量减1
package com.ljn.base; /** * @author lijinnan * @date:2013-9-12 上午9:55:32 */ public class IncDecThrea ...
- python编程系列---多线程共享全局变量出现了安全问题的解决方法
多线程共享全局变量出现了安全问题的解决方法 当多线程共享全局变量时,可能出现安全问题,解决机制----互斥锁:即在在一段与全局变量修改相关的代码中,假设一个时间片不足以完成全局变量的修改,就在这段代码 ...
- Delphi另一个多线程函数:BeginThread用法
Delphi另一个多线程函数:BeginThread━━━━━━━━━━━━━━━━━━━━━━━━━━ Delphi也提供了一个相同功能的类似函数:function BeginThread( ...
- JAVA笔记14__多线程共享数据(同步)/ 线程死锁 / 生产者与消费者应用案例 / 线程池
/** * 多线程共享数据 * 线程同步:多个线程在同一个时间段只能有一个线程执行其指定代码,其他线程要等待此线程完成之后才可以继续执行. * 多线程共享数据的安全问题,使用同步解决. * 线程同步两 ...
- Python 中多线程共享全局变量的问题
写在前面不得不看的一些P话: Python 中多个线程之间是可以共享全局变量的数据的. 但是,多线程共享全局变量是会出问题的. 假设两个线程 t1 和 t2 都要对全局变量g_num (默认是0)进行 ...
- 用 python 实现一个多线程网页下载器
今天上来分享一下昨天实现的一个多线程网页下载器. 这是一个有着真实需求的实现,我的用途是拿它来通过 HTTP 方式向服务器提交游戏数据.把它放上来也是想大家帮忙挑刺,找找 bug,让它工作得更好. k ...
- python_way ,day11 线程,怎么写一个多线程?,队列,生产者消费者模型,线程锁,缓存(memcache,redis)
python11 1.多线程原理 2.怎么写一个多线程? 3.队列 4.生产者消费者模型 5.线程锁 6.缓存 memcache redis 多线程原理 def f1(arg) print(arg) ...
- C++实现一个多线程同步方式的协同工作程序示例
多线程并发程序与协同程序其实是不同的概念.多线程并发是多个执行序同时运行,而协同程序是多个执行序列相互协作,同一时刻只有一个执行序列.今天想到的是将两者结合起来,拿现实生活中的例子来说,假设一个班级有 ...
随机推荐
- 让DIV中文字换行显示
1. <style> div { white-space:normal; word-break:break-all; word-wrap:break-word; } </style& ...
- 集成学习之Adaboost算法原理小结
在集成学习原理小结中,我们讲到了集成学习按照个体学习器之间是否存在依赖关系可以分为两类,第一个是个体学习器之间存在强依赖关系,另一类是个体学习器之间不存在强依赖关系.前者的代表算法就是是boostin ...
- ASP.NET(IIS)出现"没有为请求类型"GET"找到 HTTP 处理程序"
pasting 环 境:windows 2003 IIS6.0 程序支持:ASP.NET .NET版本:2.0.4.0 问 题:之前服务器上各大网站运行良好,今天突然出现:" ...
- html5 canvas常用api总结(一)
1.监听浏览器加载事件. window.addEventListener("load",eventWindowLoaded,false); load事件在html页面加载结束时发生 ...
- Sql Server 内存相关计数器以及内存压力诊断
在数据库服务器中,内存是数据库对外提供服务最重要的资源之一, 不仅仅是Sql Server,包括其他数据库,比如Oracle,MySQL等,都是一类非常喜欢内存的应用. 在Sql Server服务器中 ...
- “为什么DirectX里表示三维坐标要建一个4*4的矩阵?”
0x00 前言 首先要说明的是,本文的标题事实上来自于知乎上的一个同名问题:为什么directX里表示三维坐标要建一个4*4的矩阵? - 编程 .因此,正如Milo Yip大神所说的这个标题事实上是存 ...
- Azure Backup (3) 使用Azure备份服务,备份Azure虚拟机
<Windows Azure Platform 系列文章目录> 本将介绍,如何使用Azure备份服务,备份Azure虚拟机. 我们先预先创建2台Windows VM (命名为LeiVM00 ...
- Moon.Orm 常见查询实例
一.Moon.Orm框架总述 (您还用hibernate?实体框架?) 1.框架名:Moon 意思是月亮,而非Mono.因为很喜欢明月,所以以此为名.它是一个.NET下的Orm框架. 2.发展历史:历 ...
- 创建或打开解决方案时提示"DotNetCore.1.0.1-SDK.1.0.0.Preview2-003131-x86"错误的解决方案
提示"DotNetCore.1.0.1-SDK.1.0.0.Preview2-003131-x86"错误的解决方案: 1.检查是否有C:\Program Files (x86)\d ...
- C#开发微信门户及应用(19)-微信企业号的消息发送(文本、图片、文件、语音、视频、图文消息等)
我们知道,企业号主要是面向企业需求而生的,因此内部消息的交流显得非常重要,而且发送.回复消息数量应该很可观,对于大企业尤其如此,因此可以结合企业号实现内部消息的交流.企业号具有关注安全.消息无限制等特 ...