[WCF]设置拦截器捕捉到request和reply消息
WCF进阶学习ing...
在熟练掌握了ABC的使用以后,就开始想着去了解WCF是怎么通信的了。首先是服务描述语言wsdl,它定义了服务的描述等等,用于让外界知道这个服务的ABC是什么。另外一个比较重要的就是消息。
WCF是通过消息进行通讯的,一般是使用SOAP形式。服务端的信道监听器接收到消息之后,对消息进行反序列化,解码,然后通过激活对象,再去invoke相应的操作,操作的结果(返回值)再通过编码,序列化,传送给调用者,调用者再对消息进行反序列化,解码,最后拿到结果。所以在这个过程中,对消息的理解和熟悉对于我们理解WCF的操作流程是很大的帮助的。
然后我们就开始拦截这个消息来看看这个消息到底是什么。。。废话不多说,上code。。
首先创建一个提供复数计算的服务,使用共享C的方式,项目结构如下,服务端和客户端都是一个控制台程序

服务契约代码 IComplexCalculate.cs:
namespace Cookiezhi.WcfStudy.Contracts.ServiceContracts
{
[ServiceContract(Namespace="http://www.cookiezhi.com/service/complex")]
public interface IComplexCalculate
{
/// <summary>
/// 加
/// </summary>
[OperationContract]
Complex Add(Complex a, Complex b); /// <summary>
/// 减
/// </summary>
[OperationContract]
Complex Subtract(Complex a, Complex b); /// <summary>
/// 乘
/// </summary>
[OperationContract]
Complex Multiply(Complex a, Complex b); /// <summary>
/// 取模
/// </summary>
[OperationContract]
double Modulus(Complex a);
}
}
数据契约 Complex.cs:
namespace Cookiezhi.WcfStudy.Contracts.DataContracts
{
[DataContract(Namespace = "http://www.cookiezhi.com/data/complex")]
public class Complex
{
/// <summary>
/// 实数
/// </summary>
[DataMember]
public double A { get; set; } /// <summary>
/// 虚数
/// </summary>
[DataMember]
public double B { get; set; }
}
}
服务契约实现 ComplexCalculateService:
namespace Cookiezhi.WcfStudy.Services
{
public class ComplexCalculateService : IComplexCalculate
{
public Complex Add(Complex a, Complex b)
{
return new Complex()
{
A = a.A + b.A,
B = a.B + b.B
};
} public Complex Subtract(Complex a, Complex b)
{
return new Complex()
{
A = a.A - b.A,
B = a.B - b.B
};
} public Complex Multiply(Complex a, Complex b)
{
return new Complex()
{
A = a.A * b.A - a.B * b.B,
B = a.A * b.B + a.B * b.A
};
} public double Modulus(Complex a)
{
return Math.Sqrt(a.A * a.A + a.B * a.B);
}
}
}
采用配置文件方式去设置服务 app.config:
<system.serviceModel>
<behaviors>
<serviceBehaviors>
<behavior name="mexBehavior">
<serviceMetadata httpGetEnabled="true"/>
</behavior>
</serviceBehaviors>
</behaviors> <services>
<service name="Cookiezhi.WcfStudy.Services.ComplexCalculateService" behaviorConfiguration="mexBehavior">
<endpoint address="" binding="basicHttpBinding" contract="Cookiezhi.WcfStudy.Contracts.ServiceContracts.IComplexCalculate" />
<endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange" />
<host>
<baseAddresses>
<add baseAddress="http://127.0.0.1:9999/complexcalcservice"/>
</baseAddresses>
</host>
</service>
</services> </system.serviceModel>
然后服务端的main方法里启动服务:
namespace Cookiezhi.WcfStudy.Hosting
{
class Program
{
static void Main(string[] args)
{
using(ServiceHost host = new ServiceHost(typeof(ComplexCalculateService)))
{
host.Opened += delegate
{
Console.WriteLine("Service {0} started", host.Description.Name);
}; host.Open(); Console.ReadKey();
}
}
}
}
OK, 服务端好了,我们启动一下

再看一下WSDL

OK是好的,这些对于做过WCF相关的朋友们都是轻车熟路了,下面是客户端,通过配置文件加ChannelFactory方式来创建并调用
App.config
<system.serviceModel>
<client>
<endpoint name="ComplexCalculateService" address="http://127.0.0.1:9999/complexcalcservice" binding="basicHttpBinding" contract="Cookiezhi.WcfStudy.Contracts.ServiceContracts.IComplexCalculate" />
</client>
</system.serviceModel>
Main方法
namespace Cookiezhi.WcfStudy.Client
{
class Program
{
static void Main(string[] args)
{
using(ChannelFactory<IComplexCalculate> factory = new ChannelFactory<IComplexCalculate>("ComplexCalculateService"))
{
IComplexCalculate proxy = factory.CreateChannel(); Complex a = new Complex() { A = , B = };
Complex b = new Complex() { A = , B = };
Complex result = null; result = proxy.Add(a, b);
Console.WriteLine("Add result is {0} + {1}i", result.A, result.B); Console.ReadKey();
}
}
}
}
调用服务:

