lambda表达式实战分析 下篇
引子
还记得上次分享的练习题
有一箱东港九九草莓,想吃其中最大的一颗草莓?(可模拟所需的所有数据)
自己实现筛选工具类
public static <T> T getMax(List<T> list, Comparator<T> comparator) {
Objects.requireNonNull(list);
Iterator<T> iterator = list.iterator();
T result = iterator.next();
while (iterator.hasNext()) {
T next = iterator.next();
int value = comparator.compare(result, next);
if (value > 0) {
result = next;
}
}
return result;
}
其中Comparator不是系统的类,是我自己实现的接口
Comparator类代码:
@FunctionalInterface
public interface Comparator<T> {
int compare(T t1, T t2);
static <E> Comparator<E> comparing(ToIntFunction<E> function) {
return (E b1, E b2) -> function.applyAsInt(b2) - function.applyAsInt(b1);
}
static <E> Comparator<E> comparing(ToIntFunction<E> function, ToIntFunction<E> function2) {
return (E b1, E b2) -> function.applyAsInt(b2) - function.applyAsInt(b1) == 0 ? function2.applyAsInt(b2) - function2.applyAsInt(b1) : function.applyAsInt(b2) - function.applyAsInt(b1);
}
default Comparator<T> thenCompare(ToIntFunction<T> function){
return (T t1, T t2) -> compare(t1, t2) == 0 ? Comparator.comparing(function).compare(t1, t2) : compare(t1, t2);
}
}
使用lambda如何实现
private static void filterVersion1() {
Strawberry strawberry = CollectionUtils.getMax(RAW_STRAWBERRY_LIST,
(Strawberry b1, Strawberry b2) -> b2.getWeight() - b1.getWeight());
Out.println("我想吃最大的草莓是:");
Out.println(strawberry.toString());
}
通过方法引用如何实现
private static int getComparator(Strawberry b1, Strawberry b2) {
return b2.getWeight() - b1.getWeight();
}
private static void filterVersion2() {
Strawberry strawberry = CollectionUtils.getMax(RAW_STRAWBERRY_LIST,
Practice2Client::getComparator);
Out.println("我想吃最大的草莓是:");
Out.println(strawberry.toString());
}
在使用方法引用时,取方法名要格外恰当,好的方法名能让代码像是在描述问题而不是在解决问题
现在我不想吃最大的草莓了,我想吃最甜的草莓,要怎么做呢?
高阶函数:
如果一个函数的输入或者输出也是函数,那么这个函数就是高阶函数。举个例子
private static com.sugarya.interfaces.Comparator<Strawberry> comparing(Function<Strawberry, Integer> function) {
return (Strawberry b1, Strawberry b2) -> function.apply(b2) - function.apply(b1);
}
private static void filterVersion4() {
Strawberry strawberry = CollectionUtils.getMax(RAW_STRAWBERRY_LIST, comparing(Strawberry::getSweetness));
Out.println("我想吃最甜的草莓是:");
Out.println(strawberry.toString());
}
如果最大的草莓重量相同时,我想吃其中最甜的,怎么做?
private static com.sugarya.interfaces.Comparator<Strawberry> comparing(Function<Strawberry, Integer> function, Function<Strawberry, Integer> function2) {
return (Strawberry b1, Strawberry b2) -> function.apply(b2) - function.apply(b1) == 0 ? function2.apply(b2) - function2.apply(b1) : function.apply(b2) - function.apply(b1);
}
/**
* 找到最大的草莓,如果一样大,就要其中最甜的
*/
private static void filterVersion5() {
Strawberry strawberry = CollectionUtils.getMax(RAW_STRAWBERRY_LIST,
comparing(Strawberry::getWeight, Strawberry::getSweetness));
Out.println("我想吃最甜的草莓是:");
Out.println(strawberry.toString());
}
java8的接口允许定义静态方法,要定义实例方法,需要关键词default,即为默认方法
如果要在其他类里再次筛选,代码就要重复,因此,考虑把逻辑封装到接口里,代码如下:
有没有办法像链式调用那样,在外层添加新的判断逻辑呢?
可以的,Comparator接口里,实现thenCompare方法。
default Comparator<T> thenCompare(ToIntFunction<T> function){
return (T t1, T t2) -> compare(t1, t2) == 0 ? Comparator.comparing(function).compare(t1, t2) : compare(t1, t2);
}
/**
* 最大的草莓,如果一样大,就要其中最甜的
*/
private static void filterVersion7() {
Strawberry strawberry = CollectionUtils.getMax(RAW_STRAWBERRY_LIST,
Comparator.comparing(Strawberry::getWeight).thenCompare(Strawberry::getSweetness));
Out.println("我想吃最甜的草莓是:");
Out.println(strawberry.toString());
}
复合Lambda表达式
这里我们自己实现了一个Lambda表达式的复合使用。把多个简单的Lambda复合成复杂的表达式
lambda重构
Lambda与匿名内部类的差异
匿名类中this代表的是函数式接口对象,但是在Lambda表达式中代表的是外部所在的类。
lambda表达式里的局部变量不能和外部的变量重名,匿名内部类可以
匿名内部类可以实现非函数式接口,lambda表达式只能作为是函数式接口的传入
lambda表达式的类型问题
lambda表达式反映的是结构
private static void testLambda() {
testLambdaSign((Task) () -> {
println("");
});
}
private static void testLambdaSign(Runnable runnable) {
println("testLambdaSign Runnable");
}
private static void testLambdaSign(Task task) {
println("testLambdaSign task");
}
lambda实战应用
表达式解决设计模式与生俱来的设计僵化问题
lamdba表达式与设计模式
对策略方法,观察者模式时,把设计模式里涉及到函数式接口替换成lambda表达式表示即可。
模板设计模式
模板设计模式,通过继承抽象类,实现抽象方法来实现。这时候可以直接引入一个新的参数,来替换抽象类,函数描述符。
public abstract class AbstractComputer {
public void work(){
powerOn();
int hardware = checkHardware();
loadOS(hardware);
}
public void work(Consumer<Integer> consumer){
powerOn();
int hardwareParam = checkHardware();
consumer.accept(hardwareParam);
}
protected void powerOn(){
Out.println("开启电源");
}
protected int checkHardware(){
Out.println("检查硬件");
return 1;
}
protected abstract void loadOS(int hardware);
}
责任链模式
而责任链模式使用了复合lambda表达式
把数据结构里的链式思维应用在面向对象的编程里,将每一个链条的节点看作是一个对象,每个对象拥有不同的处理逻辑。一个业务逻辑被看作从链条的首端发出,途径一个个链节点,至末端结束。
面向对象的编程:
- 使用父类&子类建立思维体系
- 通过类与类的相互联系和通信来组织逻辑
- 用重写来表达变化
- 用抽象方法表达抽象
- 用接口来表达抽象
链节点:作为对象,我们建立一个基类来表示,具体的每个节点是它的子类。
每个对象/节点处理不同的逻辑:不同是变化,需要抽象出来,可以用抽象方法表达抽象,也可以用接口表达抽象。这里我们用抽象方法来做
链式思维:每个当前节点,持有下一个节点,如果是双向的链式,则再持有上一个节点。常用的方式,当前节点的输出作为下一个节点的输入
代码实现
public abstract class AbstractNode<T> {
private AbstractNode<T> nextNode;
public T startChain(T t){
T element = handle(t);
if(nextNode != null){
return nextNode.handle(element);
}
return t;
}
public AbstractNode<T> getNextNode() {
return nextNode;
}
public void setNextNode(AbstractNode<T> nextNode) {
this.nextNode = nextNode;
}
public abstract T handle(T t);
}
FirstNode
public class FirstNode extends AbstractNode<String> {
@Override
public String handle(String s) {
return "Hello, " + s;
}
}
SecondNode
public class SecondNode extends AbstractNode<String> {
@Override
public String handle(String s) {
return s.replace("lamda","lambda");
}
}
使用
private static void testChainByLambda() {
Function<String, String> firstNode = s -> "Hello, " + s;
Function<String, String> secondNode = s -> s.replace("lamda", "lambda");
Function<String, String> chain = firstNode.andThen(secondNode);
String result = chain.apply("This is lamda");
Out.println("chain result = " + result);
}
简单工厂
工厂设计模式,使用到构造函数的方法引用
public class FruitFactory2 {
private static Map<FruitType, Supplier<? extends Fruit>> FRUIT_MAP = new HashMap<>();
static {
FRUIT_MAP.put(FruitType.Apple, Apple::new);
FRUIT_MAP.put(FruitType.Strawberry, Strawberry::new);
}
public static <T extends Fruit> T createFruit(FruitType fruitType) {
Objects.requireNonNull(fruitType);
return (T)FRUIT_MAP.get(fruitType).get();
}
}
Java8的lambda表达式与Kotlin,Python的差异
这些设计模式在使用lambda表达式重写会更简洁,但是也有人疑问,lambda只是让代码简洁而已,我用匿名内部类或接口仍然可以健壮的实现功能啊。对于这个疑问,我的理解是,如果赞同函数式编程是未来的方向,那么就很有必要使用lambda表达式,lambda表达式这套新的符号背后承载的是不同以往的思维方式--即所谓的“函数式”的思维。不够好的式Java8的lambda是不完全的函数式,但是从另一个角度来说,这也是好的,java8不完全的函数式能让java 7的程序员更容易和平稳得从面向对象过渡到函数式编程。
函数式的程度由低到高是:
Java8 《 Kotlin 《 Python
我分别举个Koltin和Python的例子
Kotlin实现链式
private fun startChain(str: String): String{
val firstNode: (String) -> String = {s -> "Hello, " + s}
val secondNode = { s: String -> s.replace("lamba", "lambda")}
return secondNode(firstNode(str))
}
Kotlin比Java8更加函数式
Python实现链式
def startChain(x):
firstNode = lambda x: "Hello, " + x
secondNode = lambda x: x.replace("lamda", "lambda")
return secondNode(firstNode(x))
print(startChain("This is lamda"))
Python比Kotlin更加函数式
欢迎关注CodeThings