[WCF编程]10.操作:事件
一、事件概述
基础的WCF回调机制并不能阐明客户端与服务之间交互的本质。双向回调的规范使用可以通过事件来完成。客户端发生的相关事项都可以通过事件通知客户端或者多个客户端。事件可能源于直接的客户端调用,也可能来源于服务监听器。激活事件的服务称为发布者,而接受事件答得客户端则称为订阅者。如下图所示:
![]()
与回调操作相比,WCF更重视对事件的运作。从本质讲,事件代表了发布者与订阅者之间更加松散的关系,他优于客户端和服务之间的关系。处理事件时,服务通常会为多个订阅客户端发布同样的事件。发布者一般不会考虑订阅者的回调数顺序,也不会考虑订阅者在处理事件时可能出现的错误。所有的发布者都直到它应该将事件传递给订阅者。如果事件出现问题,服务对此也束手无策。此外,服务并不关心订阅者返回的结果。因此,事件处理操作的返回值应该为void,而不需要返回值,因而应该被标记为单向操作。建议将事件分解为单独的回调契约,而不要在相同的契约中将事件和常规的回调混在一起。
public interface IMyEvent
{
[OperationContract(IsOneWay=true)]
void OnEvent1();[OperationContract(IsOneWay</span>=<span style="color: #0000ff">true</span><span style="color: #000000">)]
</span><span style="color: #0000ff">void</span> OnEvent2(<span style="color: #0000ff">int</span><span style="color: #000000"> number); [OperationContract(IsOneWay</span>=<span style="color: #0000ff">true</span><span style="color: #000000">)]
</span><span style="color: #0000ff">void</span> OnEvent3(<span style="color: #0000ff">int</span> number,<span style="color: #0000ff">string</span><span style="color: #000000"> text);
}</span></pre></div>
在订阅端,即使使用了单向回调操作,事件处理方法的实现也应该是执行周期较短的操作。其原因有二:第一,如果需要发布大量的事件,以至于超过了订阅者的能力,且由于队列正在处理前一个事件,无法将回调放入队列中,就会导致发布者被阻塞。阻塞了发布者就会阻止事件到达其它的订阅者;第二,如果存在大量的事件订阅者,每个订阅者所累加起来的处理事件就会操作发布者的超时值。
发布者可以为它的契约添加专门的操作,允许客户端显式地订阅事件或取消对事件的订阅。如果发布者支持多个事件类型,也可以允许订阅者选择它们希望订阅或取消订阅的事件。
服务如何从内部管理订阅者列表及选择它们的参数,完全属于服务端的实现细节,不会影响到客户端。发布者可以使用.NET委托管理订阅者列表和发布者自身的行为,也可以使用泛型进行管理。
[Flags]
public enum EvenType
{
Event1 = 1,
Event2 = 2,
Event3 = 4,
AllEvents = Event1 | Event2 | Event3
}[ServiceContract(CallbackContract </span>= <span style="color: #0000ff">typeof</span><span style="color: #000000">(IMyEventsCallback))]
</span><span style="color: #0000ff">public</span> <span style="color: #0000ff">interface</span><span style="color: #000000"> IMyContract
{
[OperationContract]
</span><span style="color: #0000ff">void</span><span style="color: #000000"> DoSomething(); [OperationContract]
</span><span style="color: #0000ff">void</span><span style="color: #000000"> Subscribe(EvenType mask); [OperationContract]
</span><span style="color: #0000ff">void</span><span style="color: #000000"> UnSubscribe(EvenType mask);
} [ServiceBehavior(InstanceContextMode</span>=<span style="color: #000000">InstanceContextMode.PerCall)]
</span><span style="color: #0000ff">public</span> <span style="color: #0000ff">class</span><span style="color: #000000"> MyPublisher : IMyContract
{
</span><span style="color: #0000ff">private</span> <span style="color: #0000ff">static</span> Action m_Event1 = <span style="color: #0000ff">delegate</span><span style="color: #000000"> { };
</span><span style="color: #0000ff">private</span> <span style="color: #0000ff">static</span> Action<<span style="color: #0000ff">int</span>> m_Event2 = <span style="color: #0000ff">delegate</span><span style="color: #000000"> { };
</span><span style="color: #0000ff">private</span> <span style="color: #0000ff">static</span> Action<<span style="color: #0000ff">int</span>, <span style="color: #0000ff">string</span>> m_Event3 = <span style="color: #0000ff">delegate</span><span style="color: #000000"> { }; </span><span style="color: #0000ff">public</span> <span style="color: #0000ff">void</span><span style="color: #000000"> Subscribe(EvenType mask)
{
IMyEventsCallback subscriber </span>= OperationContext.Current.GetCallbackChannel<IMyEventsCallback><span style="color: #000000">();
</span><span style="color: #0000ff">if</span> ((mask & EvenType.Event1) ==<span style="color: #000000"> EvenType.Event1)
{
m_Event1 </span>+=<span style="color: #000000"> subscriber.OnEvent1;
}
</span><span style="color: #0000ff">if</span> ((mask & EvenType.Event2) ==<span style="color: #000000"> EvenType.Event2)
{
m_Event2 </span>+=<span style="color: #000000"> subscriber.OnEvent2;
}
</span><span style="color: #0000ff">if</span> ((mask & EvenType.Event3) ==<span style="color: #000000"> EvenType.Event3)
{
m_Event3 </span>+=<span style="color: #000000"> subscriber.OnEvent3;
}
} </span><span style="color: #0000ff">public</span> <span style="color: #0000ff">void</span><span style="color: #000000"> UnSubscribe(EvenType mask)
{
IMyEventsCallback subscriber </span>= OperationContext.Current.GetCallbackChannel<IMyEventsCallback><span style="color: #000000">();
</span><span style="color: #0000ff">if</span> ((mask & EvenType.Event1) ==<span style="color: #000000"> EvenType.Event1)
{
m_Event1 </span>-=<span style="color: #000000"> subscriber.OnEvent1;
}
</span><span style="color: #0000ff">if</span> ((mask & EvenType.Event2) ==<span style="color: #000000"> EvenType.Event2)
{
m_Event2 </span>-=<span style="color: #000000"> subscriber.OnEvent2;
}
</span><span style="color: #0000ff">if</span> ((mask & EvenType.Event3) ==<span style="color: #000000"> EvenType.Event3)
{
m_Event3 </span>-=<span style="color: #000000"> subscriber.OnEvent3;
}
} </span><span style="color: #0000ff">public</span> <span style="color: #0000ff">static</span> <span style="color: #0000ff">void</span><span style="color: #000000"> FireEvent(EvenType eventType)
{
</span><span style="color: #0000ff">switch</span><span style="color: #000000"> (eventType)
{
</span><span style="color: #0000ff">case</span><span style="color: #000000"> EvenType.Event1:
{
m_Event1();
</span><span style="color: #0000ff">return</span><span style="color: #000000">;
}
</span><span style="color: #0000ff">case</span><span style="color: #000000"> EvenType.Event2:
{
m_Event2(</span><span style="color: #800080">42</span><span style="color: #000000">);
</span><span style="color: #0000ff">return</span><span style="color: #000000">;
}
</span><span style="color: #0000ff">case</span><span style="color: #000000"> EvenType.Event3:
{
m_Event3(</span><span style="color: #800080">42</span>, <span style="color: #800000">"</span><span style="color: #800000">Hello</span><span style="color: #800000">"</span><span style="color: #000000">);
</span><span style="color: #0000ff">return</span><span style="color: #000000">;
}
</span><span style="color: #0000ff">default</span><span style="color: #000000">:
{
</span><span style="color: #0000ff">throw</span> <span style="color: #0000ff">new</span> InvalidOperationException(<span style="color: #800000">"</span><span style="color: #800000">Unkown event type</span><span style="color: #800000">"</span><span style="color: #000000">);
}
}
} </span><span style="color: #0000ff">public</span> <span style="color: #0000ff">void</span><span style="color: #000000"> DoSomething()
{
</span><span style="color: #0000ff">throw</span> <span style="color: #0000ff">new</span><span style="color: #000000"> NotImplementedException();
}
}</span></pre></div>
服务契约IMyContract定义了Subscribe()和UnSubscribe()方法。这些方法接受一个枚举类型EventType,枚举类型的字段均被设置为2的指数。这就能使订阅客户端通过枚举值掩码标识事件类型是订阅还是取消订阅。例如,订阅Event1和Event3,但不订阅Event2订阅者会调用如下的Subsctibe()方法:
class Program
{
static void Main(string[] args)
{
IMyEventCallback subscriber = new MySubscriber();
InstanceContext context = new InstanceContext(subscriber);
MyContractClient proxy = new MyContractClient(context);
proxy.Subscribe(Service.EvenType.Event1 | Service.EvenType.Event3);
proxy.DoSomething();
proxy.Close();
Console.Read();}
} </span><span style="color: #0000ff">public</span> <span style="color: #0000ff">class</span><span style="color: #000000"> MySubscriber : IMyEventCallback
{
</span><span style="color: #0000ff">public</span> <span style="color: #0000ff">void</span><span style="color: #000000"> OnEvent1()
{
Console.WriteLine(</span><span style="color: #800000">"</span><span style="color: #800000">OnEvent1 Action</span><span style="color: #800000">"</span><span style="color: #000000">);
} </span><span style="color: #0000ff">public</span> <span style="color: #0000ff">void</span> OnEvent2(<span style="color: #0000ff">int</span><span style="color: #000000"> number)
{
Console.WriteLine(</span><span style="color: #800000">"</span><span style="color: #800000">OnEvent2 Action,number={0}</span><span style="color: #800000">"</span><span style="color: #000000">,number);
} </span><span style="color: #0000ff">public</span> <span style="color: #0000ff">void</span> OnEvent3(<span style="color: #0000ff">int</span> number, <span style="color: #0000ff">string</span><span style="color: #000000"> text)
{
Console.WriteLine(</span><span style="color: #800000">"</span><span style="color: #800000">OnEvent3 Action,number={0},text={1}</span><span style="color: #800000">"</span><span style="color: #000000">, number,text);
}
}</span></pre></div>
MyPublisher内部维持了三个静态委托,每一个委托对应一个事件类型。
Subscribe()与UnSubscribe()方法都检查;额传入参数EventType值,从对应的委托中添加或移除订阅者的回调。为了触发事件,MyPublisher提供了静态方法FireEvent()。他根据EventType值判断应该触发哪一个事件,然后调用对应的委托。
示例代码:下载
[WCF编程]10.操作:事件的更多相关文章
- [WCF编程]10.操作:流操作
一.流操作概述 在默认情况下,当客户端调用服务时,服务只有在接收到完整的消息后才会被调用,同样,客户端只有在包含了调用结果的返回消息被完整接受时,才会解除对它的阻塞. 对于数据量小的消息,这种交换模式 ...
- [WCF编程]10.操作:回调操作
一.回调操作概述 WCF支持服务将调用返回给它的客户端.在回调期间,许多方面都将颠倒过来:服务将成为客户端,客户端将编程服务.回调操作可以用在各种场景和应用程序中,但在涉及事件或者服务发生时间需要通知 ...
- [WCF编程]10.操作:请求/应答操作
一.调用操作概述 WCF除了支持经典的服务端-客户端的请求/应答操作外,还提供了对其他操作类型的内建支持,包括:即发即弃的单向调用:允许服务将调用返回给客户端的双向回调:允许客户端或服务器处理大量负荷 ...
- [WCF编程]10.操作:单向操作
一.单向操作概述 WCF提供了单向操作,一旦客户端调用,WCF会生成一个请求,但没有相关的应答信息返回给客户端.所以,单向操作是不能有返回值,服务抛出的任何异常都不会传递给客户端. 理想情况下,一旦客 ...
- jquery css事件编程 位置 操作
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- Windows 10开发基础——指针事件和操作事件(一)
主要内容: 1.指针事件 2.操作事件 1.指针事件 指针事件由各种活动输入源引发,包括触摸.触摸板.笔和鼠标(它们替代传统的鼠标事件).指针事件基于单一输入点(手指.笔尖.鼠标光标),但不支持基于速 ...
- .Net-WCF-图书:《WCF编程》
ylbtech-.Net-WCF-图书:<WCF编程> <WCF编程>是2008年1月机械工业出版社出版的图书,作者是Juval Lowy.Clemens Vasters. 1 ...
- WCF初探-10:WCF客户端调用服务
创建WCF 服务客户端应用程序需要执行下列步骤: 获取服务终结点的服务协定.绑定以及地址信息 使用该信息创建 WCF 客户端 调用操作 关闭该 WCF 客户端对象 WCF客户端调用服务存在以下特点: ...
- WCF编程系列(五)元数据
WCF编程系列(五)元数据 示例一中我们使用了scvutil命令自动生成了服务的客户端代理类: svcutil http://localhost:8000/?wsdl /o:FirstServic ...
随机推荐
- 作业六:团队项目——编写项目的Spec
主要内容: 各组结合所选项目,编写项目的规格说明书(Spec),Spec应至少包含以下内容:(20分) 1. Spec的目标 2. 项目的典型用户和场景 3. 项目的用例模型 4. 项目中涉及到的术语 ...
- 求两点之间最短路径-Dijkstra算法
Dijkstra算法 1.定义概览 Dijkstra(迪杰斯特拉)算法是典型的单源最短路径算法,用于计算一个节点到其他所有节点的最短路径.主要特点是以起始点为中心向外层层扩展,直到扩展到终点为止.D ...
- Webstorm 10 for mac osx 注册机,序列号,kegen
小菜最近get到mac体验机会,早就耳闻mac非常适合做开发,于是迫不及待的安装各种开发工具,不知不觉,轮到前端开发神器webstorm了,看了一下官网的价格,心拔凉拔凉的. 果断搜索注册机,搜到的结 ...
- Dash文档制作教程
前言 什么是Dash 面向程序员的文档库(Mac) 代码片段管理工具 这是强烈推荐给每天在各种API文档中摸爬滚打的程序员们的神器. 为什么要自己制作文档 官方的源中没有相关文档 文档在离线下体验更好 ...
- Ubuntu系统字体安装
用惯了Windows,刚转到Ubuntu时总感觉字体显示没那么亲切,尤其是中文字体,在网页上显示特别怪.有些软件对中文字体的支持也不好,WebStorm中的Git logs中文也显示乱码.把系统语言设 ...
- Hystrix框架4--circuit
circuit 在Hystrix调用服务时,难免会遇到异常,如对方服务不可用,在这种情况下如果仍然不停地调用就是不必要的,在Hystrix中可以配置使用circuit,当达到一定程度错误,就会自动调用 ...
- Html5 实现灯笼绘制
最近在学习Html5,就用JavaScript在Canvas试着绘制了一个灯笼,并作了简要的说明. 具体绘制思路在页面上有说明,不再赘述,代码如下: <script type="tex ...
- WPF入门教程系列二十——ListView示例(二)
第四步.WPF后台逻辑代码编写 在后台用Entity Framework 6.1的Code First方式获取数据库中的数据.同时,在“刷新”按钮的方法中进行数据绑定.操作步骤如下: 1) 在“刷新 ...
- Sql Server系列:流程控制语句
T-SQL中用来编写流程控制模块的语句有:BEGIN...AND语句.IF...ELSE语句.CASE语句.WHILE语句.GOTO语句.BREAK语句.WAITFOR语句和RETURN语句. 1 B ...
- [汇编与C语言关系]3. 变量的存储布局
以下面C程序为例: #include <stdio.h> ; ; ; int c; int main(void) { ; char b[] = "Hello World" ...