java设计模式-模板方法(七)_java中模板模式的远程调用-程序员宅基地

技术标签: 模板方法  java设计模式  

比如我们项目中有很多调用第三方系统的功能(如支付宝或微信),实际无非就是以下几步:
1.组装调用支付宝或微信需要的参数
2.请求支付宝或微信
3.解析支付宝或微信的响应参数
4.针对调用异常处理(非必须,子类可重写)
这是固定的流程或者模板。
接下来咱们以请求支付宝和微信为例:

先定义一个抽象类
package com.lifeng.patterns.templatemethod;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.HashMap;
import java.util.Map;

/**
 * Created by lifeng on 2019/10/16.
 * 远程调用抽象类
 */
public abstract class AbstractRemoteCall {
    
    private static final Logger log = LoggerFactory.getLogger(AbstractRemoteCall.class);

    public Map<String,Object> execute(Map<String,Object> reqMap) throws Exception{
    
        Object vo = getBean(reqMap);
        Map<String, Object> rspMap = new HashMap<String, Object>();
        Object ret = null;
        try {
    
            log.debug("请求数据为:"+vo);
            ret = executeBiz(vo);
            log.debug("返回数据为:"+ret);
        } catch (Exception e) {
    
            exceptionHandle(e, rspMap);
        }
        rspMap=getRspMap(ret,rspMap);
        return rspMap;
    }

    /** 组装请求参数bean*/
    protected abstract Object getBean(Map<String, Object> reqMap);

    /** 执行业务*/
    public abstract Object executeBiz(Object obj) throws Exception;

    /** 解析转换响应参数*/
    public abstract Map<String, Object> getRspMap(Object ret,Map<String, Object> rspMap) throws Exception;

    /** 异常处理*/
    protected void exceptionHandle(Exception e,Map<String, Object> rspMap) throws Exception{
    
        if(e instanceof NullPointerException){
    
            log.error("参数异常",e);
            rspMap.put("code","423");
            rspMap.put("msg","参数异常");
        }else if(e instanceof Exception){
    
            log.error("系统异常",e);
            rspMap.put("code","500");
            rspMap.put("msg","异常异常");
        }
    }
}

抽象类定义了流程模板,在抽象类中定义了一系列的操作,每个操作可以使具体的,也可以是抽象的,每个操作对应一个算法的步骤,在子类中可以重新定义或实现这些步骤。execute这个方法用于定义一个算法结构,模板方法不仅可以调用在抽象类中实现的基本方法,也可以调用在抽象类的子类中实现的基本方法,还可以调用其他对象中的方法。

用于实现在父类中声明的抽象基本操作,也可以覆盖在父类中已经实现的具体基本操作

调用支付宝实现类

涉及到的支付宝和微信的Dto 略,自己写下哦。。。

/**
 * Created by lifeng on 2019/10/16.
 * 调用支付宝
 */
public class AliPayReromeCall extends AbstractRemoteCall{
    
    @Override
    protected Object getBean(Map<String, Object> reqMap) {
    
        AlipayDto alipayDto = new AlipayDto();
        alipayDto.setpId((String) reqMap.get("merid"));
        alipayDto.setOrderNo((String) reqMap.get("orderno"));
        alipayDto.setAmt((Long) reqMap.get("amt"));
        alipayDto.setBody((String) reqMap.get("body"));
        return alipayDto;
    }

    @Override
    public Object executeBiz(Object obj) throws Exception {
    
        //TODO 模拟调用远程
        AlipayDto alipayDto = new AlipayDto();
        alipayDto.setRetCode("200");
        alipayDto.setRetMsg("交易成功");
        alipayDto.setOrderNo("1111111111");
        return alipayDto;
    }

    @Override
    public Map<String, Object> getRspMap(Object ret, Map<String, Object> rspMap) throws Exception {
    
        AlipayDto alipayDto = (AlipayDto) ret;
        rspMap.put("orderno",alipayDto.getOrderNo());
        rspMap.put("rspcode",alipayDto.getRetCode());
        rspMap.put("rspmsg",alipayDto.getRetMsg());
        return rspMap;
    }
}

调用微信实现类
/**
 * Created by lifeng on 2019/10/16.
 * 调用微信
 */
public class WechatPayReromeCall extends AbstractRemoteCall{
    
    @Override
    protected Object getBean(Map<String, Object> reqMap) {
    
        WechatpayDto wechatpayDto = new WechatpayDto();
        wechatpayDto.setMchId((String) reqMap.get("merid"));
        wechatpayDto.setMerNo((String) reqMap.get("orderno"));
        wechatpayDto.setTranAmt((Long) reqMap.get("amt"));
        wechatpayDto.setGoodsName((String) reqMap.get("body"));
        return wechatpayDto;
    }

    @Override
    public Object executeBiz(Object obj) throws Exception {
    
        //TODO 模拟调用远程
        WechatpayDto wechatpayDto = new WechatpayDto();
        wechatpayDto.setCode("200");
        wechatpayDto.setMsg("交易成功");
        wechatpayDto.setMerNo("11111111111");
        return wechatpayDto;
    }

