通用转换接口设计

通用转换接口设计


目录

  • 转换接口的定义
  • 转换接口的作用
  • 转换接口的使用
  • 转换接口的实现

1. 转换接口的定义

接口是系统内部与第三方系统协议数据的相互转换处理过程,转换接口以XML配置的形式把这种处理过程描述出来,方便接口的开发与维护;


2. 转换接口的作用

  • 增加开发效率,转换接口抽象出通用的转换处理,开发只需组装对应转换操作即可,减少了重复开发的工作,增加开发效率;
  • 降低出错率,统一协议转换之外的处理,接口开发只需配置相应的转换处理,减少代码量,降低出错概率;
  • 减少维护成本,接口配置化后,对接口协议的调整,只需要针对接口配置修改即可;

3. 转换接口的使用

接口配置以XML文件描述,XSD文件定义节点结构,节点分为转换接口和扩展节点,转换节点用于做数据转换处理,可拥有多个子节点,一般先执行父节点,后执行子节点,执行方式有顺序、选择、循环,子节点如果出现异常,父节点可捕获做处理;扩展节点用于对接口配置或转换节点扩展使用;

3.1 接口配置说明

转换定义文件:有XSD文件表示,定义具体转换节点,需描述节点名称、参数名称,参数类型和转换处理类;

<!--MD5加密-->
    <xsd:element name="md5">
        <xsd:complexType>
            <xsd:complexContent>
                <xsd:extension base="nestedConverterType">
                    <!--MD5处理类-->
                    <xsd:attribute name="clazz" type="xsd:string" 
                    fixed="com.nnk.ecsys.interfaceConverter.converter.Md5Converter"/>
                    <!--加密字符串-->
                    <xsd:attribute name="value" type="paramType" use="required"/>
                    <!--编码类型-->
                    <xsd:attribute name="charsetType" type="paramType" use="optional"/>
                    <!--大小写类型-->
                    <xsd:attribute name="caseType" type="caseType" use="optional" 
                    default="LowerCase"/>
                </xsd:extension>
            </xsd:complexContent>
        </xsd:complexType>
    </xsd:element>

转换节点:由XML节点表示,做具体的转换处理,可有多个输入参数和一个输出参数,输入参数由具体转换节点定义,属性name表示输出参数,转换结果以name值作为名称存入【上下文变量】;

输入参数:值可直接用字符串表示,如需引用【上下文变量】则使用${变量名}表示,还可以相互组合, 如:value="${参数2} AAA ${参数2}",如需要引用对象的参数,则使用${变量名.参数名}表示,基本用法和EL表达式类似;

扩展节点:可自定义扩展节点,针对接口配置或转换节点做处理,现有扩展节点,import可导入其他接口配置,loadProperty可加载properties文件变量;

示例:
interface-trade.xml

<?xml version="1.0" encoding="UTF-8"?>
<converter xmlns="http://www.007ka.com/schema/converter"
           xmlns:extend="http://www.007ka.com/schema/converter/extend"
           xmlns:interface="http://www.007ka.com/schema/converter/interface">
    <!--导入配置文件-->
    <extend:loadProperty location="properties/common.properties"/>
    <!--定义请求对象-->
    <getRequest name="request"/>
    <!--组装请求报文-->
    <interface:tryException errorCode="CEC_FAILED" errorMsg="请求报文组装异常">
        <!--转换参数-->
        <set name="orderReq" value="${request.reqeustParam}" />
        <interface:mapget name="cardType" key="${orderReq.providerId}" mappingName="providerMap"/>
        <!--组装协议-->
        <set name="orderInfo" 
            value="${nnk_merId}|${orderReq.orderId}|${cardType}|${orderReq.unitCount}|"/>
        <interface:md5 name="sign" value="${orderInfo}|${inter_channelKey}" 
            charsetType="${inter_charset}"/>
        <interface:newInstance name="httpParam" value="java.util.HashMap">
            <set name="httpParam.Orderinfo" value="${orderInfo}"/>
            <set name="httpParam.Sign" value="${sign}"/>
        </interface:newInstance>
    </interface:tryException>
    <!--调用http接口-->
    <interface:tryException errorCode="CEC_UNKNOWN" errorMsg="网络异常">
        <interface:http name="httpResponse" url="${inter_interUrl}" 
            paramMap="${httpParam}" charsetType="${inter_charset}"/>
    </interface:tryException>
    <!--解析响应报文-->
    <interface:tryException errorCode="CEC_UNKNOWN" errorMsg="响应报文解析异常">
        <!--解析报文-->
        <interface:xmlToMap name="mapRes" value="${httpResponse}" />
        <set name="mapRes" value="${mapRes.root}"/>
        <!--协议验签-->
        <interface:verifyMd5 charsetType="${inter_charset}" sign="${mapRes.Sign}" 
            value="${mapRes.MerID}|${mapRes.OrderID}|${mapRes.TranStat}|${inter_channelKey}"/>
        <!--组装响应对象-->
        <interface:newInstance name="orderRes" 
            value="com.nnk.ecsys.database.mapper.order.entity.ExternOrderOrderInfo">
            <set name="orderRes.orderId" value="${mapRes.OrderID}"/>
            <set name="orderRes.partnerOrderId" value="${mapRes.TranOrder}"/>
            <set name="orderRes.partnerOrderReceiveErrorCode" value="${mapRes.TranStat}"/>
            <set name="orderRes.partnerOrderReceiveErrorMsg" value="${mapRes.TranInfo}"/>
        </interface:newInstance>
    </interface:tryException>
    <!--返回响应结果-->
    <setResponse value="${orderRes}"/>
