在项目中,我们经常会使用到http+xml的接口,而且不仅仅的是一个,可能会有多个http的接口需要实时的交互.但是http接口的接收消息的公共部分是一样的,只有每个接口的报文解析和返回报文是不同的,此时考虑到把变化和不变化的隔离出来,采取用策略模式,把公共的部分代码抽取隔离出来,每个http接口的不同的处理逻辑单独自己处理,这样也方便了后期的修改和扩展,可以很方便的修改单独的接口处理逻辑和添加新的http接口到项目中使用,而不用修改以前的设计.下面就http+xml接口的接收采用简单的策略模式实现:

项目的基础:SSH架构

首先实现BaseReceiveHttpIntfAction 基类,实时的接收报文动作和返回结果报文动作:具体的解析逻辑和获取返回报文逻辑抽取隔离出去:

/**
* 〈一句话功能简述〉<br>
* 〈功能详细描述〉 http 接收接口基类实现 父类实现为prototype 不是单例的
*
* 基于策略模式 后面增加子类接收实现的方法 请参照:CrmReceiveHttpIntfAction的实现
*
* @author lilin
* @see [相关类/方法](可选)
* @since [产品/模块版本] (可选)
*/
@Controller
@Scope(value = "prototype")
public abstract class BaseReceiveHttpIntfAction extends BasicAction {
/**
*/
private static final long serialVersionUID = -4667071743433536439L; private IHttpMessageHandle httpMessageHandle; /**
*
* 功能描述: <br>
* 〈功能详细描述〉用于 子类 注入父类属性
*
* @param httpMessageHandle
* @see [相关类/方法](可选)
* @since [产品/模块版本](可选)
*/
public void setHttpMessageHandle(IHttpMessageHandle httpMessageHandle) {
this.httpMessageHandle = httpMessageHandle;
} public String doBusiness() {
InputStream inputStream = null;
try {
inputStream = request.getInputStream();
} catch (IOException e1) {
logger.error("IOException: ", e1);
}
logger.info("@@HttpClient 解析接收报文:");
SAXReader sb = new SAXReader();
String returnStr = null;
try {
Document document = sb.read(inputStream);
logger.info(document.asXML());
// 保存报文信息
Map<String, Object> map = httpMessageHandle.handleMessage(document);
// 拼接返回报文
returnStr = httpMessageHandle.getResponseMsg(map, document);
logger.info("@@ 报文解析成功!");
} catch (DocumentException e) {
logger.info("@@HttpClient 解析报文出错!", e);
}
logger.info("@@ 返回报文 : " + returnStr);
writeResponse(returnStr, response);
return null;
} /**
*
* 功能描述: <br>
* 〈功能详细描述〉
*
* @param respXML
* @param response
* @throws IOException
* @see [相关类/方法](可选)
* @since [产品/模块版本](可选)
*/
private void writeResponse(String respXML, HttpServletResponse response) {
OutputStream out = null;
try {
byte[] data = respXML.getBytes("UTF-8");
out = response.getOutputStream();
out.write(data);
} catch (IOException e) {
logger.error("Write response error: ", e);
} finally {
if (out != null) {
try {
out.flush();
out.close();
} catch (IOException e) {
logger.error("IOException: ", e);
}
}
}
} }

然后:规定好子类需要注入的公共接口IHttpMessageHandle, 接口定义需要解析报文的解析方法,以及获取放回报文的方法:

/**
* 〈一句话功能简述〉<br>
* 〈功能详细描述〉
*
* @author lilin
* @see [相关类/方法](可选)
* @since [产品/模块版本] (可选)
*/
public interface IHttpMessageHandle { /**
*
* 功能描述: <br>
* 〈功能详细描述〉处理接收到的报文信息
*
* @param document
* @return
* @see [相关类/方法](可选)
* @since [产品/模块版本](可选)
*/
Map<String, Object> handleMessage(Document document); /**
*
* 功能描述: <br>
* 〈功能详细描述〉获取需要返回的报文信息
*
* @param map
* @param document
* @return
* @see [相关类/方法](可选)
* @since [产品/模块版本](可选)
*/ String getResponseMsg(Map<String, Object> map, Document document); }

然后:实现实际的处理报文信息的类:CrmHttpMessageHandle 实时处理CRM系统发送的xml报文信息,实时组装需要返回的报文信息:

