1.1 线程池
不管是继承Thread还是遵从Runnable接口,都需要重写Run方法,而且每一个线程对象有且只能执行一次,之后就会被销毁。
利用Runnable接口来提供执行目标,而且借助于Thread执行线程。
线程池 ==> 可以容纳多个线程的容器
程序可以从线程池获取线程来完成目标代码
同时也可以将线程归还给线程池。
省去了创建线程和销毁线程这样非常繁琐的操作。
线程池任务执行参考图:
1.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) {
// 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();
}
}
1.3 从匿名内部类引入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());
}
});
// lambda表达式
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()));
}
}
2 Lambda表达式
2.1 思想
service.submit(new Runnable() {
@Override
public void run() {
System.out.println(Thread.currentThread().getName());
}
});
匿名内部类方式来作为线程池执行目标代码
1. 这个方法需要的参数是Runnable接口的实现类对象
2. Runnable接口目标是为了提供一个run 方法, What will be run
3. What will be run??? where??? run方法内容
这里需要Runnable提供Run方法,提供Run方法方法体
“说重点”
需要Run方法方法体
2.2 Lambda表达式格式
service.submit(() -> System.out.println(Thread.currentThread().getName()));
() -> System.out.println(Thread.currentThread().getName())
Lambda表达式
() 参数列表
-> 做什么事情,就是对应方法体
箭头之后的代码就是正常语句
(参数列表) -> {代码语句}
2.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("明天早上自己炸油条"));
}
/**
* 执行Cook实现类对象方法
*
* @param cook Cook接口的实现类对象
*/
public static void invokeCook(Cook cook) {
cook.cooking();
}
}
2.4 Lambda表达式使用,有参数有返回值
package bkDay18.lambda;
import java.util.Arrays;
import java.util.Comparator;
public class Demo2 {
public static void main(String[] args) {
Person[] persons = {new Person("张三",20),
new Person("李四",18),
new Person("王五",22),
new Person("赵六",16)};
Arrays.sort(persons, new Comparator<Person>() {
@Override
public int compare(Person o1, Person o2) {
return o1.getAge() - o2.getAge();
}
});
/*
* 1.可以省略数据类型
* 2.可以直接利用返回值,{}和return都省略掉
*/
Arrays.sort(persons, (o1,o2) -> o2.getAge() - o1.getAge());
Arrays.sort(persons, (o1,o2) -> {
return o2.getAge() - o1.getAge();
});
for(Person person : persons) {
System.out.println(person);
}
System.out.println("-------------");
/*
* 有参数 有返回值
*
* 标准lambda
*/
Arrays.sort(persons, (Person o1,Person o2) -> {
return o1.getAge() - o2.getAge();
});
for(Person p : persons) {
System.out.println(p);
}
}
}
package bkDay18.lambda;
interface A{
int add(int num1,int num2);
}
public class Demo3 {
public static void main(String[] args) {
System.out.println(new A() {
@Override
public int add(int num1, int num2) {
return num1+ num2;
}
}.add(1, 4));
test(2, 3, new A() {
@Override
public int add(int num1, int num2) {
return num1 + num2;
}
});
/*
* 1.数据类型可以省略,尽管两个数的数据类型不一样
* 2.实现的代码不一定是一行 可以在大括号内完成自己的目标
*/
test(0, 5, (num1,num2) -> num1 + num2);
test(5, 5, (int num1,int num2) -> {
return num1 + num2;
});
test(6, 6, (int num1,int num2) -> num1 + num2);
}
public static void test(int num1,int num2,A a) {
System.out.println(a.add(num1, num2));
}
}
2.5 Lambda表达式使用前提
- 有且只有一个缺省属性为public abstract方法的接口,例如 Comparator接口,Runnable接口
- 使用lambda表达式是有一个前后要求约束的
方法的参数为接口类型,或者说局部变量使用调用方法,可以使用lambda也OK- 有且只有一个抽象方法的接口,称之为【函数式接口】
package bkDay18.lambda;
/*
* 有且只有一个抽象方法的接口,称之为【函数式接口】
*/
abstract class AbstractType{
public abstract void test();
public void test1() {
System.out.println("1111111111");
}
}
public class Demo4 {
public static void main(String[] args) {
new AbstractType() {
@Override
public void test() {
System.out.println("1234567");
}
}.test();
new AbstractType() {
@Override
public void test() {
}
}.test1();
}
}