Java命令模式以及来自lambda的优化(二)

使用函数抽象行为进行优化

将这些做饭的命令抽象成函数接口,然后指定一个执行者,这样的接口在Java8 4大函数接口中属于Consumer接口,也就是消费者接口,下面就用Consumer接口进行行为抽象
厨房类依旧不变

public class Kitchen {
    public void beefRice(){
        System.out.println("一份牛肉饭做好了!");
    }

    public void scrambledEggsWithTomatoes(){
        System.out.println("一份西红柿炒鸡蛋做好了!");
    }

    public void beerDuck(){
        System.out.println("一份啤酒鸭做完啦!");
    }
}

服务员类也很容易,原先装的是一个个命令对象,现在直接一点,直接将行为放进去

public class Waiter {
    /**
     * 此时队列装载的不再是命令对象了,而是更直接的厨房类的行为
     */
    private final Queue<Consumer<Kitchen>> orders;

    public Waiter() {
        orders = new ArrayDeque<>();
    }

    /**
     * 添加订单
     * @param kitchenAction 厨房的具体行为
     */
    public final void setOrders(Consumer<Kitchen> kitchenAction) {
        System.out.printf("添加订单成功! 订单时间: %s \n", LocalDateTime.now());
        orders.add(kitchenAction);
    }

    /**
     * 这里增加一个执行者参数,来对队列中的行为进行操作
     * @param kitchen 执行者,用于执行队列中的行为
     */
    public final void notifyKitchen(Kitchen kitchen) {
        while (orders.peek() != null) {
            orders.peek().accept(kitchen);
            orders.remove();
        }
    }
}

客户端代码,简单、清晰的惊人,代码自带注释效果,无论是简短性还是可阅读性,都比之前的要好上很多,中间的营业代码很直观,直接阅读代码就很清楚的看到添加了哪些行为到订单队里中

public class Client {
    public static void main(String[] args) {
        //准备厨房,服务员,菜单命令工作
        Kitchen kitchen = new Kitchen();
        Waiter waiter = new Waiter();

        //开始营业
        System.out.println("=======================添加订单环节=======================");
        // 顾客:服务员 一份牛肉饭!
        waiter.setOrders(Kitchen::beefRice);
        // 顾客:服务员 一份啤酒鸭!
        waiter.setOrders(Kitchen::beerDuck);
        // 顾客:服务员 一份西红柿炒鸡蛋!
        waiter.setOrders(Kitchen::scrambledEggsWithTomatoes);
        // 顾客:服务员 两份啤酒鸭!
        waiter.setOrders(Kitchen::beerDuck);
        waiter.setOrders(Kitchen::beerDuck);

        System.out.println("==========服务员将订单送至厨房,厨房按照订单顺序开始做饭=========");
        //服务员通知厨房按照订单顺序开始做
        waiter.notifyKitchen(kitchen);

    }
}

输出结果

=======================添加订单环节=======================
添加订单成功! 订单时间: 2017-10-16T03:19:40.003 
添加订单成功! 订单时间: 2017-10-16T03:19:40.019 
添加订单成功! 订单时间: 2017-10-16T03:19:40.020 
添加订单成功! 订单时间: 2017-10-16T03:19:40.020 
添加订单成功! 订单时间: 2017-10-16T03:19:40.021 
==========服务员将订单送至厨房,厨房按照订单顺序开始做饭=========
一份牛肉饭做好了!
一份啤酒鸭做完啦!
一份西红柿炒鸡蛋做好了!
一份啤酒鸭做完啦!
一份啤酒鸭做完啦!

Process finished with exit code 0

到这里,已经没有其他类的代码了!类的数量由原先的7个变成了3个,并且由于服务员类(Invoke角色)依然存在,原先的解耦复合控制扩展等优点,一个都没少,与此同时客户端的代码也清爽了不少。试想一下,假如现在厨房有100道做菜的方法,按照原先的方法实现的类的数量应该是3(客户端+厨房+服务员) + 1(抽象命令接口) + 100(具体命令接口) = 104个类,而采用lambda之后,依旧只需要三个类!并且原先的优点完全保留了下来。

函数补充优化(update 17.10.24)

在整理关于方法引用转换函数接口资料的时候忽然觉得命令模式优化这里似乎可以再稍微变动一下使得代码呈现链式(之前也有过类似的想法,但是没想到门路,今天想到了,就补充一下)
上文的代码其实已经成型了,这里做的优化是将多次调用设置成链式调用法。

链式调用一

只需要将waiter类的setOrders方法的返回值设置为this本身即可。

    public final Waiter setOrders(Consumer<Kitchen> kitchenAction) {
        System.out.printf("添加订单成功! 订单时间: %s \n", LocalDateTime.now());
        orders.add(kitchenAction);
        return this;
    }

客户端链式调用

