美文网首页
java8之通过行为参数化传递代码

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

作者: freelands | 来源:发表于2017-01-08 16:19 被阅读1802次

    在实习的这一段时间里,我深刻体会到一个道理,就是用户的需求不断的在变化,因而自己的代码也要进行重构,这说明了一个问题,我设计的代码不够好。所以,我觉得作为程序员来说,很重要的一点就是,让你的代码能够适应变化。

    行为参数化

    行为参数化,就是可以帮助我们处理频繁变化需求的一种软件开发模式,通俗的说,就是拿出一个代码块,把它准备好,却不去执行它。这个代码块以后可以被程序的其他部分调用,这就意味着我们可以推迟这块代码的执行。

    背景

    农场里有很多苹果,我们要去筛选特定的苹果,并记录下来。接下来,我们通过不断变化的需求来重构我们的代码。以下的场景都是我虚拟出来的,便于故事的发展,哈哈。

    片段一

    开始的别人给我们提出的需求是,筛选出来所有颜色是绿色的苹果。我心里想,你是不是鄙视我,这么简单,遍历一下把绿色苹果放到新的集合不就行了吗?我可是程序员哎,不是来打杂的,我可是要写牛逼的代码的。算了,还是写吧。代码如下:

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

    我把代码提交上去了,然后屁颠屁颠的做其他事情了,好像完成了一个很大的项目。。。

    片段二

    第二天,那个人又找到我,说道,你能不能实现一个筛选红色苹果的接口。
      我心里自言自语,卧槽,这需求变化那么快,昨天苹果还是绿的,今天就变红了,尼玛,这万恶的商家,催熟剂就是厉害。以后吃苹果还是吃青的吧。开始我准备把上面的代码copy下来,然后"green"改成"red",但是仔细一想,万一明天,让我写一个黑苹果的接口,那我不又要写一个。于是把color抽象出来,变成一个参数传进来,这样你要什么颜色的我都有了。突然,发现自己好厉害,抽象我都会。。。自我陶醉中。代码如下:

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

    写完之后,我心里在那嘀咕,万恶的资本主义,我把颜色都抽象出来了,看你需求怎么变。哥,可是--程序员。

    片段三

    晚上回去睡个安稳觉后,第二天,那人又来了,对,还是那个男人。我问他,是不是我写的接口不行。那人回到到,不是,是这边又有新的需求。突然之间,整个人都傻了。还有需求,算了,谁让我是程序员呢。之后,他和我说道,新的需求是区分,大苹果和小苹果,大苹果是重量大于500g的。二话没说,扭头就去撸代码。代码如下:

    public static List<Apple> filterApplesByWeight(List<Apple> inventory, int weight) {
            List<Apple> result = new ArrayList<>();
            for (Apple apple : inventory) {
                if (apple.getWeight() > weight) {
                    result.add(apple);
                }
            }
            return result;
        }
    

    聪明的我,早已想到需求会变,这次直接把这个接口抽象好,内心不禁有暗暗自喜。想到世界还有自己这种程序员,就说明世界还有希望。

    片段四

    第二天,他又来了,我知道,又要干活了。新的需求是,过滤那些红色的且重量大于500g的苹果。代码如下:

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

    不一会我就设计好了,但是却没有之前的开心了,不知道明天又会有什么新的需求,宝宝心里好苦啊。

    片段五

    之后令我很惊讶的是,没有新的需求提出了,我一遍暗地自喜,一遍在思考一个问题。为什么,需求变,我的代码就要变化那么大呢?需求都是和选择苹果相关,选择什么样的苹果是通过参数来传递的,那么有没有一种方法,通过一个借口来实现传递不同的参数呢?
      如果熟悉设计模式的你,应该能想到策略模式。接口一致,按需传递该接口对应的实例。也是面向接口编程的一种体现。先设计一个接口,用来承载选择苹果的逻辑,代码如下:

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

    现在我们可以用不同的实现来传递不同的逻辑了,如下:

    //绿色苹果
    public class GreenApplePredicate implements ApplePredicate {
        @Override
        public boolean test(Apple apple) {
            return apple.getColor().equals("green");
        }
    }
    
    //大苹果
    public class BigApplePredicate implements ApplePredicate {
        @Override
        public boolean test(Apple apple) {
            return apple.getWeight() > 500;
        }
    }
    
    //代码主体
    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;
        }
    

    可以看出,代码主体以后我们就不会改变了,只要通过传入不同的行为(过滤逻辑),就行了,而且这里只传递一个参数,却可以传递多种行为,多种行为,一个参数,是不是很nice。我又开始沾沾自喜。。感觉自己又变得厉害了,晚上回去睡得很任性。

    片段六

    第二天,还是没有新的需求,我们都知道,程序员没事做,这宝贵的资源就白白的浪费,别人不心疼,我都心疼。于是,自己又去捣鼓昨天的那块代码去了。
      刚开始用起来还是蛮舒服的,但是用久了发现,每次一个新的逻辑都要去实现这个接口,太难受了,突然,想到了java还有匿名类,于是,之后直接用匿名类来实现这个逻辑。代码如下:

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

    这样看起来,的确比之前简洁多了,不要显式的去实现一个接口。但是细细的观察了一下,这里面真正有用的代码就这一行:

    return apple.getColor().equals("green");
    

    那么有没有什么办法,只给filterApples方法只传递这样真正逻辑性的代码呢?幸好,java8提供了行为参数化的支持。好吧,既然都支持了,还等什么,干吧,下面这一行代码就解决了。

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

    什么,我没有看错吧,这么神?你并没有看错,传递的参数叫做lambda表达式,如果还嫌不够简洁,你可以把类型去掉,像这样:

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

    经历了以上几个阶段,我们重构了很多次代码,让接口能够适应需求的变化,一方面,我们又不想编写冗余的代码,通过lambda表达式来传递核心代码,程序员不在做一些不必要的工作,真的是太好了。

    相关文章

      网友评论

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

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