通过行为参数传递代码

应对不断变化的需求

目标

在软件工程中一个众所周知的问题就是,不管你做什么,用户的需求肯定会变。比如一位农民第一天可能有一个想要查找库存中所有绿色苹果的功能,但第二天可能又想要找出重量大于 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 实战》学习笔记

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,088评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,715评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,361评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,099评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 60,987评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,063评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,486评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,175评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,440评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,518评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,305评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,190评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,550评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,880评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,152评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,451评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,637评论 2 335

推荐阅读更多精彩内容