多线程_03(生产者与消费者、多线程模拟迅雷用3个线程下载100M资源过程、2个线程交叉顺序打印)

本文主要是介绍多线程_03(生产者与消费者、多线程模拟迅雷用3个线程下载100M资源过程、2个线程交叉顺序打印),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

生产者消费者、案例

  • 1.生产者消费者模式概述
  • 2.生产者消费者案例
  • 3.多线程相关题目案例
    • 1. 用多线程代码来模拟,迅雷用3个线程下载100M资源的过程。
    • 2. 创建2个线程,打印从0到99这100个数字,要求线程交叉顺序打印。

1.生产者消费者模式概述

生产者消费者模式是一个十分经典的多线程协作的模式,弄懂生产者消费者问题能够让我们对多线程编程的理解更加深刻。所谓生产者消费者问题,实际上主要是包含了两类线程:

  • 一类是生产者线程用于生产数据
  • 一类是消费者线程用于消费数据

为了解耦生产者和消费者的关系,通常会采用共享的数据区域,就像是一个仓库

  • 生产者生产数据之后直接放置在共享数据区中,并不需要关心消费者的行为
  • 费者只需要从共享数据区中去获取数据,并不需要关心生产者的行为
    在这里插入图片描述
    void wait():导致当前线程等待,直到另一个线程调用该对象的notify方法或notifyAll()方法
    void notify():唤醒正在等待对象监视器的单个线程
    void notifyAll():唤醒正在等待对象监视器的所有线程

2.生产者消费者案例

生产者消费者案例中包含的类:

  • 奶箱类(Box):定义一个成员变量,表示第x瓶奶,提供存储牛奶和获取牛奶的操作
  • 生产者类(Producer):实现Runnable接口,重写run()方法,调用存储牛奶的操作
  • 消费者类(Customer):实现Runnable接口,重写run()方法,调用获取牛奶的操作
  • 测试类(BoxDemo):里面有main方法,main方法中的代码步骤如下
    [1] 创建奶箱对象,这是共享数据区域
    [2] 创建生产者对象,把奶箱对象作为构造方法参数传递,因为在这个类中要调用存储牛奶的操作
    [3] 创建消费者对象,把奶箱对象作为构造方法参数传递,因为在这个类中要调用获取牛奶的操作
    [4] 创建2个线程对象,分别把生产者对象和消费者对象作为构造方法参数传递
    [5] 启动线程

Customer.java

