永久无限免费看的appseo优化方法网站快速排名推广渠道
秋招结束后,间接性堕落了一段时间,学习几乎停止下来了。内心甚是焦灼,感觉生活很无趣!为了在参加工作后能够快速上手和成为一名优秀的中级开发者,从这篇文章开始将不断学习优秀的编码经验,学习是永无止境的。需要静下来,慢慢来!下面进入新篇章,技术提升篇。
应用情景
在工作中,往往我们的需求是多变的,那么如果我们只是简单的硬编码解决暂时的需求,那么当需求进行变更的时候我们的接口就需要变更来应对新的需求。但是有些情况下我们可以写一些通用接口来因对这种变化,即把频繁变化的东西提取出来,交给接口调用者来设计。在这种情况下,我们的接口就无需变化,而当需求变化了只需要重写频繁变化的对象即可。这就是策略模型的一种应用,接下来将用具体的例子来说明这种编码方式的具体实现。
此时,我们有个实体集合,需要通过对实体类的属性进行过滤选择出符合我们条件的一些实体。例如,第一周产品经理要求对属性一进行过滤,第二周需求变更又要把属性二加进去过滤,第三周又要把属性三考虑进去。。。。。于是,为了应对这种变化,我们的接口应该使用策略模式。如下:
具体实现
假设我们的实体简单一些,使用经典Student
作为实体,代码如下:
package strategy.demo.entity;public class Student {private String name;private int age;private int score;private String address;public Student(String name, int age, int score, String address){this.name = name;this.age = age;this.score = score;this.address = address;}public String getName() {return name;}public int getAge() {return age;}public int getScore() {return score;}public String getAddress() {return address;}public String toString(){return "Student [name=" + name + ", age=" + age + ", score=" + score + ", address=" + address + "]";}
}
首先,我们看下如果是硬编码的话,两个不同的需求应该会写下如下代码:
/*** 需求1:根据分数过滤学生* @param students* @param age* @return*/public List<Student> filterStudentByAge(List<Student> students, int age){ArrayList<Student> result = new ArrayList<>();for (Student student : students) {if (student.getAge() > age){result.add(student);}}return result;}/*** 需求2:根据分数和年龄过滤学生* @param students* @param age* @param score* @return*/public List<Student> filterStudentByScore(List<Student> students,int age , int score){ArrayList<Student> result = new ArrayList<>();for (Student student : students) {if (student.getScore() > score && student.getAge() > age){result.add(student);}}return result;}
上面的代码,需求2
的代码是对需求1
的升级维护,如果后面继续改动这个接口呢?所以,我们需要把频繁变动的东西提取出来作为一个策略对象,这里频繁变动的东西就是对实体Student
属性的判断,因此我们抽取出一个接口:
package strategy.demo.service;
import strategy.demo.entity.Student;
public interface filterStrategy {/*** 根据stduent的不同属性进行谓词过滤* @param student* @return*/public boolean filterStudent(Student student);
}
这个接口只有一个待实现的方法,接受一个Student
对象,返回是否符合过滤条件。而这个接口是由调用者实现的,其实这个设计思路在JDK
中非常常见。我们需要对一个对象集合 sort
排序,需要按照自己的排序策略重写Comparator
接口,还记得嘛?如下:
List<Student> allStudents = StudentHelps.getAllStudents();
allStudents.sort(new Comparator<Student>() {@Overridepublic int compare(Student o1, Student o2) {return o1.getAge()-o2.getAge;}
});
因此,使用策略模式,我们只需要写下一个过滤方法即可,其他的过滤策略由调用者去实现,如下:
public List<Student> filterStudentByStrategy(List<Student> students, filterStrategy strategy){ArrayList<Student> result = new ArrayList<>();for (Student student : students) {if (strategy.filterStudent(student)){result.add(student);}}return result;}
调用者实现时如下:
List<Student> allStudents = StudentHelps.getAllStudents();
List<Student> students3 = filterStudentByStrategy(allStudents, new filterStrategy() {@Overridepublic boolean filterStudent(Student student) {return student.getScore() > 80 && student.getAge() > 20;}
});
如此一来,无论对student
的过滤需求如何变化,始终可以通过 filterStrategy
接口实现,因此就更大程度上遵守高内聚低耦合的开发原则。这就是设计模式的美妙之处。
扩展
其实,在很多情况下,接口开发者会提供一个基本功能的 filterStrategy
接口实现类,里面有常用的过滤策略,调用者只需要用即可不需要每次都自己定义。这在JDK
源码和SpringBoot
中有太多的例子了,简单常用的由接口方提供,复杂不常用的我允许调用者自定义!
另外一个扩展就是,调用者在这里写匿名类可以使用Lambda
语法,具体参考我的博客 Lambda表达式常见用法 。Lambda
语法也是提供工作效率的神器,有时间推荐学习!因此,调用者使用Lambda
可以这样编写代码:
List<Student> students4 = filterStudentByStrategy(allStudents, student -> student.getScore() > 80 && student.getAge() > 20);
也就相当于把判断条件参数化了,这种做法叫做自定义函数接口(看起来是函数作为一个参数传递),看不懂的话去看下我的博客就会了,以上就是本期内容!共勉加油!