`

Java多线程-生产者于消费者

阅读更多
闲来无事,好久没有接触Java的高级东西了,害怕后期会忘记而寻找资料无门,所以写一个最简单的并且典型的多线程入门例子

/**
 * 
 * @author Administrator
 */
public class BigCup {

	/**
	 * 容器的大小
	 */
	private int size;

	/**
	 * 容器当前的数量
	 */
	private int count;

	public BigCup(int i) {
		this.size = i;
	}

	public void push(String name) {
		System.out.println(name + "放进去一个面包");
		count++;
	}

	public void pop(String name) {
		System.out.println(name + "拿出来一个面包");
		count--;
	}

	public int getSize() {
		return size;
	}

	public void setSize(int size) {
		this.size = size;
	}

	public int getCount() {
		return count;
	}

	public void setCount(int count) {
		this.count = count;
	}

}

public class Producer implements Runnable {

	/**
	 * 生产者的名字
	 */
	private String name;

	/**
	 * 拿着当前容器的引用
	 */
	private BigCup bc;

	public Producer(String name, BigCup bc) {
		super();
		this.name = name;
		this.bc = bc;
	}

	@Override
	public void run() {
		Thread.currentThread().setName(name);
		//Thread.currentThread().setPriority(1);
		set();
	}

	@SuppressWarnings("static-access")
	public void set() {
		for (int i = 0; i < 100; i++) {
			synchronized (bc) {
				while (bc.getCount() >= bc.getSize()) {
					try {
						System.out.println("满了,请取东西...");
						bc.wait();
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}

				try {
					Thread.currentThread().sleep(300);

				} catch (InterruptedException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
				bc.push(name);

				bc.notify();
				Thread.yield();
			}
		}
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public BigCup getBc() {
		return bc;
	}

	public void setBc(BigCup bc) {
		this.bc = bc;
	}

}
public class Consumer implements Runnable {

	/**
	 * 消费者的名字
	 */
	private String name;

	/**
	 * 拿着当前容器的引用
	 */
	private BigCup bc;

	public Consumer(String name, BigCup bc) {
		super();
		this.name = name;
		this.bc = bc;
	}

	@Override
	public void run() {
		Thread.currentThread().setName(name);
		//Thread.currentThread().setPriority(8);
		get();
	}

	@SuppressWarnings("static-access")
	public void get() {
		for (int i = 0; i < 100; i++) {
			synchronized (bc) {
				while (bc.getCount() < 1) {
					try {
						System.out.println("空了,请放东西...");
						bc.wait();
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}
				try {
					Thread.currentThread().sleep(300);

				} catch (InterruptedException e) {
					e.printStackTrace();
				}
				bc.pop(name);

				bc.notify();
				Thread.yield();
			}
		}
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public BigCup getBc() {
		return bc;
	}

	public void setBc(BigCup bc) {
		this.bc = bc;
	}

}

测试类:
public class Test {

	public static void main(String[] args) {

		BigCup bc = new BigCup(10);

		Consumer c = new Consumer("旺旺", bc);

		Producer p = new Producer("淘宝", bc);

		Thread t1 = new Thread(p);
		t1.start();
		
		Thread t2 = new Thread(c);
		t2.start();
		
		Thread t3 = new Thread(c);
		t3.start();
		Thread t4 = new Thread(c);
		t4.start();
		Thread t5 = new Thread(c);
		t5.start();

	}
}

打印的结果如下
空了,请放东西...
空了,请放东西...
淘宝放进去一个面包
旺旺拿出来一个面包
空了,请放东西...
空了,请放东西...
淘宝放进去一个面包
淘宝放进去一个面包
淘宝放进去一个面包
满了,请取东西...
旺旺拿出来一个面包
旺旺拿出来一个面包
旺旺拿出来一个面包
空了,请放东西...
空了,请放东西...
淘宝放进去一个面包
淘宝放进去一个面包
旺旺拿出来一个面包
旺旺拿出来一个面包
空了,请放东西...
空了,请放东西...



同步代码块:
    1.被synchronized包围的代码块称为同步代码块
    2.synchronized代码块需要接收一个参数,该参数是一个对象,可以称为该代码块的锁对象(又称同步锁)。
    3.若某线程一旦持有锁对象的监视器,其它线程即不能同时持有该锁对象的监视器,从而实现线程基于特定锁对象的互斥。
    4.注意:这里表示的是线程独占锁对象的监视器,而非锁对象,其它线程仍可以访问该锁对象的方法和属性。
    5.多个线程以竞争的方式争夺锁对象的监视器,线程执行完该代码块,即释放锁对象的监视器。

同步实例方法:
    就是在方法前面加上关键字synchronized,即实现了方法的同步,相当于整个方法是一个同步代码块,其锁对象是this

同步静态方法:
    就是在静态方法前面加上关键字synchronized,也是将整个方法看做一个同步代码块,其锁对象较特别,是将所属类作为锁对象

Java提供了一个精心设计的线程间通信机制即wait-notify机制,通过该机制可以实现线程协作,wait-notify机制是通过使用wait()、notify()和notifyAll()三个方法来实现的,这三个方法均定义在Object类中,是final修饰的实例方法,这三个方法必须在synchronized代码中调用,而且只有锁对象才能调用这三个方法。即持有锁对象监视器的线程才能调用锁对象的这三个方法;

wait()方法:
    调用该方法的线程退出监视器并进入等待状态,直到其他线程进入相同的监视器并调用notify( )方法。

notify()方法:
    通知等待(该方法所属对象)监视器的(多个)线程中的一个结束等待;即唤醒一个等待(当前线程所持有)监视器的线程。

notifyAll()方法:
    通知等待(该方法所属对象)监视器的所有线程结束等待;即唤醒所有等待(当前线程所持有)监视器的线程。


后记,还有几个方法没有涉及到,时间有限,后期会慢慢的同步。


分享到:
评论

相关推荐

    Java多线程-生产者与消费者问题

    ### Java多线程-生产者与消费者问题 #### 一、生产者与消费者问题概览 **1.1 概要** 生产者与消费者问题是计算机科学中一个多线程同步的经典问题。它描述了两个线程如何共享有限资源的场景:一个是生产者...

    Java多线程实现生产者消费者

    本示例中的“生产者-消费者”模型是一种经典的多线程问题,它模拟了实际生产环境中的资源分配与消耗过程。下面我们将详细探讨如何在Java中实现这个模型。 首先,我们要理解生产者-消费者模型的基本概念。在这个模型...

    java多线程实现生产者和消费者

    在并发编程中,"生产者-消费者"模式是一种经典的解决问题的范式,用于协调两个或更多线程间的协作,其中一部分线程(生产者)生成数据,另一部分线程(消费者)消费这些数据。 生产者-消费者模型的核心在于共享资源...

    java多线程经典模型生产者消费者

    java多线程经典模型生产者消费者java多线程经典模型生产者消费者java多线程经典模型生产者消费者java多线程经典模型生产者消费者java多线程经典模型生产者消费者java多线程经典模型生产者消费者java多线程经典模型...

    Java多线程 生产者-消费者模式

    生产者-消费者模式是一种经典的多线程设计模式,用于解决数据共享问题,尤其是在一个线程生产数据而另一个线程消费数据的情况下。在这个模式中,生产者负责生成数据并放入共享的数据结构(如队列),而消费者则从这...

    java-线程-生产者消费者实例

    生产者-消费者实例可以很好帮助初学java的同学们理解和学习多线程的知识

    java多线程例子-生产者消费者

    在本示例中,“java多线程例子-生产者消费者”旨在展示如何利用多线程来实现生产者和消费者模式。这种模式是并发编程中的经典设计模式,用于协调生产数据和消费数据的两个不同线程。 生产者消费者模式的基本概念是...

    java多线程(生产者与消费者)

    生产者消费者模式是一种设计模式,它描述了两个不同类型的线程如何协作完成工作:生产者负责生成数据,而消费者负责消耗这些数据。为了使这个模式在多线程环境下安全运行,我们需要使用Java中的同步机制。 1. **...

    Java线程间的通信----生产者消费者模型

    在Java编程中,线程间的通信是多线程编程中的一个重要概念,特别是在处理并发和协作任务时。生产者消费者模型是一种经典的...在实际项目中,生产者消费者模型被广泛应用于缓存管理、数据库批量操作、消息队列等场景。

    【IT十八掌徐培成】Java基础第08天-04.多线程-生产者-消费者.zip

    通过学习和实践"生产者-消费者"模型,开发者能够更好地理解和掌握Java多线程的精髓,为构建高并发、高效率的应用打下坚实的基础。在实际项目中,这种模型常被用于实现缓存管理、消息队列以及数据库连接池等场景。

    java实现多线程经典模型生产者消费

    java实现多线程经典模型生产者消费java实现多线程经典模型生产者消费java实现多线程经典模型生产者消费java实现多线程经典模型生产者消费java实现多线程经典模型生产者消费java实现多线程经典模型生产者消费java实现...

    java多线程实现生产者消费者关系

    在实际应用中,我们常常会遇到一种典型的多线程问题——生产者消费者模型。这个模型描述了两种类型的线程:生产者线程负责创建或生产资源,而消费者线程则负责消耗这些资源。在Java中,我们可以利用同步机制来实现...

    【IT十八掌徐培成】Java基础第08天-05.多线程-生产者-消费者2.zip

    在Java中实现生产者-消费者模型,主要依赖于线程同步机制,包括`wait()`, `notify()` 和 `notifyAll()` 方法,它们都是在 `Object` 类中定义的。这些方法配合 `synchronized` 关键字使用,可以确保线程安全地访问...

    java多线程实现生产者消费者问题

    用java多线程,实现生产者消费者同步和互斥的问题,操作系统中的经典问题

    线程同步--生产者消费者问题

    总的来说,理解和掌握生产者消费者问题及其解决策略是Java多线程编程的重要部分,这不仅有助于编写高效、可靠的并发代码,也是提升软件设计能力的关键一步。通过不断实践和学习,开发者能够更好地应对复杂的并发场景...

    JAVA实现线程间同步与互斥生产者消费者问题

    本项目通过一个生产者消费者问题的实例,展示了如何在Java中实现线程间的同步与互斥。 生产者消费者问题是经典的并发问题之一,它涉及到两个类型的线程:生产者和消费者。生产者负责生成数据(产品),而消费者则...

    典型的多线程--生产和消费

    "典型的多线程--生产和消费"这个主题,通常指的是生产者-消费者问题,这是一种经典的设计模式,用于解决如何在一个数据结构(如队列)中高效地存储和检索数据,同时避免生产者过于快速地生成数据而消费者无法及时...

    多线程简易实现生产者消费者模式

    生产者消费者模式是一种经典的多线程同步问题解决方案,它源于现实世界中的生产流水线,用于描述生产者(Producer)和消费者(Consumer)之间的协作关系。在这个模式中,生产者负责生成产品并放入仓库,而消费者则从...

Global site tag (gtag.js) - Google Analytics