/**
* 〈一句话功能简述〉<br>
* 〈功能详细描述〉
*
* @author lilin
* @see [相关类/方法](可选)
* @since [产品/模块版本] (可选)
*/
@Service
public class CrmHttpMessageHandle implements IHttpMessageHandle { private Logger logger = Logger.getLogger(CrmHttpMessageHandle.class); @Resource
private LowPriceTaskService lowPriceTaskService; @Override
public String getResponseMsg(Map<String, Object> map, Document document) {
logger.info("CrmHttp Get ResponseMsg:"); printLogByEntry(map); Document doc = DocumentHelper.createDocument();
// MbfService 拼接
Element mbfService = DocumentHelper.createElement("MbfService");
doc.add(mbfService);
Element output1 = DocumentHelper.createElement("output1");
mbfService.add(output1); // MbfHeader 拼接
Element mbfHeader = DocumentHelper.createElement("MbfHeader");
output1.add(mbfHeader); PriceCommonService.addElement(mbfHeader, "ServiceCode", (String) map.get("ServiceCode"));
PriceCommonService.addElement(mbfHeader, "Operation", (String) map.get("Operation"));
PriceCommonService.addElement(mbfHeader, "AppCode", (String) map.get("AppCode"));
PriceCommonService.addElement(mbfHeader, "UId", (String) map.get("UId")); // ServiceResponse 拼接
Element serviceResponse = DocumentHelper.createElement("ServiceResponse");
mbfHeader.add(serviceResponse); PriceCommonService.addElement(serviceResponse, "Status", null);
PriceCommonService.addElement(serviceResponse, "Code", null);
PriceCommonService.addElement(serviceResponse, "Desc", null); // MbfBody 拼接
Element mbfBody = DocumentHelper.createElement("MbfBody");
output1.add(mbfBody); PriceCommonService.addElement(mbfBody, "reFlag", (String) map.get("reFlag"));
PriceCommonService.addElement(mbfBody, "errorMessage", (String) map.get("errorMessage")); logger.info("CrmHttp Final ResponseMsg:" + doc.asXML());
return doc.asXML();
} @Override
public Map<String, Object> handleMessage(Document document) {
logger.info("CrmHttp Start HandleMessage:"); Map<String, Object> res = new HashMap<String, Object>();
LowPriceTask task = new LowPriceTask(); Element root = document.getRootElement();
Element outElement = root.element("input1"); Element mbfHeader = outElement.element("MbfHeader"); String serviceCode = mbfHeader.elementTextTrim("ServiceCode");
res.put("ServiceCode", serviceCode);
logger.info("## ServiceCode : " + serviceCode);
String operation = mbfHeader.elementTextTrim("Operation");
res.put("Operation", operation);
logger.info("## operation : " + operation);
String appCode = mbfHeader.elementTextTrim("AppCode");
res.put("AppCode", appCode);
logger.info("## appCode : " + appCode);
String uId = mbfHeader.elementTextTrim("UId");
res.put("UId", uId);
logger.info("## uId : " + uId); // 设置返回值
if (!StringUtils.isEmpty(serviceCode) && !StringUtils.isEmpty(operation)) {
task.setMsg(document.asXML());
task.setScanNum(0);
task.setScanResult("F");
task.setStoreTime(DateUtils.getDate2()); lowPriceTaskService.saveTask(task); res.put("reFlag", "S");
res.put("errorMessage", "S");
logger.info("@@HttpClient 保存报文信息成功!");
} else {
res.put("reFlag", "E");
res.put("errorMessage", "报文头有问题");
} return res;
}

最后:实现实际的接收报文信息的子类action:继承基类,实时注入父类的接口实现,最终提供给客户端的url:http://lilin.com/lilin-web/crmReceiveHttpIntfAction.do

/**
* 〈一句话功能简述〉<br>
* 〈功能详细描述〉
*
* @author lilin
* @see [相关类/方法](可选)
* @since [产品/模块版本] (可选)
*/
@Controller
public class CrmReceiveHttpIntfAction extends BaseReceiveHttpIntfAction { /**
*/
private static final long serialVersionUID = -104341558961432099L; @Resource
private IHttpMessageHandle crmHttpMessageHandle; @PostConstruct
public void injectHttpMessageHandle() {
super.setHttpMessageHandle(crmHttpMessageHandle);
}
}

这样,简单的http+xml报文+策略模式的实现,就完成了.后期需要扩展新的接口接口使用,只要仅仅实现自己的

1,接收子类<参考CrmReceiveHttpIntfAction >

2,消息处理类:<参考CrmHttpMessageHandle>

3,提供最终的服务方的url给客户端调用即可;

http接口服务方结合策略模式实现总结的更多相关文章

  1. 小白写了一堆if-else,大神实在看不下去了,竟然用策略模式直接摆平了

    这里涉及到一个关键词:策略模式,那么到底什么是策略模式呢?本文就来好好给大家讲讲策略模式,大家可以带着如下几个问题来阅读本文:   1. 如何通过策略模式优化业务逻辑代码(可以根据自己从事的工作思考) ...

  2. C++设计模式-Strategy策略模式

