并发 - 这段java多线程代码有问题吗?
PHPz
PHPz 2017-04-17 11:30:19
[Java讨论组]
public class Main {
public static void main(String[] args) {
MyThread mt = new MyThread();
new Thread(mt).start();
new Thread(mt).start();
new Thread(mt).start();
}
}

class MyThread implements Runnable {
private int ticket = 20;

public void run() {
for (int i = 0; i < 20; i++) {
if (this.ticket > 0) {
System.out.println("线程:"+Thread.currentThread().getName()+" 卖票:ticket" + this.ticket--);
}
}
}
}
PHPz
PHPz

学习是最好的投资!

全部回复(3)
迷茫

不知道你想要干什么。不过要是买票,可能下面才是对的。
public synchronized void run() ....

天蓬老师

你这个涉及到一个共享变量的问题,ticket变量被三个线程同时访问,就可能导致多个线程读到同样的值,这显然是错误的。而且这只是其中一种情况。更复杂的是--操作并不是原子的,这就会产生另一种情况,如果ticket不是int,而是long,递增/递减是分步进行的,这中间可能另一个线程同时向ticket写值,从而产生更离奇的输出。
所以就要对共享变量进行同步控制,synchronized可以用在方法上,也可以以同步块的方式进行。更简单的上使用java.util.concurrent.atomic.AtomicInteger,它提供原子的递增/递减方法。
而且,在输出的地方也会出问题。因为字符串的拼接也不是原子的,所以肯出现输出混乱的情况。

大家讲道理

1楼的建议不错,但是直接对方法加锁的话,性能下降比较严重。。。一般是对ticket这个变量加锁,会好一点。

热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习
PHP中文网抖音号
发现有趣的

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号