多线程笔记 - provider-consumer
通过多线程实现一个简单的生产者-消费者案例(笔记).
首先定义一个要生产消费的数据类 :
public class Data {
private String id;
private String name;
public Data(String id, String name) {
this.id = id;
this.name = name;
}
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "Data{" +
"id='" + id + '\'' +
", name='" + name + '\'' +
'}';
}
}
生产者首先需要一个装载数据的容器, 产生的数据, 需要往里面放.
public class Provider implements Runnable {
//1. 装载数据的容器
private BlockingQueue<Data> queue;
//2. 运行标志
private volatile boolean isRunning = true;
//3. 为数据产生id
private static AtomicInteger count = new AtomicInteger();
//4. 随机休眠
private static Random r = new Random();
public Provider(BlockingQueue queue){
this.queue = queue;
}
@Override
public void run() {
while (isRunning){
try {
//随机休眠, 模拟产生数据逻辑耗时
Thread.sleep(r.nextInt(100));
//产生一个数据id, 为了避免多线程产生的id重复, 所以这里使用 AtomicInteger
int id = count.incrementAndGet();
//创建数据
Data data = new Data(Integer.toString(id), "data" + id);
//打印创建日志
System.out.println( Thread.currentThread().getName() + " ++++++ " + id);
//将数据加载到容器中
if(!this.queue.offer(data, 1, TimeUnit.SECONDS)){
System.out.println(id + " 提交失败......");
}
}
catch (InterruptedException e) {
e.printStackTrace();
}
}
}
public void stop(){
this.isRunning = false;
}
public void reset(){
this.isRunning = true;
}
}
消费者, 也需要知道数据存放的容器, 才能从里面拿取数据进行处理.
public class Consumer implements Runnable {
//1. 数据容器, 从中取数据进行消费
private BlockingQueue<Data> queue;
public Consumer(BlockingQueue<Data> queue){
this.queue = queue;
}
//随机休眠
private static Random r = new Random();
@Override
public void run() {
while (true){
try {
//从队列中取数据
Data data = this.queue.take();
//如果获取到的数据为空, 则不进行处理
if(data == null){
continue;
}
//随机休眠, 模拟消费数据逻辑处理耗时
Thread.sleep(r.nextInt(1000));
//打印消费日志
System.out.println( Thread.currentThread().getName() + " ------ " + data.getId());
}
catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
测试方法:
public static void main(String[] args){
//BlockingQueue<Data> queue = new LinkedBlockingQueue<>(16);
BlockingQueue<Data> queue = new ArrayBlockingQueue<>(16);
List<Provider> providers = new ArrayList<>();
for (int i = 0; i < 10; i++) {
Provider p = new Provider(queue);
providers.add(p);
}
List<Consumer> consumers = new ArrayList<>();
for (int i = 0; i < 2; i++) {
Consumer c = new Consumer(queue);
consumers.add(c);
}
ExecutorService pool = Executors.newCachedThreadPool();
for (Provider provider : providers) {
pool.execute(provider);
}
for (Consumer consumer : consumers) {
pool.execute(consumer);
}
try {
Thread.sleep(3000);
}
catch (InterruptedException e) {
e.printStackTrace();
}
for (Provider provider : providers) {
provider.stop();
}
}
这里特意用了一个有界队列, 并且特意设置了生产者多于消费者.
打印日志:

+ 号代表产生了数据, 但是不一定加入到容器里了.
- 号代表消费成功, 数据从队列中删除
这里需要注意一个问题:
生产消费都是有速度的, 也就是说, 如果消费速度小于生产速度
1. 使用有界队列 - 会丢数据, 需要对这部分数据做特殊处理
2. 使用无界队列 - 可能会让电脑宕机, 数据积累越来越多, 导致内存不足, 卡死或者直接死机.
需要对应用场景进行分析, 才能决定使用哪种方式.
多线程笔记 - provider-consumer的更多相关文章
- 2016/1/25 多线程 作业 方法一 继承Thread 方法二 实现Runnable 多线程笔记
/* * 1,尝试定义一个继承Thread类的类,并覆盖run()方法, * 在run()方法中每隔100毫秒打印一句话.*/ package Stream; //方法一 继承Thread 实现多线程 ...
- Java基础知识强化之多线程笔记01:多线程基础知识(详见Android(java)笔记61~76)
1. 基础知识: Android(java)学习笔记61:多线程程序的引入 ~ Android(java)学习笔记76:多线程-定时器概述和使用
- dubbo 学习笔记 -- provider端
服务端的配置文件: provider.xml <?xml version="1.0" encoding="UTF-8"?> <beans ...
- 如何理解springcloud微服务项目中,eureka,provider,consumer它们之间的关系?
eureka负责注册provider和consumer的服务信息 provider负责与数据库进行交互,实现数据持久化,并给consumer提供服务 consumer与前端交互,通过与Eureka同源 ...
- 这份java多线程笔记,你真得好好看看,我还没见过总结的这么全面的
1.线程,进程和多线程 1.程序:指指令和数据的有序集合,其本身没有任何意义,是一个静态的概念 2.进程:指执行程序的一次执行过程,是一个动态的概念.是系统资源分配的单位(注意:很多多线程是模拟出来的 ...
- iOS多线程笔记
在iOS开发中,有三种多线程处理方式: 1. 利用NSThread 2. NSOperation和NSOperationQueue 3. 利用GCD(Grand Central Dispatch) 使 ...
- Java基础知识强化之多线程笔记05:Java中继承thread类 与 实现Runnable接口的区别
1. Java中线程的创建有两种方式: (1)通过继承Thread类,重写Thread的run()方法,将线程运行的逻辑放在其中. (2)通过实现Runnable接口,实例化Thread类. 2. ...
- QT多线程笔记
1.QT多线程涉及到主线程和子线程之间交互大量数据的时候,使用QThread并不方便,因为run()函数本身不能接受任何参数,因此只能通过信号和槽的交互来获取数据,如果只是单方面简单交互数据还过得去, ...
- Java基础知识强化之多线程笔记05:Java程序运行原理 和 JVM的启动是多线程的吗
1. Java程序运行原理: Java 命令会启动Java 虚拟机,启动 JVM,等于启动了一个应用程序,也就是启动了一个进程.该进程会自动启动一个 “主线程” ,然后主线程去调用某个类的 m ...
随机推荐
- MySQL-5.7.29 的安装与配置
解压缩,我这里的解压路径是:D:\Program Files\Java MySQL-5.7.29下载地址:https://www.jianguoyun.com/p/DcKSfd8Q6LnsBxiY8c ...
- 【洛谷4424】[HNOI_AHOI2018]寻宝游戏(我也不知道括号里该写啥)
题目 洛谷 4424 分析 感觉思路比较神仙. 对于按位与和按位或两种运算,显然每一位是独立的,可以分开考虑. 对于某一位,「与 \(0\)」会将这一位变成 \(0\),「或 \(1\)」会将这一位变 ...
- 每天敲一点code
下面这段代码摘自 <C#并发编程经典实例> 并行LINQ static IEnumerable<int> MultiplyBy2(IEnumerable<int> ...
- Idea破解至2089年
我是用的版本是2018.3.6,别的朋友使用的是2019的某个版本,不过关都不影响破解 下载jar包:链接:https://pan.***baidu.***com/s/1aRR0***2YNI9jew ...
- livego+obs+flv.js 搭建视频直播
一.流程 主播通过 obs软件通过直播 ->推流到->直播服务器 客户通过浏览器 访问站点->flv.js拉取直播服务器视频流并播放 二.环境 centos7 直播服务器 https ...
- Ubuntu固定多个静态ip
步骤: 1.sudo vim /etc/network/interfaces 加入下列内容 auto eth0#此处查看自己的ip信息是eth0还是eth1等等 iface eth0 inet sta ...
- 机器学习李航——Adaboost课本例题实现
例8.1Adaboost的例子 注意求D3或者D4的时候只需要把w替换一下就行,记得还得改阈值.这个代码算个半自动的,因为还需要手动改一下. import numpy as np def getA(e ...
- windows丢失文件的恢复技巧
这几天在使用STVD调试程序的时候,突然跳出来一个“共享冲突”错误,当时并没有在意,点确定后赶紧CTRL+S,然后就一直死在那里了... 结束任务,重启STVD,提示找不到main.c,到此也不以为然 ...
- 【干货】零基础30分钟让你拥有一个完整属于自己的短视频APP系统
目录 一.附言: 1 二.购买域名和购买服务器: 2 三.搭建服务器环境: 5 四.配置APP前端部分: 8 1.工具以及文件准备: 9 2.配置后端接口地址 11 3.配置APP启动图和启动图标 ...
- linux下面误删root里面的文件夹 恢复方法
手残吧 /root/ 里面的文件删除了. .mkdir /root cp -a /etc/skel/.[!.]* /root 主要是吧 /etc/skel/里面的文件拷贝回去就行了~~~哈.. 转自: ...