    @Override
    public Map<String, Object> getRspMap(Object ret, Map<String, Object> rspMap) throws Exception {
    
        WechatpayDto wechatpayDto = (WechatpayDto) ret;
        rspMap.put("orderno",wechatpayDto.getMerNo());
        rspMap.put("rspcode",wechatpayDto.getCode());
        rspMap.put("rspmsg",wechatpayDto.getMsg());
        return rspMap;
    }
}
客户端调用时:
public class MyClient {
    
    public static void main(String[] args) throws Exception {
    
        //请求支付宝
        AliPayReromeCall aliPayReromeCall = new AliPayReromeCall();
        Map<String,Object> payReqMap = new HashMap<String,Object>();
        payReqMap.put("merid","4543345454");
        payReqMap.put("orderno","20191012112312345");
        payReqMap.put("amt",1L);
        payReqMap.put("body","iphone11");
        Map<String,Object> resMap = aliPayReromeCall.execute(payReqMap);
        System.out.println("支付宝返回:"+resMap);

        //请求微信
        WechatPayReromeCall wechatPayReromeCall = new WechatPayReromeCall();
        Map<String,Object> wresMap = wechatPayReromeCall.execute(payReqMap);
        System.out.println("微信返回:"+resMap);
    }
}

假如某天又对接了银联云闪付,流程还一样的,只需实现抽象类,实现父类的方法。也符合

开闭原则(OCP):是5大设计原则中最基础的设计原则,指的是一个软件实体如类、接口等应该对扩展开放,对修改关闭。
模板方法:定义一个操作算法中的框架,而将这些步骤延迟加载到子类中

优点:
1.让父类控制子类方法的调用顺序。只需要考虑方法的实现
2.不需要考虑方法在何种情况下被调用。实现代码复用。
3.模板类,将核心算法与具体子类相分离,子类只需要覆写具体的某几步流程即可
常用场景:
1.一次性实现一个算法的不变部分,并将可变的行为留给子类来实现;
2.各子类中公共的行为应被提取出来并集中到一个公共父类中以避免代码重复

是不是很简单,你可能平时开发也使用这是设计,只是不知道它原来是模板设计模式。哈哈
使用了模板方法的地方:servlet中HttpServlet的父类GenericServlet的service方法,HttpServlet重写了service,这段代码流程是固定的,子类需要重写doGet 或者doPost等
在这里插入图片描述
还有RedisTemplate JdbcTemplate都使用了模板方法。
代码参考码云gitee https://gitee.com/lifengit/javapatterns

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/u010509052/article/details/102600501

智能推荐

spark2原理分析-RDD的shuffle简介_rdd shuffle-程序员宅基地

文章浏览阅读868次。概述本文介绍RDD的Shuffle原理,并分析shuffle过程的实现。RDD Shuffle简介spark的某些操作会触发被称为shuffle的事件。shuffle是Spark重新分配数据的机制,它可以对数据进行分组,该操作可以跨不同分区。该操作通常会在不同的执行器(executor)和主机之间复制数据,这使shuffle成为复杂且非常消耗资源的操作。Shuffle背景为了理解shuf..._rdd shuffle

python3 os.system 异步执行_Python执行系统命令的方法 os.system(),os.popen(),commands-程序员宅基地

文章浏览阅读3.8k次。最近在做那个测试框架的时候发现 Python 的另一个获得系统执行命令的返回值和输出的类。1.最开始的时候用 Python 学会了 os.system() 。这个方法是拥塞的。os.system('ping www.baidu.com')2.通过 os.popen() 返回的是 file read 的对象,对其进行读取 read() 的操作可以看到执行的输出。这个方法是后台执行,不影响后续脚本运行..._os.system异步执行

CM+CDH安装搭建全过程(总结版)_cloudera manager server gc cpu usage is at 10% or -程序员宅基地

文章浏览阅读2.9k次。目录第一次搭建CM、CDH第二次搭建CM、CDH搭建环境:搭建过程:报错过程:总结复盘:第三次搭建CM、CDH搭建环境:搭建过程:报错过程:总结复盘:第四次搭建CM、CDH搭建环境:搭建过程:报错过程:总结复盘:第一次搭建CM、CD..._cloudera manager server gc cpu usage is at 10% or more of total process time

内核开发调试printk_printk 头文件-程序员宅基地

