前言

  代理层的主要工作是调用Web Service,将在FCL层序列化好的Json数据字符串Post到Web Service,然后获得Reponse,再从响应流中读取到调用结果Json字符串,在Dispatcher反序列化成数据对象,在UI层呈现出来。

HttpHelper类(参考自:http://blog.csdn.net/eriloan/article/details/7000790)

  刚开始是直接在Proxy类中直接写的Post方法,后来看到这篇帖子,将Http相关的部分封装成了工具类HttpHelper。原帖中还包含了使用TCPSocket发送请求调用WebService的内容。

 namespace ProjectmsMGT_Proxy
{
public class HttpHelper
{
/// <summary>
/// Http请求URL
/// </summary>
public string Url { set; get; } /// <summary>
/// 请求参数Key(约定服务方法参数名同名)
/// 在使用Post方式向服务器端发送请求的时候,请求数据中包含了参数部分,参数部分我们需要告诉WebService接口方法,实参要传给的接口方法行参名,由RequestParaKey指定
/// 当然当接口方法有多个形参时,就不建议单独设计这样一个属性,直接在sendMsg中添加,此处只是为了突出RequestParaKey的重要性
/// </summary>
public string RequestParaKey { set; get; } /// <summary>
/// 证书文件路径
/// </summary>
public string CertificateFilePath { set; get; } /// <summary>
/// 证书文件口令
/// </summary>
public string CertificateFilePwd { set; get; } /// <summary>
/// 构造函数,不使用证书
/// </summary>
/// <param name="url"></param>
/// <param name="requestParaKey"></param>
public HttpHelper(string url, string requestParaKey)
{
this.Url = url;
this.RequestParaKey = requestParaKey;
} /// <summary>
/// 构造函数,使用证书
/// </summary>
/// <param name="url"></param>
/// <param name="requestParaKey"></param>
/// <param name="certFilePath"></param>
/// <param name="certFilePwd"></param>
public HttpHelper(string url, string requestParaKey, string certFilePath, string certFilePwd)
{
this.Url = url;
this.RequestParaKey = requestParaKey;
this.CertificateFilePath = certFilePath;
this.CertificateFilePwd = certFilePwd;
} /// <summary>
/// 使用Get方式,发送Http请求
/// </summary>
/// <param name="methodName">所请求的接口方法名</param>
/// <param name="isLoadCert">是否加载证书</param>
/// <returns>响应字符串</returns>
public string CreateHttpGet(string methodName, bool isLoadCert)
{
HttpWebRequest request = CreateHttpRequest(methodName, @"GET", isLoadCert); return CreateHttpResponse(request);
} /// <summary>
/// 使用Post方式,发送Http请求
/// </summary>
/// <param name="methodName">所请求的接口方法名</param>
/// <param name="sendMsg">请求参数(不包含RequestParaKey部分)</param>
/// <param name="isLoadCert">是否加载证书</param>
/// <returns>响应字符串</returns>
public string CreateHttpPost(string methodName, string sendMsg, bool isLoadCert)
{
//创建Http请求
HttpWebRequest request = CreateHttpRequest(methodName, @"POST", isLoadCert);
if (null != sendMsg && !"".Equals(sendMsg))
{
//添加请求参数
AddHttpRequestParams(request, sendMsg);
} //获得响应
return CreateHttpResponse(request);
} /// <summary>
/// 将请求参数写入请求流
/// </summary>
/// <param name="request"></param>
/// <param name="sendMsg"></param>
private void AddHttpRequestParams(HttpWebRequest request, string sendMsg)
{
//将请求参数进行URL编码
string paraUrlCoded = System.Web.HttpUtility.UrlEncode(RequestParaKey) + "=" +
System.Web.HttpUtility.UrlEncode(sendMsg); byte[] data = Encoding.UTF8.GetBytes(paraUrlCoded);
request.ContentLength = data.Length;
Stream requestStream = null;
using (requestStream = request.GetRequestStream())
{
//将请求参数写入流
requestStream.Write(data, , data.Length);
} requestStream.Close();
} /// <summary>
/// 创建HttpRequest
/// </summary>
/// <param name="methodName"></param>
/// <param name="requestType">POST或者GET</param>
/// <param name="isLoadCert"></param>
/// <returns>HttpWebRequest对象</returns>
private HttpWebRequest CreateHttpRequest(string methodName, string requestType, bool isLoadCert)
{
HttpWebRequest request = null;
try
{
string requestUriString = Url + "/" + methodName;
request = (HttpWebRequest)WebRequest.Create(requestUriString);
if (isLoadCert)
{
//创建证书
X509Certificate2 cert = CreateX509Certificate2();
//添加证书认证
request.ClientCertificates.Add(cert);
}
request.KeepAlive = true;
request.ContentType = "application/x-www-form-urlencoded";
request.Method = requestType;
}
catch (Exception)
{
//Console.WriteLine("创建HttpRequest失败。原因:" + e.Message);
request = null;
} return request;
} /// <summary>
/// 创建请求响应
/// </summary>
/// <param name="request"></param>
/// <returns>响应字符串</returns>
private string CreateHttpResponse(HttpWebRequest request)
{
String str;
HttpWebResponse response = null;
Stream responseStream = null;
XmlTextReader responseReader = null;
try
{
using (response = (HttpWebResponse)request.GetResponse())
{
//获得响应流
responseStream = response.GetResponseStream();
responseReader = new XmlTextReader(responseStream);
responseReader.MoveToContent();
str = responseReader.ReadInnerXml();
}
}
catch (Exception e)
{
str = "[{\"Rescode\":\"0\",\"Resmsg\":\"通信失败。原因:" + e.Message + "\"}]";
}
finally
{
if (null != response)
{
responseReader.Close();
responseStream.Close();
response.Close();
}
} return str;
} /// <summary>
/// 创建证书
/// </summary>
/// <returns>X509Certificate2对象</returns>
private X509Certificate2 CreateX509Certificate2()
{
X509Certificate2 cert = null;
try
{
cert = new X509Certificate2(CertificateFilePath, CertificateFilePwd);
ServicePointManager.ServerCertificateValidationCallback =
new RemoteCertificateValidationCallback(ServerCertificateValidationCallback);
}
catch (Exception)
{
//Console.WriteLine("创建X509Certificate2失败。原因:" + e.Message);
cert = null;
}
return cert;
} /// <summary>
/// Verifies the remote Secure Sockets Layer (SSL) certificate used for authentication
/// </summary>
/// <param name="obj">An object that contains state information for this validation</param>
/// <param name="cer">The certificate used to authenticate the remote party</param>
/// <param name="chain">The chain of certificate authorities associated with the remote certificate</param>
/// <param name="error">One or more errors associated with the remote certificate</param>
/// <returns>A Boolean value that determines whether the specified certificate is accepted for authentication</returns>
private bool ServerCertificateValidationCallback(object obj, X509Certificate cer, X509Chain chain, System.Net.Security.SslPolicyErrors error)
{
return true;
}
}
}

  HttpHelper中把SSL证书的部分也包含进来,但是证书认证机制部分ServerCertificateValidationCallback还没设计,各位大神可以自行发挥。

代理类Proxy

  有了HttpHelper之后,代理类的代码就比较明了了。

 namespace ProjectmsMGT_Proxy
{
public class ProjectmsProxy
{
private readonly string Url = "http://59.68.29.106:8087/IFT_Project.asmx";//通过配置文件获取Web Service地址
private readonly string requestParaKey = "paramaters";//服务端所有接口函数统一的参数名
private HttpHelper httpHelper; public ProjectmsProxy()
{
//初始化
Initialize();
} private void Initialize()
{
httpHelper = new HttpHelper(this.Url, this.requestParaKey);
} /// <summary>
/// 使用Get方式调用WebService,不带参数
/// </summary>
/// <param name="methodName"></param>
/// <param name="parasJsonStr"></param>
/// <param name="requestType"></param>
/// <returns></returns>
public string Excute(string methodName, string parasJsonStr, string requestType)
{
return httpHelper.CreateHttpGet(methodName, false);
} /// <summary>
/// 默认使用Post方式调用WebService,带参数
/// </summary>
/// <param name="methodName"></param>
/// <param name="parasJsonStr"></param>
/// <returns></returns>
public string Excute(string methodName, string parasJsonStr)
{
return httpHelper.CreateHttpPost(methodName, parasJsonStr, false);
} /// <summary>
/// 默认使用Post方式调用WebService,不带参数
/// </summary>
/// <param name="methodName"></param>
/// <returns></returns>
public string Excute(string methodName)
{
return httpHelper.CreateHttpPost(methodName, null, false);
}
}
}

  Proxy中重载了Excute方法,三个参数的表示使用Get方式调用WebService(因为不建议在Get方式下传参给Web Service),两个参数和一个参数的Excute默认是使用Post方式带参数和不带参数的情况。

总结

  将方法名作为参数Post到Web Service可以减少很多重复代码,不需要对服务端的每个接口函数做写一个代理函数,这是使用Post方式比使用添加Web服务引用方式更加灵活。

基于Web Service的客户端框架搭建三:代理层(Proxy)的更多相关文章

  1. 基于Web Service的客户端框架搭建四:终结篇

    前言 这是这个系列的终结篇,前面3个博客介绍了一下内容: 1.使用Http Post方式调用Web Service 2.客户端框架之数据转换层 3.客户端框架之代理层 框架结构 框架是基于C#的,在V ...

  2. 基于Web Service的客户端框架搭建二:数据转换层(FCL)

    引言 要使用WebService来分离客户端与服务端,必定要使用约定好两者之间的数据契约.Json数据以其完全独立于语言的优势,成为开发者的首选.C# JavaScriptSerializer为Jso ...

  3. 基于Web Service的客户端框架搭建一:C#使用Http Post方式传递Json数据字符串调用Web Service

    引言 前段时间一直在做一个ERP系统,随着系统功能的完善,客户端(CS模式)变得越来越臃肿.现在想将业务逻辑层以下部分和界面层分离,使用Web Service来做.由于C#中通过直接添加引用的方来调用 ...

  4. 基于JavaScript的REST客户端框架

    现在REST是一个比较热门的概念,REST已经成为一个在Web上越来越常用的应用,基于REST的Web服务越来越多,包括Twitter在内的微博客都是用REST做为对外的API,先前我曾经介绍过“基于 ...

  5. 《基于 Web Service 的学分制教务管理系统的研究与实现》论文笔记(十一)

    标题:基于 Web Service 的学分制教务管理系统的研究与实现 一.基本内容 时间:2014 来源:苏州大学 关键词:: 教务管理系统 学分制 Web Service 二.研究内容 1.教务管理 ...

  6. 基于Docker的TensorFlow机器学习框架搭建和实例源码解读

    概述:基于Docker的TensorFlow机器学习框架搭建和实例源码解读,TensorFlow作为最火热的机器学习框架之一,Docker是的容器,可以很好的结合起来,为机器学习或者科研人员提供便捷的 ...

  7. MyEclipse构建Web Service(Xfire框架)

    以下是本人原创,如若转载和使用请注明转载地址.本博客信息切勿用于商业,可以个人使用,若喜欢我的博客,请关注我,谢谢!博客地址 任务要求: 使用Xfire实现一个简单的CalculatorWebServ ...

  8. JAVA开发Web Service几种框架介绍

    郑重声明:此文为转载来的,出处已不知了,侵告删. 在讲Web Service开发服务时,需要介绍一个目前开发Web Service的几个框架,分别为Axis,axis2,Xfire,CXF以及JWS( ...

  9. SOAP: java+xfire(web service) + php客户端

    作者: 吴俊杰 web service这项技术暂不说它有多落伍,但是项目中用到了,没法逃避!    xml和json各有各的好处,但是JSON无疑是当今数据交互的主流了.客户soap服务器端用的是 j ...

随机推荐

  1. 25个Linux相关的网站

    下面是25个最具有影响力,也是最重要的Linux网站,这些网站提供了Linux的分发包,软件,文件,新闻,以及其它所有的关于Linux的东西.关于Linux的分发包历史,可以看看本站的这篇文章< ...

  2. (转)mysql command line client打不开(闪一下消失)的解决办法

    转自:http://www.2cto.com/database/201209/153858.html 网上搜索到的解决办法: 1.找到mysql安装目录下的bin目录路径. 2.打开cmd,进入到bi ...

  3. 图像像素转换 8-bit 16-bit 32-bit游戏效果

    https://www.ipcfun.com/8bit-you-xi-hua-mian-fu-yuan-3d-li-ti-tu.html https://bbs.csdn.net/wap/topics ...

  4. day23(事务管理)

    事务管理 事务管理两种方式: 向下传递,ThreadLocal 向下传递的方式(依赖) 缺点:不利于测试 Service层 获取连接conn(Connection) 转账(conn) 收账(conn) ...

  5. JVM可支持的最大线程数

    转微博,因为他也是转载  不知道原出处 一.认识问题: 首先我们通过下面这个 测试程序 来认识这个问题:运行的环境 (有必要说明一下,不同环境会有不同的结果):32位 Windows XP,Sun J ...

  6. Kali Linux渗透测试实战 1.3 渗透测试的一般化流程

    1.3 渗透测试的一般化流程 凡事预则立,不预则废,做任何事情都要有一个预先的计划.渗透测试作为测试学科的一个分支,早已形成了完整的方法论.在正式开始本书的实践教学章节之前,我也想谈一谈使用Kali ...

  7. MySQL--当mysqldump --single-transaction遇到alter table(1)

    部分生产环境采用mysqldump --single-transaction的方式在夜间进行数据库备份,而同事恰好在备份期间执行了alter table操作,操作部分成功部分失败,为啥呢? ##=== ...

  8. 聊聊如何设计千万级吞吐量的.Net Core网络通信!

    聊聊如何设计千万级吞吐量的.Net Core网络通信! 作者:大石头 时间:2018-10-26 晚上 20:00 地点:QQ群-1600800 内容:网络通信, 网络库使用方式 网络库设计理念,高性 ...

  9. .net中的SelectList在Html.DropdownList中的使用

    .net中的SelectList可以用于前端下拉框的内容填充 譬如:Html.DropdownList(下拉框标签名称, SelectList实例) 实际上,上述Html.DropdownList的第 ...

  10. EF学习笔记-2 EF之支持复杂类型的实现

    使用过.NET的小伙伴们知道,在我们的实体模型中,除了一些简单模型外,还有一些复杂类型,如几个简单的类型组合而成的类型:而EF除了在实现基本的增删改查之外,也支持复杂类型的实现. 那么如何手动构造复杂 ...