[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 ...
随机推荐
- [译] 理解数组在 PHP 内部的实现(给PHP开发者的PHP源码-第四部分)
文章来自:http://www.hoohack.me/2016/02/15/understanding-phps-internal-array-implementation-ch 原文:https:/ ...
- MySQL 变量和条件
概述 变量在存储过程中会经常被使用,变量的使用方法是一个重要的知识点,特别是在定义条件这块比较重要. mysql版本:5.6 变量定义和赋值 #创建数据库 DROP DATABASE IF EXIST ...
- [译]Introducing ASP.NET vNext and MVC 6
原文:http://www.infoq.com/news/2014/05/ASP.NET-vNext?utm_source=tuicool Part of the ASP.NET vNext init ...
- Git 进阶指南
转自: https://github.com/kaiye/kaiye.github.com/issues/7 在掌握了基础的 Git 使用 之后,可能会遇到一些问题.以下是猫哥筛选总结的部分常见问题, ...
- java.lang.IllegalArgumentException: 'sessionFactory' or 'hibernateTemplate' is required
java.lang.IllegalArgumentException: 'sessionFactory' or 'hibernateTemplate' is required 严重: Exceptio ...
- Redis命令笔记
一.string类型:(1)set :设置key对应的值为string类型的value,例:set name helloworld(2)get :获取key对应的值为string类型的value,例: ...
- HTML5_01之表单新特性
1.WebStorm快捷键: Ctrl+Alt+(向下方向键):快速复制当前行 Alt+(向上/下方向键):移动当前行 Ctrl+D:删除当前行 Ctrl+/:快速(取消)注释当前行 Ctrl+Alt ...
- jsp 分页, 判断是第一页,和最后一页.
<% //页的行数 int pagesize =20; //当前页 int currentPage = 1; try { currentPage = Integer.parseInt(reque ...
- python守护线程
如果你设置一个线程为守护线程,就表示你在说这个线程是不重要的,在进程退出的时候,不用等待这个线程退出.如果你的主线程在退出的时候,不用等待那些子线程完成,那就设置这些线程的daemon属性.即在线程开 ...
- [转]深入理解JavaScript的变量作用域
1.JavaScript的作用域链 2.函数体内部,局部变量的优先级比同名的全局变量高. 3.JavaScript没有块级作用域. 4.函数中声明的变量在整个函数中都有定义. 5.未使用var关键字定 ...