前戏做完了,我们开始进入主题:
我们需要拦截消息,并把消息打印出来,那么我们就需要一个拦截器,叫做MessageInspector,WCF为我们提供了两种拦截器:
客户端拦截器 IClientMessageInspector
提供两个接口
BeforeSendRequest:向服务器发送请求前执行
AfterReceiveReply:接收到服务器的回复消息后执行
服务端拦截器 IDispatchMessageInspector
他也提供两个接口
AfterReceiveRequest:invoke操作之前执行
BeforeSendReply:发送reply给客户端之前执行
在这里我们在服务端设置个拦截器,然后打印出请求和回复的消息,所以我们使用IDispatchMessageInspector这个接口
实现接口 MessageInspector.cs
namespace Cookiezhi.WcfStudy.Hosting.MessageInspect
{
public class MessageInspector : IDispatchMessageInspector
{
public object AfterReceiveRequest(ref Message request, IClientChannel channel, InstanceContext instanceContext)
{
Console.WriteLine(request.ToString());
return DateTime.Now;
} public void BeforeSendReply(ref Message reply, object correlationState)
{
Console.WriteLine(reply.ToString());
DateTime requestTime = (DateTime)correlationState; var duration = DateTime.Now - requestTime;
Console.WriteLine(duration);
}
}
}
其中AfterReceiveRequest先执行,然后去执行远程方法,然后再执行BeforeSendReply,所以在这里加了一个操作计时的功能(可选)。
然后我们要将这个拦截器给寄宿在我们的终结点上,所以需要定义一个终结点行为(EndpointBehavior),并寄宿在服务上。
MessageInspectorBehavior.cs,在ApplyDispatchBehavior方法实现中将我们新建的Inspector实例加到dispatcher的MessageInspectors中
namespace Cookiezhi.WcfStudy.Hosting.MessageInspect
{
public class MessageInspectorBehavior : IEndpointBehavior
{
public void AddBindingParameters(ServiceEndpoint endpoint, BindingParameterCollection bindingParameters)
{
} public void ApplyClientBehavior(ServiceEndpoint endpoint, ClientRuntime clientRuntime)
{
} public void ApplyDispatchBehavior(ServiceEndpoint endpoint, EndpointDispatcher endpointDispatcher)
{
endpointDispatcher.DispatchRuntime.MessageInspectors.Add(new MessageInspector()); } public void Validate(ServiceEndpoint endpoint)
{
}
}
}
最后创建一个配置元素用于在配置文件中给终结点配置这个行为.
namespace Cookiezhi.WcfStudy.Hosting.MessageInspect
{
public class MessageInspectorExtensionElement : BehaviorExtensionElement
{
public override Type BehaviorType
{
get { return typeof(MessageInspectorBehavior); }
} protected override object CreateBehavior()
{
return new MessageInspectorBehavior();
}
}
}
下面就是配置这个行为了
App.config
<?xml version="1.0" encoding="utf-8" ?>
<configuration> <system.serviceModel>
<extensions>
<behaviorExtensions>
<add name="messageInspector" type="Cookiezhi.WcfStudy.Hosting.MessageInspect.MessageInspectorExtensionElement, Cookiezhi.WcfStudy.Hosting"/>
</behaviorExtensions>
</extensions> <behaviors>
<serviceBehaviors>
<behavior name="mexBehavior">
<serviceMetadata httpGetEnabled="true"/>
</behavior>
</serviceBehaviors>
<endpointBehaviors>
<behavior name="messageInspector">
<messageInspector />
</behavior>
</endpointBehaviors>
</behaviors> <services>
<service name="Cookiezhi.WcfStudy.Services.ComplexCalculateService" behaviorConfiguration="mexBehavior">
<endpoint address="" binding="basicHttpBinding" contract="Cookiezhi.WcfStudy.Contracts.ServiceContracts.IComplexCalculate" behaviorConfiguration="messageInspector" />
<endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange" />
<host>
<baseAddresses>
<add baseAddress="http://127.0.0.1:9999/complexcalcservice"/>
</baseAddresses>
</host>
</service>
</services> </system.serviceModel> <startup>
<supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.5" />
</startup>
</configuration>
客户端的代码不要做出任何的改变,
然后我们尝试一下