</converter>

TestMain.java

//接口配置资源
Resource resource = ResourceUtils.getResource("interface-trade.xml");
//转换构建器
ConverterStackerBuilder converterStackerBuilder = new DefaultConverterStackerBuilder();
//创建转换器
ConverterStacker converterStacker = converterStackerBuilder.buildConverterStacker(resource);
//执行转换
Order.ExternOrderInfo tradeRequest= Order.ExternOrderInfo.getDefaultInstance();
ExternOrderOrderInfo tradeResponse = (ExternOrderOrderInfo) converterStacker.invoke(tradeRequest);

4. 转换接口的实现

4.1 配置文件的解析与扩展

DefinitionReader            // 配置读取类,读取解析接口配置文件
NamespaceHandler            // 命名空间处理类,包含命名空间内每个节点的解析对象
NamespaceHandlerResolver    // 命名空间管理类,管理每个命名空间的XSD文件路径和处理类
DefinitionParser            // 节点解析类,解析具体的转换节点或扩展节点
ConverterDefinition         // 转换节点定义信息类,包含返回值名称,参数列表,转换执行类名,子节点列表等信息
ConverterDefinitionContext  // 转换定义上下文信息类,包含转换节点结构信息,常量参数对象等信息

处理流程:

1.DefinitionReader读取接口配置,递归解析节点信息,

2.根据节点命名空间调用NamespaceHandlerResolver获取指定NamespaceHandler处理

2.NamespaceHandler内部根据节点名称获取对应的DefinitionReader

3.DefinitionReader解析完成后把ConverterDefinition返回DefinitionReader,并存入ConverterDefinitionContext中

4.递归解析完成后最后得到ConverterDefinitionContext对象

扩展说明:

1.添加扩展节点解析类DefinitionParser,解析方法可获取ConverterDefinitionContext对象,可自定义转换节点,或对接口配置或转换节点做扩展处理;

2.添加命名空间处理类NamespaceHandler,描述节点名称与DefinitionParser解析类对应关系

3.添加扩展配置信息,在目录META-INF.converter下添加扩展配置:XSD文件、handlers.properties、schemas.properties,在NamespaceHandlerResolver对象初始化的时候,会读取运行环境下所有Jar下此目录的扩展文件;

            XSD文件:定义扩展节点结构信息,以及指明命名空间名称
handlers.properties:描述命名空间与之对应的NamespaceHandler类
 schemas.properties:描述命名空间与之对应的XSD文件路径

4.2 转换器的构建

ConverterDefinitionContext  // 转换定义上下文信息类,包含转换节点结构信息,常量参数对象等信息
ConverterStacker            // 转换器,执行具体的转换操作
Converter                   // 转换节点执行类
ConverterStackerBuilder     // 转换器构建类

处理流程:

1.通过DefinitionReader读取接口配置,得到ConverterDefinitionContext;

2.ConverterStackerBuilder递归遍历ConverterDefinition,通过ConverterDefinition实例化Converter对象;

3.在Converter实例化过程,将会把Converter参数列表中存在的部分常量引用, 根据ConverterDefinitionContext常量列表设置为具体的值;

4.遍历完成后,每个Converter对象会有一个父节点喝多个子节点,形成树形架构的Converter集合;

5.最后使用Converter集合构建ConverterStacker对象;

4.3 转换器执行过程

ConverterStacker            // 转换器,执行具体的转换操作
Converter                   // 转换节点执行类
ConverterRequest            // 转换请求对象,作为执行过程中变量存储对象

处理流程:

1.接受转换请求,组装ConverterRequest;

2.执行开始,把根转换节点倒序放入执行栈中,然后循环出栈执行Converter;

3.在执行的过程中,根据ConverterRequest变量集合,实例化Converter的参数列表,如过程中存在未实例化的参数,将会抛出异常;

4.参数列表实例化完成后, 把变量列表传给Converter执行转换方法,完成后把结果以name为变量名把结果存入ConverterRequest;

5.转换执行完成后,获取当前Converter的后续节点倒序存入执行栈中,注意:获取后续节点的逻辑将由Converter本身实现,超类默认实现是获取子节点,但如果像选择、循环之类的结构,此方法逻辑将会重写;

6.如在Converter执行过程中出现异常,则会进入异常捕获处理分支,详情看代码;

7.循环执行到执行栈为空时,返回结果对象

4.4 上下文变量管理

ConverterRequest            // 转换器请求对象
ConverterDefinitionContext  // 转换定义上下文信息类,包含转换节点结构信息,常量参数对象等信息
AssemblyParam               // 装配参数对象

处理流程:

详情看代码
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,457评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,837评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,696评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,183评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,057评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,105评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,520评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,211评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,482评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,574评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,353评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,897评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,489评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,683评论 2 335

推荐阅读更多精彩内容