【java】多线程同步生产者消费者问题
package 多线程;
class Producer implements Runnable{
private Data data;
public Producer(Data data){
this.data=data;
}
@Override
public synchronized void run() {
for(int i=0;i<50;i++){
if(i%2==0){
this.data.setTitle("饼干");
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
this.data.setValue("麦香饼干");
}else{
this.data.setTitle("饮料");
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
this.data.setValue("果汁好喝");
}
}
}
}
class Consumer implements Runnable{
private Data data;
public Consumer(Data data){
this.data=data;
}
@Override
public synchronized void run() {
for(int i=0;i<50;i++){
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("消费:"+this.data.getTitle()+"->"+this.data.getValue());
}
}
}
class Data{
private String title;
private String value;
public void setTitle(String title) {
this.title = title;
}
public String getTitle() {
return title;
}
public void setValue(String value) {
this.value = value;
}
public String getValue() {
return value;
}
}
public class TestProducerConsumer {
public static void main(String[] args) {
Data data=new Data();
new Thread(new Producer(data)).start();
new Thread(new Consumer(data)).start();
}
}
数据错乱
package 多线程;
class Producer implements Runnable{
private Data data;
public Producer(Data data){
this.data=data;
}
@Override
public void run() {
for(int i=0;i<50;i++){
if(i%2==0){
this.data.set("饼干","麦香饼干");
}else{
this.data.set("饮料","果汁好喝");
}
}
}
}
class Consumer implements Runnable{
private Data data;
public Consumer(Data data){
this.data=data;
}
@Override
public void run() {
for(int i=0;i<50;i++){
System.out.println("消费:"+this.data.get());
}
}
}
class Data{
private String title;
private String value;
public synchronized void set(String title,String value) {
//若不加synchronized仍会有数据错乱现象
this.title = title;
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
this.value=value;
}
public synchronized String get() {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
return title+"->"+value;
}
}
public class TestProducerConsumer {
public static void main(String[] args) {
Data data=new Data();
new Thread(new Producer(data)).start();
new Thread(new Consumer(data)).start();
}
}
解决错乱,但有重复现象(非奇偶交替)
package 多线程;
class Producer implements Runnable{
private Data data;
public Producer(Data data){
this.data=data;
}
@Override
public void run() {
for(int i=0;i<50;i++){
if(i%2==0){
this.data.set("饼干","麦香饼干");
}else{
this.data.set("饮料","果汁好喝");
}
}
}
}
class Consumer implements Runnable{
private Data data;
public Consumer(Data data){
this.data=data;
}
@Override
public void run() {
for(int i=0;i<50;i++){
System.out.println("消费:"+this.data.get());
}
}
}
class Data{
private boolean flag=true;//如果不初始设置为true可能会在第一个消费时取得null
private String title;
private String value;
public synchronized void set(String title,String value) {
if(this.flag==false)
try {
super.wait();
} catch (InterruptedException e1) {
e1.printStackTrace();
}
//若不加synchronized仍会有数据错乱现象
this.title = title;
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
this.value=value;
this.flag=false;
super.notify();
}
public synchronized String get() {
if(this.flag==true)
try {
super.wait();
} catch (InterruptedException e1) {
e1.printStackTrace();
}
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
this.flag=true;
super.notify();
return title+"->"+value;
}
}
public class TestProducerConsumer {
public static void main(String[] args) {
Data data=new Data();
new Thread(new Producer(data)).start();
new Thread(new Consumer(data)).start();
}
}
解决重复问题
【java】多线程同步生产者消费者问题的更多相关文章
- Java多线程同步——生产者消费者问题
这是马士兵老师的Java视频教程里的一个生产者消费者问题的模型 public class ProduceConsumer{ public static void main(String[] args) ...
- Java 多线程同步生产者消费者问题-monitor
对这个问题更深一点理解是,每一个线程都在竞争这个类的实例的monitor对象. java会为每个object对象分配一个monitor,当某个对象的同步方法(synchronized methods ...
- java多线程解决生产者消费者问题
import java.util.ArrayList; import java.util.List; /** * Created by ccc on 16-4-27. */ public class ...
- java多线程模拟生产者消费者问题,公司面试常常问的题。。。
package com.cn.test3; //java多线程模拟生产者消费者问题 //ProducerConsumer是主类,Producer生产者,Consumer消费者,Product产品 // ...
- Java多线程_生产者消费者模式2
在我的上一条博客中,已经介绍到了多线程的经典案列——生产者消费者模式,但是在上篇中用的是传统的麻烦的非阻塞队列实现的.在这篇博客中我将介绍另一种方式就是:用阻塞队列完成生产者消费者模式,可以使用多种阻 ...
- JAVA多线程之生产者 消费者模式 妈妈做面包案例
创建四个类 1.面包类 锅里只可以放10个面包 ---装面包的容器2.厨房 kitchen 生产面包 和消费面包 最多生产100个面包3.生产者4消费者5.测试类 多线程经典案例 import ja ...
- 【多线程】java多线程实现生产者消费者模式
思考问题: 1.为什么用wait()+notify()实现生产者消费者模式? wait()方法可以暂停线程,并释放对象锁 notify()方法可以唤醒需要该对象锁的其他线程,并在执行完后续步骤,到了s ...
- Java多线程之生产者消费者问题<一>:使用synchronized keyword解决生产者消费者问题
今天看了一片博文,讲Java多线程之线程的协作,当中作者用程序实例说明了生产者和消费者问题,但我及其它读者发现程序多跑几次还是会出现死锁,百度搜了下大都数的样例也都存在bug,经过细致研究发现当中的问 ...
- Java多线程实现生产者消费者延伸问题
在操作系统中有一类问题被称为生产者消费者问题:意为,有数个生产者生产产品,有数个消费者消费产品,他们共享一定数量的缓存. 这里用java多线程编程,实现生产者消费者问题的一种延伸,橘子苹果问题. 题目 ...
- Java多线程-----实现生产者消费者模式的几种方式
1 生产者消费者模式概述 生产者消费者模式就是通过一个容器来解决生产者和消费者的强耦合问题.生产者和消费者彼此之间不直接通讯,而通过阻塞队列来进行通讯,所以生产者生产完数据之后不用等待消费者处理 ...
随机推荐
- java中处理json各种各样转换方法
JSON 即 JavaScript Object Natation,它是一种轻量级的数据交换格式,非常适合于服务器与 JavaScript 的交互.本文将快速讲解 JSON 格式,并通过代码示例演示如 ...
- Linux系列教程(二十三)——Linux的服务管理
前面我们讲解Linux软件包管理之源码包.脚本安装包时,我们介绍了rpm包和源码包由于安装位置的不同,会对服务的启动造成影响,具体是什么,本篇博客我们来详细介绍. 1.Linux服务管理总览 我们可以 ...
- jq获取被选中的option的值。jq获取被选中的单选按钮radio的值。
温故而知新,一起复习下jq的知识点. (1) jq获取被选中的option的值 <select id="select_id"> <option value=&qu ...
- Spring框架——AOP代理
我们知道AOP代理指的就是设计模式中的代理模式.一种是静态代理,高效,但是代码量偏大:另一种就是动态代理,动态代理又分为SDK下的动态代理,还有CGLIB的动态代理.Spring AOP说是实现了AO ...
- 去培训机构参加IT培训值不值
近几年,IT培训机构可谓是琳琅满目,稂莠不齐.培训Java的,培训PHP的,培训大数据的等等吧,不一而足. 自己也算是IT技术圈子待了好多年了,面试过一些机构培训出来的学生,也有几个好哥们在培训机构做 ...
- 集合、set以及HASH
集合的数据结构数据结构就是内存中保存输出数据的形式,不同的数据结构会有不同的特征.堆栈结构:先进后出 代表类(stack):应用场景:java中的方法运行时所占用的空间就是这种结构.队列结构:先进先出 ...
- 当final作用于变量、参数、方法和类时该如何处理
final变量: 对于基本类型使用final:它就是一个常量,数值恒定不变 对于对象引用使用final:使得引用恒定不变,一旦引用被初始化指向一个对象,就无法再把 它改为指向另一个对象.然而,对象自身 ...
- C# Dictionary根据Key排序
using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace Cons ...
- Python3使用PyQt5制作简单的画板/手写板
0.目录 1.前言 2.简单的画板1.0 在定点和移动中的鼠标所在处画一条线 3.简单的画板2.0 在定点和移动中的鼠标所在处画一条线 并将画过的线都保留在窗体上 4.简单的画板3.0 将按住鼠标后移 ...
- linux shell中单引号、双引号和没有引号的区别
单引号: 可以说是所见即所得:即将单引号的内的内容原样输出,或者描述为单引号里面看到的是什么就会输出什么. 双引号: 把双引号内的内容输出出来:如果内容中有命令.变量等,会先把变量.命令解析出结果,然 ...