通过行为参数传递代码

作者: 王尼小老板 | 来源:发表于2017-02-06 23:39 被阅读182次

应对不断变化的需求

目标

在软件工程中一个众所周知的问题就是,不管你做什么,用户的需求肯定会变。比如一位农民第一天可能有一个想要查找库存中所有绿色苹果的功能,但第二天可能又想要找出重量大于 150 克的水果,可能过几天又会变成既要绿色苹果重量又要大于 150 克。我们该如何应对不断变化的需求?我们最理想的状态为:工作量降到最少,新添加的功能实现起来非常简单而且易于长期维护

行为参数化是什么

行为参数化就是可以帮助你处理频繁变更的需求的一种软件开发模式。一言以蔽之,它意味着拿出一个代码块,把他准备好却不去执行它。这个代码块以后可以被你程序的其他部分调用,这意味着你可以推迟这块代码的执行。例如,你可以将代码块作为参数传递给另一个方法,稍后再去执行它。这样,这个方法的行为就基于那块代码被参数化了。

下面我们结合上面提到的农民的例子来应对不断变化的需求。

筛选绿苹果

public static List<Apple> filterGreenApples(List<Apple> inventory){
    List<Apple> result=new ArrayList<>();
    for (Apple apple : inventory) {
        if("green".equals(apple.getColor())){
            result.add(apple);
        }
    }
    return result;
}

筛选出绿色苹果很简单吧,但是农民如果要筛选红色苹果呢?最简单的解决方法就是复制这个方法,把名字改成 filterRedApples,然后改一下 if 条件来匹配红苹果。然而如果颜色更多呢?我们不可能写一大溜的方法来匹配不同的颜色吧!一个良好的原则是在编写类似的代码后尝试将其抽象化

把颜色作为参数

一种做法是给方法加一个参数,把颜色变成参数,这样就能灵活地适应变化了:

public static List<Apple> filterApplesByColor(List<Apple> inventory,String color){
    List<Apple> result=new ArrayList<>();
    for (Apple apple : inventory) {
        if(apple.getColor().equals(color)){
            result.add(apple);
        }
    }
    return result;
}

只要像下面这样调用就可以了:

    List<Apple> greenApples=filterApplesByColor(inventory,"green");
    List<Apple> redApples=filterApplesByColor(inventory,"red");

太简单了吧?那如果农民想要区分轻的苹果和重的苹果呢?

对你能想到的每个属性做筛选

我们写一个颜色和重量结合的方法,再加上一个 flag 来区分想要筛选哪个属性。

public static List<Apple> filterApples(List<Apple> inventory, String color, int weight, boolean flag) {
    List<Apple> result = new ArrayList<>();
    for (Apple apple : inventory) {
        if ((flag && apple.getColor().equals(color)) || (flag && apple.getWeight() > weight)) {
            result.add(apple);
        }
    }
    return result;
}

调用:

    List<Apple> greenApples = filterApples(inventory, "green",150,true);
    List<Apple> redApples = filterApples(inventory, "red",120,false);

上面的方法好像依然很轻松的解决了农民朋友的问题,但是客户端代码看起来糟透了,true 和 false 又是什么意思?此外解决方案还是不能很好的应对变化的需求。如果这位农民要求你对苹果的不同属性做筛选,比如大小、形状、产地等,又怎么办?而且农民要求你组合属性,做更复杂的查询,又该怎么办?你会有好多重复的 filter 方法,或一个巨大的非常复杂的方法。我们下一节就开始利用行为参数化来实现更加灵活的方法。

行为参数化

首先我们后退一步来进行更高层次的抽象:我们考虑的是苹果,需要根据 Apple 的一些属性(比如它是绿色的吗?重量超过 150 克吗?)来返回一个 boolean 值。我们把它称为谓词(即一个返回 boolean 值的函数)。让我们定义一个接口来对选择标准建模:

public interface ApplePredicate {
    boolean test (Apple apple);
}

现在我们可以使用 ApplePredicate 的多个实现代表不同的选择标准了:

筛选重量大于 150 的苹果:

public class AppleHeavyWeightPredicate implements ApplePredicate{
    @Override
    public boolean test(Apple apple) {
        return apple.getWeight()>150;
    }
}

筛选绿色苹果:

public class AppleGreenColorPredicate implements ApplePredicate{
    @Override
    public boolean test(Apple apple) {
        return "green".equals(apple.getColor());
    }
}

我们这样做已经非常类似“策略设计模式”了,在这里算法族就是 ApplePredicate,不同的策略就是 AppleHeavyWeightPredicate 或者 AppleGreenColorPredicate。

在我们的例子中根据抽象条件筛选:

public static List<Apple> filterApples(List<Apple> inventory,ApplePredicate p){
    List<Apple> result = new ArrayList<>();
    for (Apple apple : inventory) {
        if(p.test(apple)){
            result.add(apple);
        }
    }
    return result;
}

现在我们的代码已经灵活多了,读起来用起来都更容易!如果你要筛选红色并且重量大于 200 克的苹果只需要再写一个策略:

public class AppleRedAndHeavyPredicate implements ApplePredicate {
    @Override
    public boolean test(Apple apple) {
        return "red".equals(apple.getColor())&&apple.getWeight()>200;
    }
}

