事关Training2中Task4,想看看经典的两个进程并行会是什么样子

题目概述

实现简单的生产者-消费者模型

  • Tray托盘容量为1;托盘满时不能放入,空时不能取货
  • Producer生产者共需生产10个货物;每生产一个货物后会立刻尝试放入,放入成功前不会继续生产,货物按照从1-10编号;成功放入货物后需要休息0-100ms
  • Consumer消费者共需消费10个货物;只能在托盘中取货。

输出:

  • Producer放入货物时输出: "Producer put:" + 货物编号
  • Consumer取出货物时输出: "Consumer get:" + 货物编号

基本代码

为了精确定轨当前代码运行位置,在多出放置System.out.println()并表上颜色;

为了方便将10改成3

public class Task4 {
public static Tray TRAY = new Tray();
public static void main(String[] args) {
Thread producer = new Producer();
Thread consumer = new Thread(new Consumer());
producer.start();
System.out.println("\033[32m" + "==== Depart ====" + "\033[0m");
consumer.start();
}
} class Tray {
private int content;
private boolean full; Tray() {
content = 0;
full = false;
} public synchronized void put(int content) {
while (full) {
try {
System.out.println("\t[Producer] nowhere for content " + content + "! wait for get...");
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
this.content = content;
this.full = true;
System.out.println("Producer put:" + content);
notifyAll();
} public synchronized void get() {
while (!full) {
try {
System.out.println("\t[Consumer] Nothing in tray! wait for put...");
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
this.full = false;
System.out.println("Consumer get:" + content);
notifyAll();
}
} class Producer extends Thread {
public void run() {
for (int i = 1; i <= 3; ++i) {
/*(1)行*/System.out.println("\033[32m" + "Start putting " + i + "\033[m");
Task4.TRAY.put(i);
try {
System.out.println("\033[32m" + "Put finished! sleep for a while..." + "\033[m");
sleep((int) (Math.random() * 100));
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("\033[32m" + "[switch to next]" + "\033[m");
}
} } class Consumer extends Thread {
public void run() {
for (int i = 1; i <= 3; ++i) {
/*(2)行*/System.out.println("\033[31m" + "Start Getting" + "\033[0m");
Task4.TRAY.get();
System.out.println("\033[31m" + "Get finished!\n[switch to next]" + "\033[0m");
}
}
}

结果、证实与猜想

  1. start()只是启动该线程(变为Runnable状态),运行与否以及运行多少均未知(看操作系统调度?),打印depart显然不能划分两个线程的先后
  2. 不要尝试去规划多线程运行的具体情况。根据代码不同,编译后乃至运行时的顺序会变;(不知是那一层的原因,什么编译优化、运行优化啥啥不懂)(相同代码生成的程序反复运行,结果一样,不过这也可能是在相邻的时间段内,运行环境一样)

对打印语句进行调整,看运行顺序:

情况1:(1)、(2)行均不注释

顺序: consumer打印start -> consumer使用get方法 -> consumer无效get进行等待 -> producer打印start -> producer进行put方法 -> ...

(sychronized中wait()后,循环的notify使得双线程的运行可预知(?))

producer进行put方法的最后,会唤醒等待的consumer,但consumer并没有直接运行,而是等producer运行至进入休眠,才接入cpu时间开始运行。

强调:notify()等方法只是将线程从"Waiting"状态变为"Runnable"状态,是否"Run"未可知。

情况2:(1)行注释,(2)行不注释

顺序: consumer打印start -> producer使用put方法 -> ...

我们并不知consumer线程何时让给了producer,但确乎是因为producer中少了一开始的打印start语句。可能经过了什么权衡先运行producer去了

情况3:(1)行不注释,(2)行注释

顺序: consumer使用get方法 -> ...

似乎又回到了情况1,consumer一开始就进入了线程同步的方法,运行到底后在交给producer。(很合理)

情况4:(1)、(2)均注释

顺序: producer使用put方法 -> ...

似乎回到了情况2,producer一开始就进入线程同步的方法,运行到底...

以上四种情况对比后,大致可以猜想,两个线程运行且没有特殊线程控制语句干扰时,会以某种单位划分(CPU时间?代码块大小?)进行交替运行,这里体现为一行行语句交替。

突然想到操作系统课上好像讲过多道程序balabala、多线程运行的CPU时间分配啥啥,感觉这样鼓捣了半天去验证有点傻

Java多线程运行探幽的更多相关文章

  1. Java多线程运行机制的基本原理

    Java多线程运行机制的基本原理 进程和线程的区别 进程 进程是一个程序执行的实例,比如说我们打开10个IE浏览器窗口,那么就有10个进程开启.一个进程可以同时被运行若干次,进程是CPU进行资源分配和 ...

  2. Java基础知识强化之多线程笔记05:Java程序运行原理 和 JVM的启动是多线程的吗

    1. Java程序运行原理:     Java 命令会启动Java 虚拟机,启动 JVM,等于启动了一个应用程序,也就是启动了一个进程.该进程会自动启动一个 “主线程” ,然后主线程去调用某个类的 m ...

  3. 40个Java多线程问题总结

    前言 Java多线程分类中写了21篇多线程的文章,21篇文章的内容很多,个人认为,学习,内容越多.越杂的知识,越需要进行深刻的总结,这样才能记忆深刻,将知识变成自己的.这篇文章主要是对多线程的问题进行 ...

  4. Java多线程基础知识篇

    这篇是Java多线程基本用法的一个总结. 本篇文章会从一下几个方面来说明Java多线程的基本用法: 如何使用多线程 如何得到多线程的一些信息 如何停止线程 如何暂停线程 线程的一些其他用法 所有的代码 ...

  5. Java多线程系列--“JUC锁”04之 公平锁(二)

    概要 前面一章,我们学习了“公平锁”获取锁的详细流程:这里,我们再来看看“公平锁”释放锁的过程.内容包括:参考代码释放公平锁(基于JDK1.7.0_40) “公平锁”的获取过程请参考“Java多线程系 ...

  6. Java多线程--让主线程等待子线程执行完毕

    使用Java多线程编程时经常遇到主线程需要等待子线程执行完成以后才能继续执行,那么接下来介绍一种简单的方式使主线程等待. java.util.concurrent.CountDownLatch 使用c ...

  7. Java多线程 2 线程的生命周期和状态控制

    一.线程的生命周期 线程状态转换图: 1.新建状态 用new关键字和Thread类或其子类建立一个线程对象后,该线程对象就处于新生状态.处于新生状态的线程有自己的内存空间,通过调用start方法进入就 ...

  8. java 多线程 1 线程 进程

    Java多线程(一).多线程的基本概念和使用 2012-09-10 16:06 5108人阅读 评论(0) 收藏 举报  分类: javaSE综合知识点(14)  版权声明:本文为博主原创文章,未经博 ...

  9. 第一章 Java多线程技能

    1.初步了解"进程"."线程"."多线程" 说到多线程,大多都会联系到"进程"和"线程".那么这两者 ...

  10. java从基础知识(十)java多线程(下)

    首先介绍可见性.原子性.有序性.重排序这几个概念 原子性:即一个操作或多个操作要么全部执行并且执行的过程不会被任何因素打断,要么都不执行. 可见性:一个线程对共享变量值的修改,能够及时地被其它线程看到 ...

随机推荐

  1. Mac下如何添加User到group中

    原因: 使用mac的时候需要像linux一样对用户和群组进行操作,但是linux使用的gpasswd和usermod在mac上都不可以使用,mac使用dscl来对group和user操作. 介绍: $ ...

  2. 基于开源IM即时通讯框架MobileIMSDK:RainbowChat-iOS端v7.0版已发布

    关于MobileIMSDK MobileIMSDK 是一套专门为移动端开发的开源IM即时通讯框架,超轻量级.高度提炼,一套API优雅支持 UDP .TCP .WebSocket 三种协议,支持 iOS ...

  3. 网页端IM通信技术快速入门:短轮询、长轮询、SSE、WebSocket

    本文来自"糊糊糊糊糊了"的分享,原题<实时消息推送整理>,有优化和改动. 1.写在前面 对Web端即时通讯技术熟悉的开发者来说,我们回顾网页端IM的底层通信技术,从短轮 ...

  4. Solution Set -「NOIP Simu.」20221010

      会不会组题啊? 希望 trash round 少来点. 「Unkown」构造字符串   给定 \(m\) 组形如 \((x,y,z)\), 表示 \(\operatorname{lcp}(S[x: ...

  5. Solution -「NOI 2017」「洛谷 P3825」游戏

    \(\mathscr{Description}\)   Link.   给大家看个乐子: link, 懒得概括题意啦. \(\mathscr{Solution}\)   对于没有 X 的情况, 显然可 ...

  6. 【转】Mysql索引失效的情况

    在工作中经常能遇到索引失效的情况,只要索引失效就导致了SQL查询慢,服务响应慢,用户体验差的情况:所以下面我们就讨论一下MySQL中索引失效的情况 口诀 全职匹配我最爱,最左前缀要遵守: 带头大哥不能 ...

  7. WPF刮刮乐

    WPF刮刮乐 <Window x:Class="WpfApp2.MainWindow" xmlns="http://schemas.microsoft.com/wi ...

  8. 如何快速的开发一个完整的iOS直播app(点赞功能)

    客户端代码 点击小红心,发送socket给服务器,并且要传递房间Key给服务器,通知给哪个主播点赞,就能传入到对应的分组socket中 怎么传递房间key,房间Key在主播界面,一般一个客户端,只会产 ...

  9. w3cschool-Hibernate 教程

    什么是 ORM? ORM 表示 Object-Relational Mapping (ORM),是一个方便在关系数据库和类似于 Java, C# 等面向对象的编程语言中转换数据的技术.一个 ORM 系 ...

  10. Redis的高可用?(主从、哨兵、集群)

    高可用有两个含义:一是数据尽量不丢失,二是保证服务尽可能可用. AOF 和 RDB 数据持久化保证了数据尽量不丢失,那么多节点来保证服务尽可能提供服务. 一般在实际生产中,服务不会部署成单节点,主要是 ...