多线程,线程通信,线程池和Lambda表达式
1. 多线程
1.1 线程状态 WAITING(无限等待)
当某一个线程被执行wait()方法,需要等待另外的一个线程进行唤醒操作。
一下三个方法都是Object类内的方法:
public void wait();
在哪一个线程中执行,就会让当前线程进入一个无限等待状态。
1. 所在线程进入无限等待状态
2. 开启【锁对象】
public void notify();
唤醒和当前锁对象有关的无限等待线程中的一个,随机选择。
1. 唤醒一个无限等待状态线程
2. 开启【锁对象】
public void notifyAll();
唤醒所有和当前锁对象有关的无限等待线程
1. 唤醒所有线程
2. 开启【锁对象】
3. 线程进入锁对象抢占过程,就有可能进入一个锁阻塞状态。
1.2 线程执行的所有状态分析图
2. 线程通信
2.1 生活化例子
预约 --> 抢购
消费者 <--> 商品 <--> 生产者
消费者
1. 购买商品
2. 等待,不过在等待之前,需要告知生产者快点生产
生产者:
1. 生产商品
2. 休息,在休息之前,要告知消费者你快来买啊
商品:
就是两个独立线程之间的共享资源。
2.2 共享资源处理问题
现在存在两个完全无关的线程,生产者和消费者,但是商品会作为他们两者之间的共享资源。
生产者和消费者中都有一个成员变量
商品类型
【解决方案】
创建生产者或者消费者线程对象时,使用同一个商品类对象,作为构造方法参数进行初始化操作
2.3 设计程序
3. 线程池
3.1 线程池
不管是继承Thread还是遵从Runnable接口,都需要重写Run方法,而且每一个线程对象有且只能执行一次,之后就会被销毁。
利用Runnable接口来提供执行目标,而且借助于Thread执行线程。
一个餐厅
服务人员
餐厅会按照餐桌比例安排服务员人数。
每一个服务员我们都可以看做是一个线程对象
需要告知服务器做什么事情就可以了,相对于告知线程对象执行目标是什么
当你来餐厅之前,服务员在这里,你走之后,服务员依然在这类。
线程池 ==> 可以容纳多个线程的容器
程序可以从线程池获取线程来完成目标代码
同时也可以将线程归还给线程池。
省去了创建线程和销毁线程这样非常繁琐的操作。
3.2 线程池使用
public static ExecutorService newFixedThreadPool(int nThreads);
得到一个线程对象,初始化参数是要求的当前线程池中的线程数
public Future submit(Runnable target);
从线程池中获取一个线程对象,并且执行给定的Runnable接口实现类对象作为执行目标
package com. qfedu. b_executors;
import java. util. concurrent. ExecutorService;
import java. util. concurrent. Executors;
class MyThread1 implements Runnable {
@Override
public void run ( ) {
System. out. println ( "Runnable接口实现类,线程目标代码" ) ;
try {
Thread. sleep ( 1000 ) ;
} catch ( InterruptedException e) {
e. printStackTrace ( ) ;
}
System. out. println ( Thread. currentThread ( ) . getName ( ) + " 明天吃土豆牛腩" ) ;
}
}
public class Demo1 {
public static void main ( String[ ] args) {
ExecutorService service = Executors. newFixedThreadPool ( 5 ) ;
MyThread1 target = new MyThread1 ( ) ;
service. submit ( target) ;
service. submit ( target) ;
service. submit ( target) ;
service. submit ( target) ;
service. submit ( target) ;
service. submit ( target) ;
service. submit ( target) ;
}
}
3.4 从匿名内部类引入Lambda表达式
package com. qfedu. b_executors;
import java. util. concurrent. ExecutorService;
import java. util. concurrent. Executors;
public class Demo2 {
public static void main ( String[ ] args) {
ExecutorService service = Executors. newFixedThreadPool ( 5 ) ;
service. submit ( new Runnable ( ) {
@Override
public void run ( ) {
System. out. println ( Thread. currentThread ( ) . getName ( ) ) ;
}
} ) ;
service. submit ( ( ) - > System. out. println ( Thread. currentThread ( ) . getName ( ) ) ) ;
service. submit ( ( ) - > System. out. println ( Thread. currentThread ( ) . getName ( ) ) ) ;
service. submit ( ( ) - > System. out. println ( Thread. currentThread ( ) . getName ( ) ) ) ;
service. submit ( ( ) - > System. out. println ( Thread. currentThread ( ) . getName ( ) ) ) ;
}
}
4. Lambda表达式
4.1 思想 “说重点”
service. submit ( new Runnable ( ) {
@Override
public void run ( ) {
System. out. println ( Thread. currentThread ( ) . getName ( ) ) ;
}
} ) ;
4.2 Lambda表达式格式
service. submit ( ( ) - > System. out. println ( Thread. currentThread ( ) . getName ( ) ) ) ;
4.3 Lambda表达式使用,无参数无返回值
package com. qfedu. c_lambda;
interface Cook {
void cooking ( ) ;
}
public class Demo1 {
public static void main ( String[ ] args) {
invokeCook ( new Cook ( ) {
@Override
public void cooking ( ) {
System. out. println ( "麻辣香锅,孜然肉片,土豆牛肉,蒜薹肉丝" ) ;
}
} ) ;
invokeCook ( ( ) - > {
System. out. println ( "蒜蓉油麦菜,番茄鸡蛋" ) ;
} ) ;
invokeCook ( ( ) - > System. out. println ( "明天早上自己炸油条" ) ) ;
}
public static void invokeCook ( Cook cook) {
cook. cooking ( ) ;
}
}
4.4 Lambda表达式使用,有参数有返回值
package com. qfedu. c_lambda;
import java. util. Arrays;
import java. util. Comparator;
public class Demo2 {
public static void main ( String[ ] args) {
Person[ ] persons = {
new Person ( "骚磊" , 16 ) ,
new Person ( "老黑" , 50 ) ,
new Person ( "老付" , 40 ) ,
new Person ( "污云" , 35 ) ,
new Person ( "朋朋" , 14 ) ,
new Person ( "大哥" , 18 ) ,
} ;
Arrays. sort ( persons, new Comparator < Person> ( ) {
@Override
public int compare ( Person o1, Person o2) {
return o1. getAge ( ) - o2. getAge ( ) ;
}
} ) ;
Arrays. sort ( persons, ( Person o1, Person o2) - > {
return o2. getAge ( ) - o1. getAge ( ) ;
} ) ;
Arrays. sort ( persons, ( o1, o2) - > o2. getAge ( ) - o1. getAge ( ) ) ;
for ( Person person : persons) {
System. out. println ( person) ;
}
}
}
package com. qfedu. c_lambda;
interface A {
float add ( int num1, float num2) ;
}
public class Demo3 {
public static void main ( String[ ] args) {
System. out. println ( new A ( ) {
@Override
public float add ( int num1, float num2) {
return num1 + num2;
}
} . add ( 5 , 13.5F ) ) ;
test ( 5 , 13.5F , ( num1, num2) - > {
float sum = num1 + num2;
return sum;
} ) ;
}
public static void test ( int num1, float num2, A a) {
System. out. println ( a. add ( num1, num2) ) ;
}
}
4.5 Lambda表达式使用前提
1. 有且只有一个缺省属性为public abstract方法的接口,例如 Comparator<T>接口,Runnable接口
2. 使用lambda表达式是有一个前后要求约束的
方法的参数为接口类型,或者说局部变量使用调用方法,可以使用lambda也OK
3. 有且只有一个抽象方法的接口,称之为【函数式接口】