    Strategy策略模式作用:定义了算法家族,分别封装起来,让他们之间可以互相替换,此模式让算法的变化,不会影响到使用算法的客户. UML图: Strategy模式将逻辑(算法)封装到一个类(Cont ...

  3. Java设计模式6:策略模式

    策略模式 策略模式的用意是针对一组算法,将每一个算法封装到具有共同接口的独立类中,从而使得它们可以相互替换.策略模式使得算法可以在不影响到客户端的情况下发生变化. 策略模式的结构 策略模式是对算法的包 ...

  4. 策略模式c++【转】

    作用:定义了算法家族,分别封装起来,让他们之间可以互相替换,此模式让算法的变化,不会影响到使用算法的客户. UML图: Strategy模式将逻辑(算法)封装到一个类(Context)里面,通过组合的 ...

  5. 【设计模式】策略模式 (Strategy Pattern)

    策略模式是一种很简单的基础模式,用于封装一系列算法,使客户端的访问独立于算法的实现.我们可以”井中取水”来形象的描述策略模式.“取水”是一个动作,完成这个动作的方式有很多中,可以直接用手提.可以用水车 ...

  6. 设计模式-策略模式Strategy以及消灭if else

    概述 如果在开发过程中,出现大量的if else或者switch case 语句,如果这些语句块中的代码并不是包含业务逻辑,只是单纯的分流方法,那么,每一个语句块中都是一个算法或者叫策略. 背景 比如 ...

  7. 设计模式之策略模式的Python实现

    1. 策略模式解决的是什么问题 策略模式解决的应用场景是这样的: 在业务场景中,需要用到多个算法,并且每个算法的参数是需要调整的.那么当不同的行为堆砌到同一个类中时,我们很难避免使用条件语句来选择合适 ...

  8. HTTP+XML接口客户端 结合策略模式实现总结

    在项目中,我们经常会使用到http+xml的接口,而且不仅仅的是一个,可能会有多个http的接口需要实时的交互.但是http接口的发送消息的公共部分是一样的,只有每个接口的报文解析和返回报文是不同的, ...

  9. C#委托和事件?策略模式?接口回调?还不清楚的赶紧来看我扯

    早前学习委托的时候,写过一点东西,今天带着新的思考和认知,再记点东西.这篇文章扯到设计模式中的策略模式,观察者模式,还有.NET的特性之一--委托.真的,请相信我,我只是在扯淡...... 场景练习 ...

随机推荐

  1. 一个单元格占两行三列的HTML代码为

    主要是这两个属性: colspan 单元格占多少列 rowspan 单元格占多少行 <table width="200" border="1">&l ...

  2. luogu P2634 [国家集训队]聪聪可可 点分治

    Description 聪聪和可可是兄弟俩,他们俩经常为了一些琐事打起来,例如家中只剩下最后一根冰棍而两人都想吃.两个人都想玩儿电脑(可是他们家只有一台电脑)……遇到这种问题,一般情况下石头剪刀布就好 ...

  3. Project Euler 32 Pandigital products

    题意:找出所有形如 39 × 186 = 7254 这种,由 1 - 9,9个数字构成的等式的和,注意相同的积不计算两次 思路:如下面两种方法 方法一:暴力枚举间断点 /*************** ...

  4. BZOJ 4951 [WF2017]Money for Nothing (决策单调优化DP+分治)

    题目大意:略 题目传送门 不愧是$World final$的神题,代码短,思维强度大,细节多到吐..调了足足2h 贪心 我们利用贪心的思想,发现有一些工厂/公司是非常黑心的 以工厂为例,对于一个工厂$ ...

  5. 小松之LINUX驱动学习笔记之模块间函数调用通讯

    1. 符号导出函数 EXPORT_SYMBOL() EXPORT_SYMBOL标签内定义的函数对全部内核代码公开,不用修改内核代码就可以在您的内核模块中直接调用. EXPORT_SYMBOL_GPL( ...

  6. Python 不同列表时间测试

    import timeit import threading def test1(): l = [] for i in range(1000): l = l + [i] def test2(): l ...

  7. PHP面向对象(一)

    1 面向对象介绍 1.1 介绍 ​ 面向对象是一个编思想. 编程思想有面向过程和面向对象. ​ 面向过程: 编程思路集中的是过程上 ​ 面向对象: 编程思路集中在参与的对象 1.2 好处 多人合作方便 ...

  8. HDU5926 Mr. Frog’s Game

    /* HDU5926 Mr. Frog’s Game http://acm.hdu.edu.cn/showproblem.php?pid=5926 杂题水题 * */ #include <cst ...

  9. HDU2147 kiki's game

    /* HDU2147 kiki's game 博弈论 巴什博奕 http://acm.hdu.edu.cn/showproblem.php?pid=2147 题意:在一个n×m的棋盘上,初始棋子放在右 ...

  10. hadoop分布式架构和设计

    引言 Hadoop分布式文件系统(HDFS)被设计成适合执行在通用硬件(commodity hardware)上的分布式文件系统.它和现有的分布式文件系统有非常多共同点.但同一时候,它和其它的分布式文 ...