Java中的线程池 & Lambda表达式

1.线程池理解

不管是继承Thread类还是遵从Runnable接口,都需要重写Runnable方法,而且每一个线程对象有且只能执行一次,执行过后即被销毁。
利用Runnable接口来提供执行目标,而借助于Thread执行线程。

一个餐厅
有一定服务人员
餐厅会按照餐桌比例安排服务人员数量
每一个服务员可看作是一个线程对象
需要告知服务器做什么事即可,相对于告知线程对象执行目标是什么。
不论你在不在餐厅,服务员一直在

线程池:

可以容纳多个线程的容器
程序可以从线程池获取线程来完成目标代码。
同时也可以将线程归还给线程池。
省去了创建线程和销毁线程这样非常繁琐的操作

2.线程池使用

public static ExcutorService newFixedTheradPool(int nThreads);
得到一个线程对象,初始化参数 是要求的当前线程池中的线程数

public Future submit(Runnable target);
从线程池中获取一个线程对象,并且执行给定的Runnable接口实现类对象作为执行目标

线程池使用 代码展示

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();
	}
}

3.从匿名内部类引入Lambda表达式

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())); 	
	}
}

4.Lambda表达式

4.1Lambda表达式格式

service.submit(() -> System.out.println(Thread.currentThread().getName()));
/*
() -> System.out.println(Thread.curretThread().getName())
Lambda表达式
() 参数列表
-> 做什么事情,就是对应方法体
箭头之后的代码就是正常语句
	
(参数列表) -> {代码语句}

4.2Lambda表达式使用——无参数返回值

/*
 * 无参数无返回值
 */

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();
	}
}

4.3Lambda使用——有参数有返回值

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),
		};
		
		// public static <T> void sort(T[] a, Comparator<? super T> c)
		Arrays.sort(persons, new Comparator<Person>() {

			@Override
			public int compare(Person o1, Person o2) {
				return o1.getAge() - o2.getAge();
			}
		});
		
		/*
		 * 1. 有参数
		 * 2. 有返回值
		 * 
		 * (Person o1, Person o2) -> {
		 * 		return o1.getAge() - o2.getAge();
		 * }
		 * 标准Lambda
		 */
		Arrays.sort(persons, (Person o1, Person o2) -> {
			return o2.getAge() - o1.getAge();
		});
		
		/*
		 * 1. 这里可以省略数据类型
		 * 2. 可以直接利用返回值,{}和return都省了掉
		 */
		Arrays.sort(persons, (o1, o2) -> o2.getAge() - o1.getAge());
		
		for (Person person : persons) {
			System.out.println(person);
		}
	}
}
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));
	}
}

4.4Lambda表达式使用前提

1.有且只有一个缺省属性为public abstract方法的接口,例如Comparator<T>接口,Runnable接口
2.使用Lambda表达式是有一个前后要求约束的。方法的参数为接口类型,或者说局部变量使用调用
方法,可以使用Lambda
3.有且只有一个抽象方法的接口,称之为【函数式接口】
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值