泛型和模板设计模式

简介: 当两个或多个功能很大一部分实现都一样,只有其中一部分处理逻辑不同的情况下。我们通常都会采用模板设计模式来实现,这样既可以满足功能的需求也可以很好地实现代码的维护。这也正是设计模式的精髓所在。但是,如果有这样一个需求,该如何实现呢?既满足了模板设计模式的条件,也就是说两个或多个功能的总体实现流程是一致的,只是部分处理逻辑上存在差异;但有点特别的是根据不同的功能,返回值类型有所差别。

当两个或多个功能很大一部分实现都一样,只有其中一部分处理逻辑不同的情况下。我们通常都会采用模板设计模式来实现,这样既可以满足功能的需求也可以很好地实现代码的维护。这也正是设计模式的精髓所在。但是,如果有这样一个需求,该如何实现呢?既满足了模板设计模式的条件,也就是说两个或多个功能的总体实现流程是一致的,只是部分处理逻辑上存在差异;但有点特别的是根据不同的功能,返回值类型有所差别。这样的情况下我们可以通过模板设计模式结合泛型来很好地实现相应的功能。

技术要点

我们先来看下模板设计模式的技术点,模板设计模式使用的技术点是:

  • 抽象类:控制程序总体逻辑(骨架)
  • 实现类:继承于抽象类,实现具体差异部分的逻辑

此处要求返回值类型根据不同的业务,返回不同类型的响应信息,因此还需要用到泛型。

  • 泛型于设计模式的结合

具体实现如下:

响应实体泛型的定义

public class Result<T> {
    T response;

    public T getResponse() {
        return response;
    }

    public void setResponse(T response) {
        this.response = response;
    }
}

程序骨架,模板设计模式

public abstract class BaseService<T> {

    /**
     * 定义了程序的主要实现流程-骨架
     * @return
     */
    public Result handle() {
        Result result = new Result();
        method1();
        result.setResponse(method2());
        return result;
    }

    /**
     * 返回值-泛型,根据不同的业务返回不同的响应类型
     * @return
     */
    private T method2() {
        T response = initResponse();//获取子类初始化的响应实例
        System.out.println("BaseService method2");
        return response;
    }

    /**
     * 公共处理业务
     */
    private void method1() {
        System.out.println("BaseService method1");
    }

    /**
     * 响应类型-泛型,提供出去给具体实现类进行初始化
     * @return
     */
    protected abstract T initResponse();
}

模板抽象类需要注意的几个点:

  • BaseService<T>,模板类提供泛型支持
  • Result里的response返回值类型,由子类进行控制
  • 返回值类型,通过提供抽象方法,子类进行初始化来实现。protected abstract T initResponse();

至此,根据不同业务类型返回不同响应类型的模板设计模式的总体框架已经搭建完成了。

差异业务实现类

  • 定义返回值类型
  • 继承于模板抽象类-业务实现类
  • 实现模板抽象类的方法

返回值类型

/**
 *
 * 返回值类型基类,公共部分
 */
public class BaseResponse {
    private int age;
    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}
/**
 *
 * 返回值类型
 */
public class ChinaResponse extends BaseResponse {

    List<String> province;

    public List<String> getProvince() {
        return province;
    }

    public void setProvince(List<String> province) {
        this.province = province;
    }
}
/**
 *
 * 返回值类型
 */
public class AmericaResponse extends BaseResponse {
    List<String> state;

    public List<String> getState() {
        return state;
    }

    public void setState(List<String> state) {
        this.state = state;
    }
}

继承于模板抽象类-业务实现类

/**
 *
 * 业务实现类1
 */
public class ChinaServiceImpl extends BaseService<ChinaResponse> {

    public Result<ChinaResponse> handle() {
        Result result = super.handle();
        return result;
    }

    /**
     * 实现抽象方法,初始化返回值类型
     * @return
     */
    @Override
    protected ChinaResponse initResponse() {
        return new ChinaResponse();
    }
}
/**
 *
 * 业务实现类2
 */
public class AmericaServiceImpl extends BaseService<AmericaResponse> {

    /**
     * 实现抽象方法,初始化返回值类型
     * @return
     */
    @Override
    protected AmericaResponse initResponse() {
        return new AmericaResponse();
    }
}

测试

    public static void main(String[] args) {
        BaseService<ChinaResponse> baseService = new ChinaServiceImpl();
        Result<ChinaResponse> result = baseService.handle();
        ChinaResponse chinaResponse = result.getResponse();
        System.out.println(chinaResponse.toString());

        BaseService<AmericaResponse> americaService = new AmericaServiceImpl();
        Result<AmericaResponse> americaResult = americaService.handle();
        AmericaResponse americaResponse = americaResult.getResponse();
        System.out.println(americaResponse.toString());
    }

测试结果:

BaseService method1
BaseService method2
com.designpatterns.template.generic.response.ChinaResponse@28d93b30
BaseService method1
BaseService method2
com.designpatterns.template.generic.response.AmericaResponse@1b6d3586

总结

通过模板设计模式和泛型的结合,可以非常灵活地处理通用流程和处理逻辑之间的差异;又可以很好地满足不同业务不同返回值类型的需求。

相关文章
|
3月前
|
设计模式 算法 自动驾驶
常见的设计模式(模板与方法,观察者模式,策略模式)
随着时间的推移,软件代码越来越庞大,随着而来的就是如何维护日趋庞大的软件系统。在面向对象开发出现之前,使用的是面向过程开发来设计大型的软件程序,面向过程开发将软件分成一个个单独的模块,模块之间使用函数进行组合,最后完成系统的开发,每次需要修改软件,如果不涉及好各个模块的关系,就会导致软件系统难以维护,从而导致软件变得不可使用。面向对象方法用对象模拟问题域中的实体,以对象间的联系刻画实体间联系
89 2
|
3月前
|
设计模式
装饰者设计模式(二)番外篇 装饰者设计模式和静态代理设计模式区别
装饰者设计模式(二)番外篇 装饰者设计模式和静态代理设计模式区别
|
设计模式 JavaScript 前端开发
设计模式之构造函数模式
设计模式之构造函数模式
63 0
|
设计模式 前端开发 JavaScript
前端实现设计模式之工厂方法
在软件开发中,设计模式是一种被广泛应用的解决方案,它提供了一种结构化的方法来解决常见的设计问题。工厂方法模式是一种创建型设计模式,它将对象的创建延迟到子类中,从而实现了更高层次的灵活性和可扩展性。本文将介绍如何在前端中实现工厂方法模式,并提供具体的代码示例和解读。
|
设计模式
设计模式关于工厂方法学习总结
设计模式关于工厂方法学习总结
49 0
|
设计模式 算法
设计模式三种类型
设计模式三种类型
|
设计模式
设计模式——工厂模式详解(代码演示)
设计模式——工厂模式详解(代码演示)
|
设计模式 SQL Java
设计模式之 解释器模式 Java实例代码演示
设计模式之 解释器模式 Java实例代码演示
75 0
|
设计模式 JavaScript 安全
TypeScript | 设计模式05 - 代理模式
为某对象提供一种代理以控制对该对象的访问。即客户端通过代理间接地访问该对象,从而限制、增强或修改该对象的一些特性。 比如,某对象需访问目标对象,但由于某种情况,不适合或不能直接访问目标对象,通过一个中介进行访问,这个中介就是代理对象。
482 0
|
设计模式 JavaScript
TypeScript | 设计模式04 - 建造者模式
建造者模式 也叫做 生成器模式
200 0