在项目中,我们经常会使用到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. 杭电2053 WA

    #include<stdio.h> int main() { ]; while(scanf("%d",&n)!=EOF) { ;i<=;i++) { a[ ...

  2. Java中数组获取最大值

    最大值获取:从数组的所有元素中找出最大值. 实现思路: 定义变量,保存数组0索引上的元素 遍历数组,获取出数组中的每个元素 将遍历到的元素和保存数组0索引上值的变量进行比较 如果数组元素的值大于了变量 ...

  3. 洛谷P2296 寻找道路_简单BFS

    Code: #include<cstdio> #include<queue> #include<algorithm> using namespace std; co ...

  4. 洛谷P3113 [USACO14DEC]马拉松赛跑Marathon_Gold 线段树维护区间最大值 模板

    如此之裸- Code: #include<cstdio> #include<cstring> #include<cmath> #include<algorit ...

  5. jmeter出现乱码怎么解决

    本文是抄袭安大叔的性能    如果想在性能获得更好的发展  请添加公众号:测试那点事  大叔的群号:435092293  大叔曾经担任百度技术总监  很牛逼  相信大叔知道了不会怪我  毕竟我是你的学 ...

  6. JavaScript(正则表达式一)

    -------------------- 创建正则表达式: 验证匹配的两个方法 //正则表达式测试 /* var p=new RegExp("Box","i") ...

  7. soapui测试接口使用步骤

    1.新建项目 2. 定义接口 url输入接口 3.新建测试集 选择项目,右键 4.在测试集下新建测试用例 5.在测试步骤中导入要测试的请求 6.run

  8. oracle double和float,number

    float,double,number都是oracle的数值类型.1个汉子=2个英文=2个字节float表示单精度浮点数在机内占4个字节,用32位二进制描述. double表示双精度浮点数在机内占8个 ...

  9. poj 2914&&hdu 3002 全局最小割Stoer-Wagner算法模板

    #include<stdio.h> #include<string.h> #include<iostream> #define inf 0x3fffffff #de ...

  10. BA--暖通系统常见设计细节要点

    (一)系统设计问题 1.水泵在系统的设计位置: 一般而言,冷冻水泵应设在冷水机组前端,从末端回来的冷冻水经过冷冻水泵打回冷水机组:冷却水泵设在冷却水进机组的水路上,从冷却塔出来的冷却水经冷却水泵打回机 ...