热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

由一个多线程共享Integer类变量问题引起的。。。

最近看到一个多线程面试题,有三个线程分别打印A、B、C,请用多线程编程实现,在屏幕上循环打印10次ABCABC…看到这个题目,首先想到的是解决方法是定义一个Integer类对象,初始化为0

  最近看到一个多线程面试题,有三个线程分别打印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();
}
}
}
}
}

 


推荐阅读
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • JavaSE笔试题-接口、抽象类、多态等问题解答
    本文解答了JavaSE笔试题中关于接口、抽象类、多态等问题。包括Math类的取整数方法、接口是否可继承、抽象类是否可实现接口、抽象类是否可继承具体类、抽象类中是否可以有静态main方法等问题。同时介绍了面向对象的特征,以及Java中实现多态的机制。 ... [详细]
  • Java中包装类的设计原因以及操作方法
    本文主要介绍了Java中设计包装类的原因以及操作方法。在Java中,除了对象类型,还有八大基本类型,为了将基本类型转换成对象,Java引入了包装类。文章通过介绍包装类的定义和实现,解答了为什么需要包装类的问题,并提供了简单易用的操作方法。通过本文的学习,读者可以更好地理解和应用Java中的包装类。 ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • Android系统源码分析Zygote和SystemServer启动过程详解
    本文详细解析了Android系统源码中Zygote和SystemServer的启动过程。首先介绍了系统framework层启动的内容,帮助理解四大组件的启动和管理过程。接着介绍了AMS、PMS等系统服务的作用和调用方式。然后详细分析了Zygote的启动过程,解释了Zygote在Android启动过程中的决定作用。最后通过时序图展示了整个过程。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 本文介绍了Java高并发程序设计中线程安全的概念与synchronized关键字的使用。通过一个计数器的例子,演示了多线程同时对变量进行累加操作时可能出现的问题。最终值会小于预期的原因是因为两个线程同时对变量进行写入时,其中一个线程的结果会覆盖另一个线程的结果。为了解决这个问题,可以使用synchronized关键字来保证线程安全。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 模板引擎StringTemplate的使用方法和特点
    本文介绍了模板引擎StringTemplate的使用方法和特点,包括强制Model和View的分离、Lazy-Evaluation、Recursive enable等。同时,还介绍了StringTemplate语法中的属性和普通字符的使用方法,并提供了向模板填充属性的示例代码。 ... [详细]
  • 本文介绍了Java中Hashtable的clear()方法,该方法用于清除和移除指定Hashtable中的所有键。通过示例程序演示了clear()方法的使用。 ... [详细]
  • 阿,里,云,物,联网,net,core,客户端,czgl,aliiotclient, ... [详细]
  • 本文介绍了在Java中gt、gtgt、gtgtgt和lt之间的区别。通过解释符号的含义和使用例子,帮助读者理解这些符号在二进制表示和移位操作中的作用。同时,文章还提到了负数的补码表示和移位操作的限制。 ... [详细]
author-avatar
sl51866
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有