大家看下这个关于线程的代码

做一只蝴蝶 2016-05-21 10:53:33
package cn.itcast.thread;
/*
线程通讯: 一个线程完成了自己的任务时,要通知另外一个线程去完成另外一个任务.

生产者与消费者


wait(): 等待 如果线程执行了wait方法,那么该线程会进入等待的状态,等待状态下的线程必须要被其他线程调用notify方法才能唤醒。
notify(): 唤醒 唤醒线程池等待线程其中的一个。
notifyAll() : 唤醒线程池所有等待 线程。


wait与notify方法要注意的事项:
1. wait方法与notify方法是属于Object对象 的。
2. wait方法与notify方法必须要在同步代码块或者是同步函数中才能 使用。
3. wait方法与notify方法必需要由锁对象调用。

问题一:出现了线程安全问题。 价格错乱了...

*/

//产品类
class Product{

String name; //名字

double price; //价格

boolean flag = false; //产品是否生产完毕的标识,默认情况是没有生产完成。

}

//生产者
class Producer extends Thread{

Product p ; //产品

public Producer(Product p) {
this.p = p ;
}



@Override
public void run() {
int i = 0 ;
while(true){
synchronized (p) {
if(p.flag==false){
if(i%2==0){
p.name = "苹果";
p.price = 6.5;
}else{
p.name="香蕉";
p.price = 2.0;
}
System.out.println("生产者生产出了:"+ p.name+" 价格是:"+ p.price);
p.flag = true;
i++;
p.notifyAll(); //唤醒消费者去消费
}else{
//已经生产 完毕,等待消费者先去消费
try {
p.wait(); //生产者等待
} catch (InterruptedException e) {
e.printStackTrace();
}
}

}
}
}
}


//消费者
class Customer extends Thread{

Product p;

public Customer(Product p) {
this.p = p;
}


@Override
public void run() {
while(true){
synchronized (p) {
if(p.flag==true){ //产品已经生产完毕
System.out.println("消费者消费了"+p.name+" 价格:"+ p.price);
p.flag = false;
p.notifyAll(); // 唤醒生产者去生产
}else{
//产品还没有生产,应该 等待生产者先生产。
try {
p.wait(); //消费者也等待了...
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}
}

public class Demo5 {

public static void main(String[] args) {
Product p = new Product(); //产品
//创建生产对象
Producer producer = new Producer(p);
//创建消费者
Customer customer = new Customer(p);
//调用start方法开启线程
producer.start();
customer.start();


}

}

正常来说结果应该是这样的
但是我也经常能够输出这样的结果
先消费了?我没搞明白。求各位指点一下。好燥人啊。。。。
...全文
106 4 打赏 收藏 转发到动态 举报
写回复
用AI写文章
4 条回复
切换为时间正序
请发表友善的回复…
发表回复
做一只蝴蝶 2016-05-21
  • 打赏
  • 举报
回复
引用 1 楼 weikeni19 的回复:
你要吃那么多水果干什么?虽然说水果能养身 强身健体 但也不必吃那么多呀
你这。。。。大哥,咱正经点啊
weikeni19 2016-05-21
  • 打赏
  • 举报
回复
你要吃那么多水果干什么?虽然说水果能养身 强身健体 但也不必吃那么多呀
做一只蝴蝶 2016-05-21
  • 打赏
  • 举报
回复
引用 3 楼 byssh1989 的回复:
你这估计是控制台输出超过长度,最上面的输出被清楚了 你可以让线程sleep几秒钟测测看
我先前吃饭的时候也感觉是控制台的问题。刚试了一下您的方法,输出正常了。被这事耽搁了一早上,谢谢了
LOVEv海盗 2016-05-21
  • 打赏
  • 举报
回复
你这估计是控制台输出超过长度,最上面的输出被清楚了 你可以让线程sleep几秒钟测测看

23,407

社区成员

发帖
与我相关
我的任务
社区描述
Java 非技术区
社区管理员
  • 非技术区社区
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