最近看到一个多线程面试题,有三个线程分别打印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类变量问题引起的。。。的更多相关文章

  1. 并行编程之多线程共享非volatile变量,会不会可能导致线程while死循环

    背景 大家都知道线程之间共享变量要用volatilekeyword.可是,假设不用volatile来标识,会不会导致线程死循环?比方以下的伪代码: static int flag = -1; void ...

  2. 用JAVA写一个多线程程序,写四个线程,其中二个对一个变量加1,另外二个对一个变量减1

    package com.ljn.base; /** * @author lijinnan * @date:2013-9-12 上午9:55:32 */ public class IncDecThrea ...

  3. python编程系列---多线程共享全局变量出现了安全问题的解决方法

    多线程共享全局变量出现了安全问题的解决方法 当多线程共享全局变量时,可能出现安全问题,解决机制----互斥锁:即在在一段与全局变量修改相关的代码中,假设一个时间片不足以完成全局变量的修改,就在这段代码 ...

  4. Delphi另一个多线程函数:BeginThread用法

    Delphi另一个多线程函数:BeginThread━━━━━━━━━━━━━━━━━━━━━━━━━━ Delphi也提供了一个相同功能的类似函数:function BeginThread(    ...

  5. JAVA笔记14__多线程共享数据(同步)/ 线程死锁 / 生产者与消费者应用案例 / 线程池

    /** * 多线程共享数据 * 线程同步:多个线程在同一个时间段只能有一个线程执行其指定代码,其他线程要等待此线程完成之后才可以继续执行. * 多线程共享数据的安全问题,使用同步解决. * 线程同步两 ...

  6. Python 中多线程共享全局变量的问题

    写在前面不得不看的一些P话: Python 中多个线程之间是可以共享全局变量的数据的. 但是,多线程共享全局变量是会出问题的. 假设两个线程 t1 和 t2 都要对全局变量g_num (默认是0)进行 ...

  7. 用 python 实现一个多线程网页下载器

    今天上来分享一下昨天实现的一个多线程网页下载器. 这是一个有着真实需求的实现,我的用途是拿它来通过 HTTP 方式向服务器提交游戏数据.把它放上来也是想大家帮忙挑刺,找找 bug,让它工作得更好. k ...

  8. python_way ,day11 线程,怎么写一个多线程?,队列,生产者消费者模型,线程锁,缓存(memcache,redis)

    python11 1.多线程原理 2.怎么写一个多线程? 3.队列 4.生产者消费者模型 5.线程锁 6.缓存 memcache redis 多线程原理 def f1(arg) print(arg) ...

  9. C++实现一个多线程同步方式的协同工作程序示例

    多线程并发程序与协同程序其实是不同的概念.多线程并发是多个执行序同时运行,而协同程序是多个执行序列相互协作,同一时刻只有一个执行序列.今天想到的是将两者结合起来,拿现实生活中的例子来说,假设一个班级有 ...

随机推荐

  1. 【Win 10应用开发】自定义浮动层——Flyout

    最近几天总是下雨,真是“何处秋窗无雨声”,也“不知风雨几时休”. 好,进入正题. 弹出层有三种. 第一种是ContentDialog,即内容对话框,它其实类似于模态对话框,弹出后会覆盖整个窗口区域,并 ...

  2. owner:轻松管理java项目配置

    前段时间,一同事说在 github 上“活捉了”一个很有趣的开源项目,它是一个超轻量级的 jar 包,能够帮助你在 java 项目中摒弃样板式的 properties 配置代码,让你轻松自如地管理和使 ...

  3. HTML5 学习总结(二)——HTML5新增属性与表单元素

    一.HTML5新增属性 1.1.contextmenu contextmenu的作用是指定右键菜单. <!DOCTYPE html> <html> <head> & ...

  4. jQuery源码学习感想

    还记得去年(2015)九月份的时候,作为一个大四的学生去参加美团霸面,结果被美团技术总监教育了一番,那次问了我很多jQuery源码的知识点,以前虽然喜欢研究框架,但水平还不足够来研究jQuery源码, ...

  5. Vue.js学习笔记(1)

    数据的双向绑定(ES6写法) 效果: 没有改变 input 框里面的值时

  6. Unity3D中使用委托和事件

    前言: 本来早就想写写和代码设计相关的东西了,以前做2DX的时候就有过写写观察者设计模式的想法,但是实践不多.现在转到U3D的怀抱中,倒是接触了不少委托事件的写法,那干脆就在此总结一下吧. 1.C#中 ...

  7. 自己手写的自动完成js类

    在web开发中,为了提高用户体验,会经常用到输入框的自动完成功能,不仅帮助用户进行快速输入,最重要的是帮助那些“记不全要输入什么”的用户进行选择.这个功能有很多插件已经实现了,为了适应项目的特殊需求, ...

  8. app端上传文件至服务器后台,web端上传文件存储到服务器

    1.android前端发送服务器请求 在spring-mvc.xml 将过滤屏蔽(如果不屏蔽 ,文件流为空) <!-- <bean id="multipartResolver&q ...

  9. PostCSS一种更优雅、更简单的书写CSS方式

    Sass团队创建了Compass大大提升CSSer的工作效率,你无需考虑各种浏览器前缀兼,只需要按官方文档的书写方式去写,会得到加上浏览器前缀的代码,如下: .row { @include displ ...

  10. script标签中defer和async属性的区别

    这篇文章来源于JS高级程序设计第三版中关于script标签的介绍,结合查阅的资料写下的学习笔记. 向html页面中插入javascript代码的主要方法就是通过script标签.其中包括两种形式,第一 ...