博客园的园友们好,看博客园上各位大佬的文章,已陪伴了我程序员职业的三年,

如今自己同样希望能把自己从小白到菜鸟的成长过程分享给大家。不定期更新!!!

首先我本人智商不高,理解问题十分吃力,完全不属于天才的行列,因此学习每一个知识

都喜欢刨根问底,结合生活,彻彻底底理解知识的本质!

进入正题,这篇文章,主要站在一个初学者的角度,结合经典的“生产者消费者模型”,写一个java多线程例子!

首先解释几个概念:

  1、#进程:通俗的讲,就是一个程序一次执行的过程。是系统进行资源分配和调度的一个独立单位。

  2、#线程:一个进程的生命周期,由一个或若干个线程完成。是CPU调度和分派的基本单位。

  3、#并行:同一时间点或者时间段,可以处理超过一个任务的能力。

  eg:你正在lol,女朋友来电话,于是你单手操作或者侧头夹住手机,既聊天又打团,这就是你自己并行处理了,

撩妹和打游戏。这个过程,你就类似于cpu,前提你是多核。

  4、#并发:主要针对多线程提出的概念。可以在一个时间段,交替执行不同事情的能力。

  eg:你正在lol,女朋友来电话,于是你挂机去接电话,然后打完电话回来面对腾讯的裁决。或者你等着打完团,然后

面对女友的生气。这就是并发,你交替执行了不同事情。这个过程,你也类似于cpu,可以不用多核。

二、简单代码实现多线程

  2.1模型图

注释:张全蛋经过自己的努力,进入富士康工厂工作,主要负责生产时下流行的iPhoneXs,然后广大果粉在库存充足的情况下购买iPhoneXs。

这个过程共涉及以下角色和过程:

  1、生产者(张全蛋)。

  2、消费者(广大果粉)。

  3、产品(iPhoneXs)。

  4、生产产品,购买产品。

2.2 程序展示

  2.2.1  产品类

由于我们此次实验过程,主要涉及生产产品,所以我们可以忽略产品本身具有的属性和方法。

 package com.dcits.weipt;

 /**
* 产品实体类
* 因我们此次实验是针对生产产品
* 所以我们忽略产品本身所具有的方法和属性
* @author weipt
* @date 20180915*/
public class Product { }

  2.2.2 工厂类

工厂类用于生产产品,需要注意的是,在java中,生产一个产品,就是new一个产品类的实体对象。

 package com.dcits.weipt;

 import java.util.ArrayList;
import java.util.List; /**
* 生产产品的工厂类
* 生产一个产品则意味着new一个产品对象
* 消费一个产品则意味着remove一个对象
* @author weipt
* @date 20180915*/ public class ProductFactory {
private List<Product> list = new ArrayList<Product>(); //利用list保存对象 /**
* 工厂生产产品的方法
* 单位时间生产3个
* */
public void makeProduct() {
for(int i=0;i<3;i++) {
list.add(new Product());
}
} /**
* 工厂消费产品的方法
* 单位时间消费1个*/
public void moveProduct() {
list.remove(0);
} /**
* 获取产品个数*/
public int getNum() {
return list.size();
} }

  2.2.3 生产者

用于不断调用工厂类,生产产品。每当我们生产一次产品,需要调用notify/notifyAll通知或唤醒消费者来购买。