文章浏览阅读706次。进行内核开发调试在进行驱动开发的过程中往往要打印一些信息来查看是否正确类似于printf,以下将介绍在内核开发常用的调试方法。.(第一次写文章,内容可能不咋样勿喷呀)内容一、printk介绍二、如何查看并修改消息级别在应用程序采用printf打印调试、内核驱动采用printk打印调试。printk函数打印数据到console缓冲区,打印的格式方类似printf。printk函数说明头文件:<linux/kernel.h>int printk(KERN_XXX const_printk 头文件

Kafka原理、部署与实践——深入理解Kafka的工作原理和使用场景,全面介绍Kafka在实际生产环境中的部署_kafka如何负载使用一台对外的机器-程序员宅基地

文章浏览阅读2.5k次。随着互联网的发展,网站的流量呈爆炸性增长,传统的基于关系型数据库的数据处理无法快速响应。而NoSQL技术如HBase、MongoDB等被广泛应用于分布式数据存储与处理,却没有提供像关系型数据库一样的ACID特性、JOIN操作及完整性约束。因此,很多公司或组织开始转向Apache Spark、Flink、Beam等新一代大数据处理框架来处理海量数据。然而,由于新一代大数据处理框架依赖于HDFS等文件系统,导致集群规模扩容困难、成本高昂。另一方面,云计算平台的出现让用户可以快速部署、扩展大数据处理集群。_kafka如何负载使用一台对外的机器

麒麟KYLINOS桌面操作系统2303上安装tigervnc_麒麟系统电脑安装vncserver-程序员宅基地

文章浏览阅读1.4k次。hello,大家好啊,今天给大家带来在麒麟桌面操作系统2303上安装tigervnc的文章,本篇文章给大家讲述如何安装并且远程连接使用,后面会给大家更新如何将tigervnc做成桌面图标点击即可开启及关闭,欢迎大家浏览分享转发。_麒麟系统电脑安装vncserver

随便推点

设备驱动模型:总线-设备-驱动_总线设备驱动模型-程序员宅基地

文章浏览阅读1.3k次,点赞5次,收藏12次。总线是连接处理器和设备之间的桥梁代表着同类设备需要共同遵循的工作时序。总线驱动:负责实现总线行为,管理两个链表。name:指定总线的名称,当新注册一种总线类型时,会在 /sys/bus 目录创建一个新的目录,目录名就是该参数的值;bus_groups、dev_groups、drv_groups:分别表示 总线、设备、驱动的属性。通常会在对应的 /sys 目录下在以文件的形式存在,对于驱动而言,在目录 /sys/bus//driver/ 存放了驱动的默认属性;_总线设备驱动模型

TensorFlow精进之路(十五):深度神经网络简介_tensorflow 精进之路-程序员宅基地

文章浏览阅读265次。1、概述本来想用卷积神经网络来预测点东西,但是效果嘛......,还是继续学习图像类的应用吧~前面学习的神经网络都是一些基础的结构,这些网络在各自的领域中都有一定效果,但是解决复杂问题肯定不够的,这就需要用到深度神经网络。深度神经网络是将前面所学的网络组合起来,利用各自网络的优势,使整体效果达到最优。这一节就简单的记下一些常用的深度神经网络模型,因为tensorflow等框架都将这些网络实现..._tensorflow 精进之路

第九十四篇 Spark+HDFS centos7环境搭建_spark写入hdfs需要用户名密码吗-程序员宅基地

文章浏览阅读2.6k次。一、安装包下载:Spark 官网下载: https://spark.apache.org/downloads.htmlHadoop 官网下载: https://hadoop.apache.org/releases.html目前使用Spark 版本为: spark-2.4.3 Hadoop版本为: hadoop-2.10.1二、配置自登陆检测是否可以自登陆,不需要密码则配置正常:ssh localhost在搭建Hadoop环境时,出现localhost.localdomain: Permis_spark写入hdfs需要用户名密码吗

Node.js_node可以使用什么命令 ,它会自动找到该文件下的start指令,执行入口文件。-程序员宅基地

文章浏览阅读280次。nodejs。_node可以使用什么命令 ,它会自动找到该文件下的start指令,执行入口文件。

linux图片相似度检测软件下载,移动端图像相似度算法选型-程序员宅基地

文章浏览阅读293次。概述电商场景中,卖家为获取流量,常常出现重复铺货现象,当用户发布上传图像或视频时,在客户端进行图像特征提取和指纹生成,再将其上传至云端指纹库对比后,找出相似图片,杜绝重复铺货造成的计算及存储资源浪费。该方法基于图像相似度计算,可广泛应用于安全、版权保护、电商等领域。摘要端上的图像相似度计算与传统图像相似度计算相比,对计算复杂度及检索效率有更高的要求。本文通过设计实验,对比三类图像相似度计算方法:感..._linux 图片相似度对比

java isprime函数_判断质数(isPrime)的方法——Java代码实现-程序员宅基地

文章浏览阅读3.8k次。判断质数(isPrime)的方法——Java代码实现/** 质数又称素数。一个大于1的自然数,除了1和它自身外,不能被其他自然数整除的数叫做质数;否则称为合数* 100以内质数表2 3 5 7 11 13 17 19 23 29 31 37 41 43 4753 59 61 67 71 73 79 83 89 97质数具有许多独特的性质:(1)质数p的约数只有两个:1和p。(2)初等数学基本定理:..._java isprime

推荐文章

热门文章

相关标签