Great! 我们成功的拦截了请求,并将请求信息打印了出来。
总结,有了这个拦截器,我们可以做很多的事情,比如修改消息头和消息体,计算消息的大小(流量统计),统计服务调用的次数和平均时间,客户端情况,等等。
[WCF]设置拦截器捕捉到request和reply消息的更多相关文章
- 【转】WCF设置拦截器捕捉到request和reply消息
原文:https://www.cnblogs.com/yanglang/p/7063743.html 我们需要拦截消息,并把消息打印出来,那么我们就需要一个拦截器,叫做MessageInspector ...
- Spring拦截器中通过request获取到该请求对应Controller中的method对象
背景:项目使用Spring 3.1.0.RELEASE,从dao到Controller层全部是基于注解配置.我的需求是想在自定义的Spring拦截器中通过request获取到该请求对应于Control ...
- Spring Boot2.0 设置拦截器
所有功能完成 配置登录认证 配置拦截器 在spring boot2.0 之后 通过继承这个WebMvcConfigurer类 就可以完成拦截 新建包com.example.interceptor; 创 ...
- 【spring boot】在自定义拦截器中从request中获取json字符串
又这样的需求,需要在自定义的拦截器中获取request中的数据,想获取到的是JSON字符串 那需要在拦截器中写这样一个方法 public static String getOpenApiRequest ...
- spring boot拦截器中获取request post请求中的参数
最近有一个需要从拦截器中获取post请求的参数的需求,这里记录一下处理过程中出现的问题. 首先想到的就是request.getParameter(String )方法,但是这个方法只能在get请求中取 ...
- 拦截器中,request中getReader()和getInputStream()只能调用一次,构建可重复读取inputStream的request.
由于 request中getReader()和getInputStream()只能调用一次 在项目中,可能会出现需要针对接口参数进行校验等问题. 因此,针对这问题,给出一下解决方案 实现方法:先将Re ...
- axios 设置拦截器 全局设置带默认参数(发送 token 等)
应用场景: 1,每个请求都带上的参数,比如token,时间戳等. 2,对返回的状态进行判断,比如token是否过期 代码如下: [javascript] view plain copy axios.i ...
- 在SSM框架中我设置拦截器filter不能通过注解获取到实现类
我在用注解注入实现类的时候,出现了这样的错误:如以下截图: 这个地方报出的错误是说明我的一个接口类型的类没有获取到,后来我就想要是我的实现类没有获取到那么我就直接new一个实现类然后再进行调用就会出现 ...
- spring boot拦截器中获取request post请求中的参数(转)
文章转自 https://www.jianshu.com/p/69c6fba08c92
随机推荐
- php注意事项
1. 不要使用mysql_函数 这一天终于来了,从此你不仅仅"不应该"使用mysql_函数.PHP 7 已经把它们从核心中全部移除了,也就是说你需要迁移到好得多的mysqli_函数 ...
- 详解应对平台高并发的分布式调度框架TBSchedule
转载: 详解应对平台高并发的分布式调度框架TBSchedule
- BSD Apache GPL LGPL MIT
当Adobe.Microsoft.Sun等一系列巨头开始表现出对”开源”的青睐时,”开源”的时代即将到来! 最初来自:sinoprise.com/read.php?tid-662-page-e-fpa ...
- NGINX怎样处理惊群的
写在前面 写NGINX系列的随笔,一来总结学到的东西,二来记录下疑惑的地方,在接下来的学习过程中去解决疑惑. 也希望同样对NGINX感兴趣的朋友能够解答我的疑惑,或者共同探讨研究. 整个NGINX系列 ...
- 更换内核后重编virtualbox内核模块
这些天编译了一个4.1.15内核,因此vb原来的模块就不能用了,因此要重新编译(当然,reinstall也可以,觉得大动干戈,不符合个人做事风格) 如果不重编运行会有如下错误提示: # virtual ...
- Excel Access 新建空白文档/打开已有文档 提示内存或磁盘空间不足的解决方法--验证
服务器上发现,打开mdb数据库,点知道只有个空白的截面,打开已有的excel文件,一样,但多了个提示:内存磁盘空间不足或者关闭不再使用的工作表或者程序.检查过,内存和磁盘很充裕啊.那里不足啊,任务管理 ...
- warning C4305: “=”: 从“int”到“unsigned char”截断解决方法[zz]
在控制台程序中定义: float x; x=22.333; 编译会出现 warning C4305: “初始化”: 从“double”到“float”截断 系统默认此浮点数是22.333是double ...
- This tag and its children can be replaced by one <TextView/> and a compound drawable
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android=&quo ...
- python分割字符串split,filter函数用法
现有字符串,需要取出用空格分隔的第一段,操作如下 >>> product_model = ‘WS-C2960G-24TC-L – Fixed Module 0′>>> ...
- 数据库对象映射为java对象,不使用框架
方法: public static <T> List<T> processResultSetToList(ResultSet rs, Class<T> clazz) ...