线程池
我们为什么要使用线程池,因为通过Runnable接口的实现类创建的线程或者继承Thread的类都只能运行一次run方法,就会被销毁,并且我们不论是继承Thread类还是实现Runnable接口都需要重写run方法
我们可以用Runnable来提供执行目标,借助于Thread执行线程
线程池:(就是一个实现容纳线程个数的池子)
程序可以聪线程池中获取线程来完成目标代码
同时还可以将线程归还给线程池
省去了创建线程,销毁线程的繁琐操作
线程池的使用
1.我们第一步要创建一个线程池通过调用静态方法
public static ExecutorService newFixedThreadPool(int nThreads);
得到一个线程对象,初始化参数是要求当前线程池的线程数
2.通过submit方法执行线程
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) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + " 明天吃土豆牛腩");
}
}
public class Demo1 {
public static void main(String[] args) {
// 1. 创建线程池对象
ExecutorService service = Executors.newFixedThreadPool(5);
// 2. 创建一个MyThread1 Runnable接口实现类对象、
MyThread1 target = new MyThread1();
// 3. 使用线程池对象中的一个线程,指定目标代码
// 初始化线程数为5,这里使用的是线程池中已经存在的5个线程来执行代码
service.submit(target);
service.submit(target);
service.submit(target);
service.submit(target);
service.submit(target);
// 因为原本的5个线程都在被使用中,这里需要等待5个线程执行完毕,出现空闲线程
// 来执行对应的目标代码
service.submit(target);
service.submit(target);
// 4. 关闭线程池
// 一般不用关闭线程池,会随着程序的退出而关闭
// service.shutdown();
}
}
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("明天早上自己炸油条"));
}
/**
* 执行Cook实现类对象方法
*
* @param cook Cook接口的实现类对象
*/
public static void invokeCook(Cook cook) {
cook.cooking();
}
}
有参数有返回值
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("明天早上自己炸油条"));
}
/**
* 执行Cook实现类对象方法
*
* @param cook Cook接口的实现类对象
*/
public static void invokeCook(Cook cook) {
cook.cooking();
}
}
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));
/*
* 1. 数据类型真的可以省略,虽然是不同数据类型,Java语言真香
* 2. 实现的代码非一行,可以在大括号内完成你的目标
*/
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));
}
}
Lambda表达式使用前提
1.有且只有一个缺省属性为public abstract方法的接口,例如Comparator接口,Runnable接口
2.使用lambda表达式是有一个前后要求约束的,方法的参数为接口类型,或者说局部变量,或者说局部变量使用调用方法,可以使用lambda也可以
3.有且只有一个抽象方法的接口,称之为【函数式接口】