day11(多线程,唤醒机制,生产消费者模式,多线程的生命周期)
A:进程:
进程指正在运行的程序。确切的来说,当一个程序进入内存运行,即变成一个进程,进程是处于运行过程中的程序,并且具有一定独立功能。
B:线程:
线程是进程中的一个执行单元,负责当前进程中程序的执行,一个进程中至少有一个线程。一个进程中是可以有多个线程的,这个应用程序也可以称之为多线程程序。
C:简而言之:
一个程序运行后至少有一个进程,一个进程中可以包含多个线程
线程实现
实现的两种方式
继承Thread
public class MyThread extends Thread{
@Override
public void run() {
for (int i = 0; i < 100; i++) {
System.out.println(getName()+":"+i);
}
}
}
测试类
public class MyThread02 extends Thread {
@Override
public void run() {
MyThread t=new MyThread();//直接创建对象就能创建一个线程
t.start();
}
}
实现Runnable
public class MyThreadImp implements Runnable{
int num;
public MyThreadImp(int num){
this.num=num;
}
@Override
public void run() {
for (int i = 0; i < 5; i++) {
System.out.println(Thread.currentThread().getName()+num++);
}
}
}
测试类
public class Test01 {
public static void main(String[] args) {
MyThreadImp mt=new MyThreadImp(10);
Thread t=new Thread(mt);
t.setName("张三");
t.start();
Thread t1=new Thread(mt);
t1.setName("李四");
t1.start();
}
}
多线程安全问题
线程的安全性==数据的安全性
①是否存在多线程
②是否有共享数据
③是否有多条语句执行共享数据
解决办法 synchronized同步锁
eg:
public class MyThread implements Runnable {
int tickets;
public MyThread(int tickets) {
this.tickets = tickets;
}
Object o = new Object();
@Override
public void run() {
while (true) {
synchronized (o) {
if (tickets > 1) {
System.out.println(Thread.currentThread().getName() + "-"
+ tickets--);
}
}
} }
}
测试类
MyThread mt=new MyThread(100);
Thread t=new Thread(mt);
t.setName("窗口1");
Thread t2=new Thread(mt);
t2.setName("窗口2");
Thread t3=new Thread(mt);
t3.setName("窗口3");
t.start();
t2.start();
t3.start();
如果不给线程加锁 则会产生数据乱掉,可能会产生-1;结果我们不可预测。
在方法中给加锁,另一种方法则是给方法加锁eg:
public class MyThread01 implements Runnable{
static int tickets=100;
public MyThread01(int tickets) {
this.tickets = tickets;
}
@Override
public void run() {
while (true) {
method01();
} }
private synchronized void method01() {
if (tickets >= 1) {
System.out.println(Thread.currentThread().getName() + "-"
+ tickets--);
}
}
}
抢红包案列:
public class MyThread implements Runnable { int money;
int count; public MyThread(int money, int count) {
this.money = money;
this.count = count;
} @Override
public void run() {
// 产生一个随机数
method();
} private synchronized void method() {
if (count > 1 && money != 0) {
Random r = new Random();
double d = (r.nextInt(money) + 1);
money -= d;
System.out.println(Thread.currentThread().getName() + ":" + d);
count--;
} else {
System.out.println(Thread.currentThread().getName() + ":"
+ money);
}
} }
测试类
public class Hongbao {
public static void main(String[] args) {
MyThread mt=new MyThread(10,3);
Thread t=new Thread(mt);
t.setName("张三");
t.start();
t.setPriority(10);
Thread t1=new Thread(mt);
t1.setName("李四");
t1.start();
Thread t2=new Thread(mt);
t2.setName("李");
t2.start();
}
}
生产消费者模式:
解释了多线程唤醒机制(notify()是Object中的方法)
使用多线程中的案列
产品 由于只是做测试使用 没有写全
public class Student {
String name;
int age;
boolean flag;
}
生产者 如果消费者没有消费产品 则进行等待 ,只有消费了生产者才会进行生产
public class StThread implements Runnable {
private Student s;
public StThread(Student s) {
this.s = s;
}
int x;
@Override
public void run() {
while (true) {
synchronized (s) {
if (s.flag) {//如果flag=true;则说明已经有对象了 还没有消费,则进行等待
try {
s.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}// 等待
}
if (x % 2 != 0) {
s.name = "赵云";
s.age = 44;
} else {
s.name = "张飞";
s.age = 22;
}
x++;
s.flag=true;//修改标记
s.notify();//唤醒
}
}
}
}
消费者 如果消费者没有东西消费 则会进行等待 有了东西,才会进行消费
public class GtThread implements Runnable {
private Student s;
public GtThread(Student s) {
this.s = s;
}
@Override
public void run() {
while (true) {//保证一直在等待中
synchronized (s) {
if (!s.flag) {//flag=false则进行等待 等待学生对象的产生
try {
s.wait();//线程等待
} catch (InterruptedException e) {
e.printStackTrace();
}
}
System.out.println(s.name + ":" + s.age);
s.flag=false;
s.notify();
}
}
}
}
测试类
public class Test {
public static void main(String[] args) {
Student s=new Student();
StThread st=new StThread(s);
GtThread gt=new GtThread(s);
Thread t1=new Thread(st);
Thread t2=new Thread(gt);
t1.start();
t2.start();
}
}
输出结果
输出结果:
赵云:44
张飞:22
赵云:44
张飞:22
赵云:44
张飞:22
赵云:44
.......一直交替循环下去
线程的生命周期
day11(多线程,唤醒机制,生产消费者模式,多线程的生命周期)的更多相关文章
- Java多线程学习笔记--生产消费者模式
实际开发中,我们经常会接触到生产消费者模型,如:Android的Looper相应handler处理UI操作,Socket通信的响应过程.数据缓冲区在文件读写应用等.强大的模型框架,鉴于本人水平有限目前 ...
- 多线程学习之三生产者消费者模式Guarded Suspension
Guarded Suspension[生产消费者模式] 一:guarded suspension的参与者--->guardedObject(被防卫)参与者 1.1该 ...
- Linux——多线程下解决生产消费者模型
我们学习了操作系统,想必对生产消费者问题都不陌生.作为同步互斥问题的一个经典案例,生产消费者模型其实是解决实际问题的基础模型,解决很多的实际问题都会依赖于它.而此模型要解决最大的问题便是同步与互斥.而 ...
- Kafka下的生产消费者模式与订阅发布模式
原文:https://blog.csdn.net/zwgdft/article/details/54633105 在RabbitMQ下的生产消费者模式与订阅发布模式一文中,笔者以“数据接入”和“事 ...
- Python | 面试的常客,经典的生产消费者模式
本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是Python专题的第23篇文章,我们来聊聊关于多线程的一个经典设计模式. 在之前的文章当中我们曾经说道,在多线程并发的场景当中,如果我 ...
- 生产消费者模式与python+redis实例运用(中级篇)
上一篇文章介绍了生产消费者模式与python+redis实例运用(基础篇),但是依旧遗留了一个问题,就是如果消费者消费的速度跟不上生产者,依旧会浪费我们大量的时间去等待,这时候我们就可以考虑使用多进程 ...
- RabbitMQ下的生产消费者模式与订阅发布模式
所谓模式,就是在某种场景下,一类问题及其解决方案的总结归纳.生产消费者模式与订阅发布模式是使用消息中间件时常用的两种模式,用于功能解耦和分布式系统间的消息通信,以下面两种场景为例: 数据接入 假设 ...
- 生产消费者模式与python+redis实例运用(基础篇)
根据这个图,我们举个简单的例子:假如你去某个餐厅吃饭,点了很多菜,厨师要一个一个菜的做,一个厨师不可能同时做出所有你点的菜,于是你有两个选择:第一个,厨师把所有菜都上齐了,你才开始吃:还有一个选择,做 ...
- C++11 多线程编程 使用lambda创建std::thread (生产/消费者模式)
要写个tcp server / client的博客,想着先写个c++11多线程程序.方便后面写博客使用. 目前c++11中写多线程已经很方便了,不用再像之前的pthread_create,c++11中 ...
随机推荐
- Inteiilj IDEA 团队代码格式规范
目录 Intellij IDEA code format Tabs and Indents Spaces Wrapping and Braces Imports 更新 Intellij IDEA co ...
- static静态局部变量初始化问题
第一次写博客之类的东西,主要是为了记录自己的学习过程,以便于记忆的加深和与各位大神进行探讨,学习更多的东西. 本次上传主要是关于静态局部变量的初始化问题. 首先,静态局部变量和全局变量一样,数据都存放 ...
- Ubuntu 安装 Elasticsearch
1.安装java 注意:最新版本的elasticsearch(5.6.2)要求安装java8 1.sudo apt-add-repository ppa:webupd8team/java 2.sudo ...
- "\\s+"的使用
详解 "\\s+" 正则表达式中\s匹配任何空白字符,包括空格.制表符.换页符等等, 等价于[ \f\n\r\t\v] \f -> 匹配一个换页 \n -> 匹配一个换 ...
- memory management
1. 高端内存: 内存的物理寻址范围比虚拟寻址范围大的多,有一些内存页不能永久的映射到内核地址空间. 2. 高端内存和低端内存是内核对内存物理页的划分. 参考:http://ilinuxkernel. ...
- eclipse新建maven项目出现红叉解决办法
新建的maven项目,项目内代码及pom.xml没有任何问题,但项目上就是有红叉,这时点开Markers(Window–>show veiw–>Markers),查看错误的详细信息,信息上 ...
- 在winsshd 中添加id_rsa.pub 实现Windows 服务器主机自动信任Linux 客户端
文章一. 生成密钥: 在Linux主机(ssh客户端),通过ssh-keygen在建立SSH keys# ssh-keygen -t rsa (连续三次回车,即在本地生成了公钥和私钥,不设置密码)将在 ...
- 利用Google Chrome开发插件,在网页中植入js代码
Google Chrome是一个很强大的浏览器,提供了各种各样的插件,大大提升了使用了的效率,比如vimium.honx等. Google在提供这些插件的同时还允许用户开发自己的插件. 最近在写js的 ...
- 349. Intersection of Two Arrays 是否重复
不重复的: 方法一 [一句话思路]:排序之后用归并. [一刷]: 根据返回方法的类型来判断corner case.判断空.0数组的corner case还不一样,得分开写 由于先排序了,nums1[i ...
- 9-最短路径(dijkstra)
参考博客:http://www.wutianqi.com/?p=1890 #include <iostream>using namespace std;#define max 1< ...