public class Client {
    public static void main(String[] args) {
        //准备厨房,服务员,菜单命令工作
        Kitchen kitchen = new Kitchen();
        Waiter waiter = new Waiter();
        //开始营业
        System.out.println("=======================添加订单环节=======================");
        // 顾客:服务员 一份牛肉饭!
        // 顾客:服务员 一份啤酒鸭!
        // 顾客:服务员 一份西红柿炒鸡蛋!
        // 顾客:服务员 两份啤酒鸭!
        // 顾客:服务员 两份啤酒鸭!
        //这里通过setOrders完成链式调用,和之前的效果完全一样,但是提高了可读性与间接性
        waiter.setOrders(asConsumer(Kitchen::beefRice))
                .setOrders(Kitchen::beerDuck)
                .setOrders(Kitchen::scrambledEggsWithTomatoes)
                .setOrders(Kitchen::beerDuck)
                .setOrders(Kitchen::beerDuck);
        
        System.out.println("==========服务员将订单送至厨房,厨房按照订单顺序开始做饭=========");
        //服务员通知厨房按照订单顺序开始做
        waiter.notifyKitchen(kitchen);
    }
}

输出结果和上面一样,这里不再显示

链式调用二

这里是通过一个方法引用的包装器,然后通过function接口的andThen方法链式的讲订单存入订单队列中。

  • 这里与上文的链式优化一的是有区别的,区别在于,上文中无论是普通的setOrders还是return this 链式的setOrders的 日志信息是会输出5条的(进行了5次的setOrders动作),而这里只会输出一条,原因是这里的一条动作里andThen了4个动作,这一条动作链作为一个整体传入了Setorders,因此只会有一条日志信息输出。
  • 至于什么时候使用,根据实际情况,例如通知厨房做一道便当快餐,而便当快餐假设是由一份牛肉饭+一份啤酒鸭构成的,那么在这里这一个订单就可以使用这样的andThen来进行构造,这样的链式组合其实是有序的,其实这就是建造者模式的lambda表示形式。
  • 使用这样的方法,你的设计里就是包含了命令模式+建造者模式,以lambda形式表现出来,当然如果不按照顺序的话,这样的模式你还可以理解成组合的形式。
    lambda的方法引用固然使得代码清晰可见,但是坏处是一旦使用了方法引用,就无法进行lambda的链式调用,也就是无法使用andThen这样的链式方法,但是我们通过一个中转站,先将方法引用转换成函数接口,再链式调用,因为方法引用本质上是lambda的语法糖,下面是转换方法,十分简易。
public class FunctionCastUtil {
    public static <T> Consumer<T> asConsumer(Consumer<T> consumer) {
        return consumer;
    }
}

客户端代码

import static com.lambda.functionutils.FunctionCastUtil.*;

public class Client {
    public static void main(String[] args) {
        //准备厨房,服务员,菜单命令工作
        Kitchen kitchen = new Kitchen();
        Waiter waiter = new Waiter();
        //开始营业
        System.out.println("=======================添加订单环节=======================");
        // 顾客:服务员 一份牛肉饭!
        // 顾客:服务员 一份啤酒鸭!
        // 顾客:服务员 一份西红柿炒鸡蛋!
        // 顾客:服务员 两份啤酒鸭!
        // 顾客:服务员 两份啤酒鸭!
        //这里使用anThen完成链式调用,注意这里是只有一条订单,所以要注意使用的场合
        waiter.setOrders(asConsumer(Kitchen::beefRice)
                        .andThen(Kitchen::beerDuck)
                        .andThen(Kitchen::scrambledEggsWithTomatoes)
                        .andThen(Kitchen::beerDuck)
                        .andThen(Kitchen::beerDuck));

        System.out.println("==========服务员将订单送至厨房,厨房按照订单顺序开始做饭=========");
        //服务员通知厨房按照订单顺序开始做
        waiter.notifyKitchen(kitchen);
    }
}

因为输出结果不同,所以这里贴一下输出结果,根据输出结果就明白这里的订单只记录了一条,这一条里面包含了要做多种菜,在厨房那里实际完成的结果是一样的,大家可以根据不同的需要使用不同的链式优化

=======================添加订单环节=======================
添加订单成功! 订单时间: 2017-10-24T07:01:59.444 
==========服务员将订单送至厨房,厨房按照订单顺序开始做饭=========
一份牛肉饭做好了!
一份啤酒鸭做完啦!
一份西红柿炒鸡蛋做好了!
一份啤酒鸭做完啦!
一份啤酒鸭做完啦!

Process finished with exit code 0

结尾

使用lambda优化之后的命令模式在保证优点的同时极大的减少了代码量,简直完美。这就是语言特性所带来的力量,简单的几处修改就能获得如此多的受益,这也是我为什么这么喜欢lambda的原因。

欢迎加入学习交流群569772982,大家一起学习交流。

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

推荐阅读更多精彩内容