设计模式之模板方法模式

在上一篇记录中,做了菜单树的实现,其中有一行代码 Collections.sort(children);它的功能是根据菜单的父节点编号PID进行排序,实现代码如下:

复制代码
/**
 * 系统菜单实体类
 */

public class SystemMenuInfo implements Serializable,Comparable<SystemMenuInfo>{

    private static final long serialVersionUID = -612577710282986839L;

    private Integer menuId;

    private String menuName;

    private String url;

    private Integer menuLevel;

    private String iconCls;

    private Integer sortId;

    private Integer pid;

    private String description;

    private String remark;

    private String creator;

    private Date createDate;

    private String modifier;

    private Date modifyDate;

    private String menuState;

    private List<SystemMenuInfo> children;
复制代码
    /**
     * 根据sortId升序排序
     * @param menu
     * @return
     */
    @Override
    public int compareTo(@NotNull SystemMenuInfo menu) {
        if(this.sortId>menu.sortId){
            return 1;
        }else if(this.sortId<menu.sortId){
            return -1;
        } else{
            return 0;
        }
    }
}

在这里菜单类实现了Comparable接口,只需要实现compareTo中的比较逻辑,在Collections调用sort方法时就可以实现排序的功能,如果要改变比较的方式,只需要修改比较方法中的代码,从而达到排序方式改变的目的,而sort()方法中的代码基本固定,不需要做修改,这就是一处典型模板方法模式的应用。

模板模式简介:

模板方法模式主要用于对算法或行为逻辑进行封装,即如果多个类中存在某些相似的算法逻辑或者行为逻辑,可以将这些相似的逻辑提取到模板方法类中实现,然后让相应的子类根据需要实现某些自定义的逻辑。

思想:

模板方法模式准备一个抽象类,将部分逻辑以具体方法的形式实现,然后声明一些抽象方法来迫使子类实现 剩余的逻辑。不同的子类可以以不同的方式实现这些抽象方法,从而对剩余的逻辑有不同的实现。先制定一 个顶级逻辑框架,而将逻辑的细节留给具体的子类去实现。

实现:

//准备⼀个抽象类,将部分逻辑以具体⽅法的形式实现

public abstract class SortUtil {

//声明⼀些抽象方法来迫使子类实现剩余的逻辑

public abstract int compareTo(T o1, T o2);

public void sort(T arr) {

//.先制定⼀个顶级逻辑框架,而将逻辑的细节留给具体的子类去实现

for (int i = 0; i < arr.length; i++) {

for (int j = 0; j < arr.length - 1 - i; j++) {

int m = compareTo(arrj, arrj + 1);

if (m > 0) {

T temp = arrj;

arrj = arrj + 1;

arrj + 1 = temp;

}

}

}

}

}

class StringSortUtil extends SortUtil {

@Override

public int compareTo(String o1, String o2) {

return - o1.length() + o2.length();

}

}

public class Test {

public static void main(String\[\] args) {

String\[\] arr = {"aa","bbbbb","ccc"};

StringSortUtil util = new StringSortUtil();

util.sort(arr);

}

}

也可以定义一个接口,然后在模板方法sort的参数中传入这个接口,接口定义如下:

public interface SomeComparator {

public int compareTo(T o1, T o2) ;

}

public class SortUtil {

public static void sort(Object\[\] arr,SomeComparator sc) {

for (int i = 0; i < arr.length; i++) {

for (int j = 0; j < arr.length - 1 - i; j++) {

int m = sc.compareTo(arrj, arrj + 1);

if (m > 0) {

Object temp = arrj;

arrj = arrj + 1;

arrj + 1 = temp;

}

}

}

}

}

public class Test {

public static void main(String\[\] args) {

String\[\] arr = {"aa","bbbbb","ccc"};

// 迫使调用者 实现 剩余逻辑

SortUtil.sort(arr, new SomeComparator() {

@Override

public int compareTo(String o1, String o2) {

return o1.length() - o2.length();

}

});

相比于继承抽象类的方式,传入接口的方法更方便,它不需要每次使用的时候都给一个相应的子类实现,传入的这个接口也叫回调接口。

总结:

当一个方法中 有多行代码不确定时,(可能只有当使用到该方法时才能确定),一定需要使用模板模式。

模板模式的使用方式:

实现顶层业务逻辑

创建接口及抽象方法

将接口类型的对象作为参数,传入到第一步的方法中

在不确定应该写什么代码的位置 调用接口中的抽象方法

相关推荐
磊 子15 小时前
C++设计模式
javascript·c++·设计模式
许彰午18 小时前
34_Java设计模式之单例模式
java·单例模式·设计模式
石一峰6991 天前
C 语言函数设计模式实战经验
c语言·开发语言·设计模式
qq_297574672 天前
设计模式系列文章(基础篇第22篇):访问者模式——分离数据结构与操作,实现灵活扩展
数据结构·设计模式·访问者模式
刀法如飞2 天前
领域驱动 vs 本体驱动:DDD 代码建模与 Ontology 语义建模的对比分析
设计模式·架构设计·领域驱动
我爱cope2 天前
【Agent智能体26 | 多智能体-多智能体工作流】
人工智能·设计模式·语言模型·职场和发展
咖啡八杯3 天前
【无标题】
java·后端·设计模式
折哥的程序人生 · 物流技术专研3 天前
Java 23 种设计模式:从踩坑到精通 | 适配器模式 —— 让不兼容的接口也能一起工作
java·设计模式·面试·适配器模式·单一职责原则
布朗克1683 天前
33 设计模式精讲
java·单例模式·设计模式