public class Customer implements Runnable {private Box b;public Customer(Box b) {this.b = b;}@Overridepublic void run() {while (true) {//死循环获取奶,有就拿b.get();}}
}

Producer.java

public class Producer implements Runnable{private Box b;//成员变量Box,用于和Box交互public Producer(Box b) {//代参成员方法this.b = b;}@Overridepublic void run() {for (int i = 1; i < 5; i++) {b.put(i);}}
}

BoxDemo.java

public class BoxDemo {public BoxDemo() {}public static void main(String[] args) {//创建奶箱对象,这是共享数据区域Box b = new Box();
//创建生产者对象,把奶箱对象作为构造方法参数传递,因为在这个类中要调用存储牛奶的操作Producer p = new Producer(b);
//创建消费者对象,把奶箱对象作为构造方法参数传递,因为在这个类中要调用获取牛奶的操作Customer c = new Customer(b);
//创建2个线程对象,分别把生产者对象和消费者对象作为构造方法参数传递Thread t1 = new Thread(p);Thread t2 = new Thread(c);t1.start();t2.start();}
}

Box.java

public class Box {//定义一个成员变量,表示有几瓶奶private int milk;
//定义一个成员变量,表示奶箱状态private boolean state=false;public void put(int milk) {//如果有牛奶,等待消费if (state){try {wait();} catch (InterruptedException e) {e.printStackTrace();}}//如果没有牛奶,就生产牛奶this.milk = milk;System.out.println("送奶工将第" + this.milk + "瓶奶放入奶箱");//生产完毕,修改奶箱状态state=false;}public synchronized void get() {//若不加synchronized,会报IllegalMonitorStateException错误。其实质是wait();应该在锁(监视器)环境下使用//如果没有牛奶,等待生产if (!state){try {wait();} catch (InterruptedException e) {e.printStackTrace();}}System.out.println("用户拿到第" + this.milk + "瓶奶");//消费完毕之后,修改奶箱状态state=false;}
}

结果输出:(因为没有加notifyAll,未唤醒)

送奶工将第1瓶奶放入奶箱
用户拿到第1瓶奶

唤醒改进:
Customer.java
Producer.java
BoxDemo.java
Box.java

public class Box {//定义一个成员变量,表示有几瓶奶private int milk;
//定义一个成员变量,表示奶箱状态private boolean state=false;public synchronized void put(int milk) {//如果有牛奶,等待消费if (state){try {wait();} catch (InterruptedException e) {e.printStackTrace();}}//如果没有牛奶,就生产牛奶this.milk = milk;System.out.println("送奶工将第" + this.milk + "瓶奶放入奶箱");//生产完毕,修改奶箱状态state=true;notifyAll();}public synchronized void get() {//若不加synchronized,会报IllegalMonitorStateException错误。其实质是wait();应该在锁(监视器)环境下使用//如果没有牛奶,等待生产if (!state){try {wait();} catch (InterruptedException e) {e.printStackTrace();}}System.out.println("用户拿到第" + this.milk + "瓶奶");//消费完毕之后,修改奶箱状态state=false;notifyAll();}
}

结果输出:

送奶工将第1瓶奶放入奶箱
用户拿到第1瓶奶
送奶工将第2瓶奶放入奶箱
用户拿到第2瓶奶
送奶工将第3瓶奶放入奶箱
用户拿到第3瓶奶
送奶工将第4瓶奶放入奶箱
用户拿到第4瓶奶
送奶工将第5瓶奶放入奶箱
用户拿到第5瓶奶

3.多线程相关题目案例

1. 用多线程代码来模拟,迅雷用3个线程下载100M资源的过程。

每个线程每次,一次下载1兆(M)资源,直到下载完毕,即剩余的待下载资源大小为0(用一个整数表示资源大小,每次个线程每次下载多少兆(M), 剩余待下载资源就减少多少兆(M),考虑多线程的数据安全问题)

Down.java

/*** @Author:gaoyuan* @Description:* @DateTime:2021/1/22 19:55**/
public class Down implements Runnable {private int resourse = 10;//简写private Lock lock = new ReentrantLock();@Overridepublic void run() {while (true) {try {lock.lock();if (resourse >= 0) {try {Thread.sleep(200);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName() + "当前正在下载第" + resourse + "M资源");resourse--;}} finally {lock.unlock();}}}
}

DownTest.java

public class DownTest {public static void main(String[] args) {Down d = new Down();Thread t1 = new Thread(d,"线程A");Thread t2 = new Thread(d,"线程B");Thread t3 = new Thread(d,"线程C");t1.start();t2.start();t3.start();}
}

结果输出:

线程B当前正在下载第10M资源
线程B当前正在下载第9M资源
线程B当前正在下载第8M资源
线程B当前正在下载第7M资源
线程B当前正在下载第6M资源
线程B当前正在下载第5M资源
线程B当前正在下载第4M资源
线程B当前正在下载第3M资源
线程B当前正在下载第2M资源
线程B当前正在下载第1M资源
线程B当前正在下载第0M资源

2. 创建2个线程,打印从0到99这100个数字,要求线程交叉顺序打印。

要求使用线程间通信实现。即我打印一个数字,你打印一个数字,两线程协作完成。

比如:
线程1: 0
线程2: 1
线程1: 2
线程2: 3
线程1: 4
线程2: 5

方法一:
CrossPrint.java

/*** @Author:gaoyuan* @Description:交叉打印* @DateTime:2021/1/22 20:35**/
public class CrossPrint implements Runnable {private int number = 1;@Overridepublic void run() {while (true) {//指代的为CrossPrint,因为使用的是implements方式。若使用继承Thread类的方式,慎用thissynchronized (this) {
//唤醒另外一个线程,注意是this的方法,而不是ThreadnotifyAll();//唤醒所有进程,但因为synchronized上锁,只有先启动的线程1先执行,线程2进不来。try {Thread.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}if (number <= 10) {//简写System.out.println(Thread.currentThread().getName() + ":" + number);number++;try {//放弃资源,等待wait();} catch (InterruptedException e) {e.printStackTrace();}}}}}
}

CrossPrintTest.java

public class CrossPrintTest {public static void main(String[] args) {CrossPrint cp = new CrossPrint();Thread t1 = new Thread(cp,"线程1");Thread t2 = new Thread(cp,"线程2");t1.start();t2.start();}
}

结果输出;

线程1:1
线程2:2
线程1:3
线程2:4
线程1:5
线程2:6
线程1:7
线程2:8
线程1:9
线程2:10

方法二:
Print1.java

public class Print1 implements Runnable {private PrintSwap p;//设置与PrintSwap有所联系的成员变量public Print1(PrintSwap p) {this.p = p;}@Overridepublic void run() {while (true) {p.print();//调用PrintSwap类中的print方法}}
}

Print2.java

public class Print2 implements Runnable{private PrintSwap p;public Print2(PrintSwap p) {this.p = p;}@Overridepublic void run(){while (true){p.print();//调用PrintSwap类中的print方法}}
}

PrintSwap.java

public class PrintSwap {private int number = 10;//简写// private boolean flag = false;public synchronized void print() {while (number > 0) {notify();System.out.println(Thread.currentThread().getName() + ":" + number);number--;try {wait();} catch (InterruptedException e) {e.printStackTrace();}}}
}

PrintDemo.java

public class PrintDemo {public PrintDemo() {}public static void main(String[] args) {PrintSwap ps = new PrintSwap();Print1 p1 = new Print1(ps);Print2 p2 = new Print2(ps);Thread t1 = new Thread(p1,"线程1");Thread t2 = new Thread(p2,"线程2");t1.start();t2.start();}
}

结果输出;

线程2:10
线程1:9
线程2:8
线程1:7
线程2:6
线程1:5
线程2:4
线程1:3
线程2:2
线程1:1

这篇关于多线程_03(生产者与消费者、多线程模拟迅雷用3个线程下载100M资源过程、2个线程交叉顺序打印)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/535164

相关文章

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

作业提交过程之HDFSMapReduce

作业提交全过程详解 (1)作业提交 第1步:Client调用job.waitForCompletion方法,向整个集群提交MapReduce作业。 第2步:Client向RM申请一个作业id。 第3步:RM给Client返回该job资源的提交路径和作业id。 第4步:Client提交jar包、切片信息和配置文件到指定的资源提交路径。 第5步:Client提交完资源后,向RM申请运行MrAp

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

常用的jdk下载地址

jdk下载地址 安装方式可以看之前的博客: mac安装jdk oracle 版本:https://www.oracle.com/java/technologies/downloads/ Eclipse Temurin版本:https://adoptium.net/zh-CN/temurin/releases/ 阿里版本: github:https://github.com/

usaco 1.2 Transformations(模拟)

我的做法就是一个一个情况枚举出来 注意计算公式: ( 变换后的矩阵记为C) 顺时针旋转90°:C[i] [j]=A[n-j-1] [i] (旋转180°和270° 可以多转几个九十度来推) 对称:C[i] [n-j-1]=A[i] [j] 代码有点长 。。。 /*ID: who jayLANG: C++TASK: transform*/#include<

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss

cross-plateform 跨平台应用程序-03-如果只选择一个框架,应该选择哪一个?

跨平台系列 cross-plateform 跨平台应用程序-01-概览 cross-plateform 跨平台应用程序-02-有哪些主流技术栈? cross-plateform 跨平台应用程序-03-如果只选择一个框架,应该选择哪一个? cross-plateform 跨平台应用程序-04-React Native 介绍 cross-plateform 跨平台应用程序-05-Flutte

顺序表之创建,判满,插入,输出

文章目录 🍊自我介绍🍊创建一个空的顺序表,为结构体在堆区分配空间🍊插入数据🍊输出数据🍊判断顺序表是否满了,满了返回值1,否则返回0🍊main函数 你的点赞评论就是对博主最大的鼓励 当然喜欢的小伙伴可以:点赞+关注+评论+收藏(一键四连)哦~ 🍊自我介绍   Hello,大家好,我是小珑也要变强(也是小珑),我是易编程·终身成长社群的一名“创始团队·嘉宾”

hdu4431麻将模拟

给13张牌。问增加哪些牌可以胡牌。 胡牌有以下几种情况: 1、一个对子 + 4组 3个相同的牌或者顺子。 2、7个不同的对子。 3、13幺 贪心的思想: 对于某张牌>=3个,先减去3个相同,再组合顺子。 import java.io.BufferedInputStream;import java.io.BufferedReader;import java.io.IOExcepti

2. 下载rknn-toolkit2项目

官网链接: https://github.com/airockchip/rknn-toolkit2 安装好git:[[1. Git的安装]] 下载项目: git clone https://github.com/airockchip/rknn-toolkit2.git 或者直接去github下载压缩文件,解压即可。