但当我们生产超过max,需要调用wait,等待消费者购买,减少库存。

 package com.dcits.weipt;

 /**
*生产者实体类
* @author weipt
* @date 20180915*/ public class Producer implements Runnable {
private final int MAX_PRODUCT = 30;
private ProductFactory pf; public Producer(ProductFactory pfIn) {
this.pf = pfIn;
} @Override
public void run() {
produce();
} /**
* 生产者生产,产品
*/
public void produce() {
while (true) {
synchronized (pf) {
if (pf.getNum() >= MAX_PRODUCT) {
try {
System.out.println("warnning! 库存已满,请稍微再生产!");
pf.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}else {
pf.makeProduct();
System.out.println("P---》生产者生产了第【" + pf.getNum() + "】个产品");
pf.notifyAll();
}
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
}

  2.2.4 消费者

用于不断调用工厂类,购买产品。每当我们购买一次产品,需要调用notify/notifyAll通知或唤醒生产者去生产。

但当库存小于min,需要调用wait,等待生产者,增加库存。

 package com.dcits.weipt;

 /**
* 消费者实体类
* @author weipt
* @date 20180915*/ public class Consumer implements Runnable {
private final int MIN_PRODUCT = 0; //产品最小值
private ProductFactory pf; //产品工厂对象 /**
* 通过构造方法获取产品对象*/
public Consumer(ProductFactory pfIn) {
this.pf = pfIn;
} @Override
public void run() {
consume();
} /**
* 消费者从库存中取产品
*/
public void consume() {
while (true) {
synchronized (pf) { //因为生产者和消费者都是对产品操作,所以对产品进行加锁
if (pf.getNum() <= MIN_PRODUCT) {
try {
System.out.println("warnning! 库存已空,请稍微再取!");//注意顺序,要放到wait之前
pf.wait(); //等待其他线程操作,直到收到其他线程的notify
//此处我没有写notify,因为库存缺货,就只能等生产者生产,唤醒其他消费者没有用
} catch (InterruptedException e) {
e.printStackTrace();
}
}else {
System.out.println("C---》消费者取走了第【" + pf.getNum() + "】个产品");
pf.moveProduct(); //产品出库
pf.notifyAll(); //通知生产者可以继续生产
}
try {
Thread.sleep(1000); //防止日志打印过多,没有可观性
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
} }

  2.2.5 开始工作

在main方法中,开启消费者和生产者线程。

 package com.dcits.weipt;

 /**
* 公司领导
* 用于让整个系统运作起来*/ public class HelloThread{
public static void main(String[] args) {
ProductFactory pf = new ProductFactory();
Producer p = new Producer(pf);
Consumer c = new Consumer(pf);
Thread pt = new Thread(p);
Thread ct = new Thread(c);
pt.start(); //我们可以开启多个消费者或者多个生产者
ct.start();
}
}

到这里,这篇文章就彻底结束了。

注:如需索要编译好的项目源码可关注公众号mht18391859179(扫描下方二维码),回复:  免费领取

如果需要交流或者指正,可通过上述公众号,或者email:wpt191@163.com与本人联系。

笨鸟先飞,终生学习

特别鸣谢:

  1、感谢胡**,同志的交流与指导。

  2、感谢博客园,csdn,知乎等大牛文章的启迪。

结合生活,剖析《生产者消费者模型》-java多线程(一)的更多相关文章

  1. 生产者消费者模型Java实现

    生产者消费者模型 生产者消费者模型可以描述为: ①生产者持续生产,直到仓库放满产品,则停止生产进入等待状态:仓库不满后继续生产: ②消费者持续消费,直到仓库空,则停止消费进入等待状态:仓库不空后,继续 ...

  2. 生产者消费者模型java

    马士兵老师的生产者消费者模型,我感觉理解了生产者消费者模型,基本懂了一半多线程. public class ProducerConsumer { public static void main(Str ...

  3. 生产者消费者模型-Java代码实现

    什么是生产者-消费者模式 比如有两个进程A和B,它们共享一个固定大小的缓冲区,A进程产生数据放入缓冲区,B进程从缓冲区中取出数据进行计算,那么这里其实就是一个生产者和消费者的模式,A相当于生产者,B相 ...

  4. Java里的生产者-消费者模型(Producer and Consumer Pattern in Java)

    生产者-消费者模型是多线程问题里面的经典问题,也是面试的常见问题.有如下几个常见的实现方法: 1. wait()/notify() 2. lock & condition 3. Blockin ...

  5. Java多线程15:Queue、BlockingQueue以及利用BlockingQueue实现生产者/消费者模型

    Queue是什么 队列,是一种数据结构.除了优先级队列和LIFO队列外,队列都是以FIFO(先进先出)的方式对各个元素进行排序的.无论使用哪种排序方式,队列的头都是调用remove()或poll()移 ...

  6. Java多线程之~~~使用Exchanger在线程之间交换数据[这个结合多线程并行会有解决很多问题]生产者消费者模型

    http://blog.csdn.net/a352193394/article/details/39503857  Java多线程之~~~使用Exchanger在线程之间交换数据[这个结合多线程并行会 ...

  7. Java多线程-并发协作(生产者消费者模型)

    对于多线程程序来说,不管任何编程语言,生产者和消费者模型都是最经典的.就像学习每一门编程语言一样,Hello World!都是最经典的例子. 实际上,准确说应该是“生产者-消费者-仓储”模型,离开了仓 ...

  8. java多线程:线程间通信——生产者消费者模型

    一.背景 && 定义 多线程环境下,只要有并发问题,就要保证数据的安全性,一般指的是通过 synchronized 来进行同步. 另一个问题是,多个线程之间如何协作呢? 我们看一个仓库 ...

  9. java 线程池、多线程实战(生产者消费者模型,1 vs 10) 附案例源码

    导读 前二天写了一篇<Java 多线程并发编程>点我直达,放国庆,在家闲着没事,继续写剩下的东西,开干! 线程池 为什么要使用线程池 例如web服务器.数据库服务器.文件服务器或邮件服务器 ...

随机推荐

  1. SpringBoot&Shiro实现用户认证

    SpringBoot&Shiro实现用户认证 实现思路 思路:实现认证功能主要可以归纳为3点 1.定义一个ShiroConfig配置类,配置 SecurityManager Bean , Se ...

  2. A01 React+Antdesign 开发环境准备

    B站教程视频 https://www.bilibili.com/video/av38372336?from=search&seid=1131449710389099812 1 安装node.j ...

  3. Java多态实现的机制

    Java提供了编译时多态和运行时多态两种多态机制.前者是通过方法重载实现的,后者是通过方法的覆盖实现的. 在方法覆盖中,子类可以覆盖父类的方法,因此同类的方法会在父类与子类中有着不同的表现形式. 在J ...

  4. C++ const用法,看这一篇就够了!

    本文主要介绍const修饰符在C++中的主要用法,下面会从两个方面进行介绍:类定义中使用const.非类定义中使用const 1. 非类定义中使用const 非类定义中使用const是指:在除了类定义 ...

  5. JDK8内存模型—消失的PermGen

    一.JVM 内存模型 根据 JVM 规范,JVM 内存共分为虚拟机栈.堆.方法区.程序计数器.本地方法栈五个部分. 1.虚拟机栈:每个线程有一个私有的栈,随着线程的创建而创建.栈里面存着的是一种叫“栈 ...

  6. Mysql(Mariadb)数据库之Information Schema 库中GLOBAL_VARIABLES表 and SESSION_VARIABLES 表分析

    Information Schema GLOBAL_VARIABLES and SESSION_VARIABLES Tables The Information Schema GLOBAL_VARIA ...

  7. SpringBoot是如何实现自动配置的?--SpringBoot源码(四)

    注:该源码分析对应SpringBoot版本为2.1.0.RELEASE 1 前言 本篇接 助力SpringBoot自动配置的条件注解ConditionalOnXXX分析--SpringBoot源码(三 ...

  8. 免ROOT卸载手机自带软件详细教程

    一.准备条件 1.电脑一台 2.手机一部 3.WiFi 二.下载所需资源 微信扫码进入搜索,选择安卓软件卸载工具 根据图中提示,按照自己的系统进行下载 三.下载完后解压(以Windows为例),解压后 ...

  9. javaScript 基础知识汇总 (十)

    1.New Function 语法:let func = new Function ([arg1[, arg2[, ...argN]],] functionBody) //无参数示例: let say ...

  10. Java-字符输入输出(新手)

    参考手册: BufferedReader BufferedWriter: 关键字: close() 关闭流,先刷新.    newLine() 写一行行分隔符.    write() 写一个字符    ...