使用匿名类加上 Lambda 表达式:

使用匿名内部类:

我们上面所做的,依然有很多不足,我们需要声明很多只要实例化一次的类,费那么大的劲真的没必要,我们可以做的更好吗?Java 中有匿名内部类,我们利用它进一步改善代码:

    List<Apple> greenApples = filterApples(inventory, new ApplePredicate() {
        @Override
        public boolean test(Apple apple) {
            return "green".equals(apple.getColor());
        }
    });

使用 Lambda表达式

匿名内部类还是不够好,它比较笨重,占用了很多的空间。而且内部类一般比较难读。在 Java8 中新添加了 Lambda 表达式,我们如果利用了它只需要一句话就解决了问题:

    List<Apple> greenApples = filterApples(inventory, (Apple apple)->"green".equals(apple.getColor()));

将 List 类型抽象化

通往抽象的路上,我们还可以更进一步。目前我们还只适用于 Apple,我们还可以将 List 类型抽象化,从而超越眼前所要处理的问题:

public interface Predicate<T>{
    boolean test(T t);
}

修改filter方法:

public static <T> List<T> filter(List<T> list,Predicate<T> p){
    List<T> result = new ArrayList<>();
    for (T e : list) {
        if(p.test(e)){
            result.add(e);
        }
    }
    return result;
}

现在我们可以把filter方法用在香蕉、桔子、 Integer 或者是 String 上了,比如说我们要找出一个数字集合中的所有偶数,同样只需要一句话:

    List<Integer> evenNumbers=filter(numbers,(i)->i%2==0);

Lambda 表达式巩固练习

用 Comparator 升序排序

    //对苹果重量进行排序
    List<Apple> inventory = new ArrayList<>();
    inventory.add(new Apple("green", 155));
    inventory.add(new Apple("green", 177));
    inventory.add(new Apple("green", 244));
    inventory.add(new Apple("red", 123));

    //原始方法
    //Collections.sort(inventory, new Comparator<Apple>() {
    //    @Override
    //    public int compare(Apple a1, Apple a2) {
    //        return a1.getWeight() > a2.getWeight() ? -1 : a1.getWeight() < a2.getWeight() ? 1 : 0;
    //    }
    //});

    //使用 Lambda 表达式
    Collections.sort(inventory, (a1, a2) -> a1.getWeight() > a2.getWeight() ? 1 : a1.getWeight() < a2.getWeight() ? -1 : 0);
    for (Apple apple : inventory) {
        System.out.println(apple.getWeight());
    }

Thread 的使用

new Thread(()->System.out.println("Hello world")).start();

Android 中 Lambda 的简单使用

提前准备:

defaultConfig {
    ...
    jackOptions {
        enabled true
    }
}

compileOptions {
    sourceCompatibility JavaVersion.VERSION_1_8
    targetCompatibility JavaVersion.VERSION_1_8
}

点击事件调用:

    findViewById(R.id.btn).setOnClickListener((v) -> Toast.makeText(this, "hello", Toast.LENGTH_SHORT).show());

参考

本文纯属《Java8 实战》学习笔记

相关文章

  • 通过行为参数传递代码

    应对不断变化的需求 目标 在软件工程中一个众所周知的问题就是,不管你做什么,用户的需求肯定会变。比如一位农民第一天...

  • 通过行为参数化传递代码

    基础概念 在软件工程中,一个众所周知的问题是,不管你做什么,用户的需求肯定会变。行为参数化就是可以帮助你处理频繁变...

  • 通过行为参数化传递代码

    行为参数化是可以帮助你处理频繁变更的需求的一种软件开发模式 引言 1.首先我们看下实现从苹果列表中选出所有的绿色的...

  • JAVA8(二)

    行为参数化 什么是行为参数化 个人理解:把行为抽象出来进行封装,让代码适应需求的变化,并把行为或代码作为参数传递,...

  • 【Java 8实战笔记】通过行为参数化传递代码

    通过行为参数化传递代码 行为参数化是可以帮助你处理频繁变更的需求的一种软件开发模式。它意味着拿出一个代码块,将它准...

  • Java8 in Action chap2

    通过行为参数化传递代码 1.行为参数化定义: 可以帮助你处理频繁变更的需求的一种软件开发模式。一个方法可以接收不同...

  • java 8 笔记(一) 通过行为参数化传递代码

    案列:通过颜色筛选苹果一般代码逻辑写法 那么问题来了如果想筛选其他颜色的苹果,就要重新写一个筛选代码,而且这些代码...

  • 第二章:通过行为参数化传递代码

    应对不断变化的需求 行为参数化可以帮助你处理频繁变更的需求的一种软件开发模式,意味着它拿出一段代码块,准备好确不执...

  • java8之通过行为参数化传递代码

    在实习的这一段时间里,我深刻体会到一个道理,就是用户的需求不断的在变化,因而自己的代码也要进行重构,这说明了一个问...

  • 第二章 通过行为参数化传递代码

    1、行为参数化,就是一个方法接受多个不同的行为作为参数,并在内部使用它们,完成不同行为的能力。 2、行为参数化可让...

网友评论

    本文标题:通过行为参数传递代码

    本文链接:https://www.haomeiwen.com/subject/coueittx.html