Silverlight+WCF 新手实例 象棋 WCF通讯轮询(十七)

上节说到:

关于双向通讯,官方提供了N种可以双向的,不过今天要用到的,

是pollingDuplexHttpBinding,一个扩展的轮询机制的双向通讯,当你也可以尝试用上面的通讯方式去试一试。

既然是扩展,就说明默认没有,那我们首先就要添加扩展了,用的是默认官方是提供的DLL,就在我们安装的Silverlight4的目录里:

正常路径为:C:\Program Files\Microsoft SDKs\Silverlight\v4.0\Libraries\Server\System.ServiceModel.PollingDuplex.dll

 

这一节我们来实现PollingDuplexHttpBinding,轮询机制的双向通讯。

 

以下开始内容不上图片,参考Silverlight+WCF 新手实例 象棋 WCF通讯跨域(十五)

我们再开一个项目来讲解,有了Hellow,有了World,这节就叫HellowWorld

文件—》新建->项目-》Silverlight应用程序-》起名叫:HellowWorld

确定后还是:HellowWorld和HellowWorld.web应用程序,两个项目

 

我们对着解决方案右键,添加新建项目:建立WCF 服务应用程序->输入名称为:HellowWorldService

接着我们把默认的Service1.cs和Service1.svc删除:

删除后,我们新建一个新的服务,叫Service.svc

我们提前修改下服务的端口号为12321,这样添加服务引用后,不用再改配置文件的端口。

OK,这时项目情况如下:



接下来我们要为项目添加DLL,对着项目引用右键-》添加引用:



选择浏览,并定位到:C:\Program Files\Microsoft SDKs\Silverlight\v4.0\Libraries\Server\System.ServiceModel.PollingDuplex.dll



回车确定,添加引用完后,我们需要修改下服务的配置文件“Web.config”

由于轮询为扩展的,所以需要在配置文件里添加两个节点:

 


HellowWorldService web.config

<?xml version="1.0" encoding="utf-8"?>
<configuration>

  <system.web>
    <compilation debug="true" targetFramework="4.0" />
  </system.web>
  <system.serviceModel>
    <behaviors>
      <serviceBehaviors>
        <behavior>
          <!-- 为避免泄漏元数据信息,请在部署前将以下值设置为 false 并删除上面的元数据终结点 -->
          <serviceMetadata httpGetEnabled="true"/>
          <!-- 要接收故障异常详细信息以进行调试,请将以下值设置为 true。在部署前设置为 false 以避免泄漏异常信息 -->
          <serviceDebug includeExceptionDetailInFaults="false"/>
        </behavior>
      </serviceBehaviors>
    </behaviors>
      <!--这里是添加的开始-->
      <services>
          <service name="HellowWorldService.Service" >
              <endpoint address="" binding="pollingDuplexHttpBinding" contract="HellowWorldService.IService" />
              <endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange"/>
          </service>
      </services>
      <extensions>
          <bindingExtensions>
              <add name="pollingDuplexHttpBinding" type="System.ServiceModel.Configuration.PollingDuplexHttpBindingCollectionElement,System.ServiceModel.PollingDuplex, Version=4.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35"/>
          </bindingExtensions>
      </extensions>
      <!--这里是添加的结束-->
    <serviceHostingEnvironment multipleSiteBindingsEnabled="true" />
  </system.serviceModel>
 <system.webServer>
    <modules runAllManagedModulesForAllRequests="true"/>
  </system.webServer>
  
</configuration>

 

大伙照着把添加的开始和添加的结束那段Copy过去就完事了。

OK,配置完事后,我们要写服务端代码了,双向通讯,但然少不了要调用客户端的代码了。

看们看下IService.cs文件


[ServiceContract(CallbackContract=typeof(ICallBack))]
    public interface IService
    {
        [OperationContract(IsOneWay = true)]
        void SayHellow(string name);
    }
    public interface ICallBack
    {
        [OperationContract(IsOneWay=true)]
        void ShowWorld(string worldName);
    }

 

看,接口代码相当的少,就算少还是要说明一下的:

而且你敲属性的时候,是有智能提示的:



IService:服务端接口,当然就是客户端调用了。注意头顶上那个CallbackContract=typeof(ICallBack),这里指定了回调接口。

ICallBack:回调接口,我新加的,是给客户端实现,然后服务端调用。这个名称你可以随便起,和typeof里的对应上就行了。

是不是发现多了一个(IsOneWay = true)属性,什么意思?就是单向调用,不需要返回值。

所以官方推荐,如果你的函数类型返回值为void时,最好加上。

 

接着我们要实现IService接口的方法了,那ICallBack要不要实现?当然不要,都说留给客户端实现了。

哦,那我们就实现IService接口方法去了:


public class Service : IService
    {
        #region IService 成员
        public void SayHellow(string name)
        {
            name="hellow"+name;
            ICallBack callBack = OperationContext.Current.GetCallbackChannel<ICallBack>();
            callBack.ShowWorld(name);
        }
        #endregion
    }

 

看到方法没有,有一句很长的代码,来获取ICallBack接口,然后调用了那个ShowWorld方法了。

这个代码记死也行:OperationContext.Current.GetCallbackChannel<ICallBack>();

反正把ICallBack换成你自己的接口名称,就是了。然后就可以调用了。

话说ICallBack方法是留给客户端实现的,我们服务端这里先调用着先,反正你得按接口实现,按接口办事,放心的很。

 

那三行代码总来来说就是:

1。客户端调用服务端的SayHellow(传入“路过秋天”);

2。服务端收到调用,自然会知道对方从哪条路上来的,所以能够GetCallbackChannel了。

3。接约定好的接口,我调用了你的ShowWorld方法,同时把加了“hellow:路过秋天“传过去。

至此,服务端代码写完了。是不是相当相当的简单,只要理解好了。

不过服务端还是有点事,什么事?加那个跨域文件啊,谁让你独立一个服务出来。

加就加了,还是新建一个:clientaccesspolicy.xml文件,内容为:


<?xml version="1.0" encoding="utf-8"?>
<access-policy>
    <cross-domain-access>
        <policy>
            <allow-from http-request-headers="*">
                <domain uri="*"/>
            </allow-from>
            <grant-to>
                <resource path="/" include-subpaths="true"/>
            </grant-to>
        </policy>
    </cross-domain-access>
</access-policy>

 

这下服务端事件就全搞完了,接下来看客户端的了。

记得先添加服务引用-》发现->引用名称叫:HellowWorldService

OK,接着我们还是要弄一个和上两次一样的界面,来调用,从上节那里Copy来xaml的代码,-_-..这界面重复三次了:


<Grid x:Name="LayoutRoot" Background="White">
        <Button Content="WCF 调用" Height="23" HorizontalAlignment="Left" Margin="84,111,0,0" Name="btnCallWCF" VerticalAlignment="Top" Width="75" Click="btnCallWCF_Click" />
        <TextBox Height="23" HorizontalAlignment="Left" Margin="84,71,0,0" Name="txtName" VerticalAlignment="Top" Width="120" />
        <TextBlock Height="23" HorizontalAlignment="Left" Margin="228,71,0,0" Name="tbMsg" Text="显示的内容" VerticalAlignment="Top" />
    </Grid>

 

后台代码调用除了差不多也就是有一点小变化:

我们不是实例一个BasicHttp通道了,而是实例化一个PollingDuplex通道了。并设置了下每次轮询建立的有效时间为20分钟。


 private void btnCallWCF_Click(object sender, RoutedEventArgs e)
        {
            PollingDuplexHttpBinding binding = new PollingDuplexHttpBinding()
            {
                InactivityTimeout = TimeSpan.FromMinutes(20)
            };
            //Binding binding =new BasicHttpBinding();
            EndpointAddress endPoint = new EndpointAddress("http://localhost:12321/Service.svc");
            HellowWorldService.ServiceClient client = new HellowWorldService.ServiceClient(binding, endPoint);
            client.SayHellowAsync(txtName.Text);

            client.ShowWorldReceived += new EventHandler<HellowWorldService.ShowWorldReceivedEventArgs>(client_ShowWorldReceived);
        }

        void client_ShowWorldReceived(object sender, HellowWorldService.ShowWorldReceivedEventArgs e)
        {
            tbMsg.Text = e.worldName;
        }

 

客户端的接口实现是哪句啊?

看出来没,这两句就是那个ICallBack接口的实现了,当用户调用ShowWorld时候,就是tbMsg.Text=e.参数的时候了。


 client.ShowWorldReceived += new EventHandler<HellowWorldService.ShowWorldReceivedEventArgs>(client_ShowWorldReceived);
        }

        void client_ShowWorldReceived(object sender, HellowWorldService.ShowWorldReceivedEventArgs e)
        {
            tbMsg.Text = e.worldName;
        }

 

一切就绪:F5运行,输入"路过秋天"



回车调用,OK,结果出来了。



 

OK,WCF通讯基础到此就结束了,下节开始大干特干的应用于了。

提供源码下载:点击下载

 

版权声明:本文原创发表于博客园,作者为路过秋天,原文链接:

http://www.cnblogs.com/cyq1162/archive/2010/07/13/1776482.html

时间: 2025-01-07 13:52:42

Silverlight+WCF 新手实例 象棋 WCF通讯轮询(十七)的相关文章

Silverlight+WCF 新手实例 象棋 WCF通讯方式(十六)

本节小讲通讯方式,上两节的我们讲的BasicHttpBinding,是其中的一种基础的通讯方式, 还有一种双向通讯的,专业词汇叫"双工通讯",不过双工通讯一听起来太以理解, 就好像那个特性"契约",一听就感觉像卖身契一样让人觉得匪夷所思. 下面全用双向代替双工说话了. 说双向通讯好理解了吧,再简单说就是:客户端可以调用服务端的方法:服务端也可以调用客户端的方法. 对于通讯方式,上两节,我们都是用:BasicHttpBinding,除了这个,还说过还有N种的.   为

Silverlight+WCF 新手实例 象棋 WCF通讯跨域(十五)

这节我们把WCF独立一个服务出来,不寄放在Web应用程序中. 我们再开一个项目来讲解,上节名称叫Hellow,这节名称起World: 文件->新建->项目->Silverlight应用程序->起名叫:World 确定后还是:World和World.web应用程序,两个项目   我们对着解决方案右键,添加新建项目:建立WCF 服务应用程序->输入名称为:WorldService: 接着我们把默认的Service1.cs和Service1.svc删除: 删除后,我们新建一个新的服

Silverlight+WCF 新手实例 象棋 WCF通讯基础(十四)

从此节开始,我们要开始WCF通讯了. 本游戏的只是基于WCF最基础的应用,没有WCF深奥的知识,相关的深奥知识请关注专业的WCF系列文章. 我们先来实现一个Silverlight里调用Hellow的尝试: 我们新建一个新的Silverlight应用程序:文件->新建->项目->Silverlight应用程序->Hellow 建立后就一个Hellow应用程序和一个Helow.web的Web应用程序.   一:我们将WCF集成在Web应用程序中 我们对着Web应用程序右键->添加

Silverlight+WCF 新手实例 象棋 回归WCF通讯应用-登陆(十八)

前四节,我们讲了通讯基础,从这节起,我们回归到项目中来,要将前面的WCF通讯知识应用进来. 之前的项目大伙没丢把,重新发下载地址:之前第3阶段代码:点此下载   我们为Chess项目的解决方案里,再添加WCF应用服务程序 由于Silverlight+WCF 新手实例 象棋 WCF通讯跨域(十五)已截图,所以这里不截图了. 步骤: 1.对着解决方案->右键->添加新建项目->选择WCF应用服务程序->输入:GameService 2.删除默认的IService1.cs和IServic

Silverlight+WCF 新手实例 象棋 主界面-事件区-游戏开始(二十七)

本专题出产简单原由: 一开始的初衷,只是想写个简单的单机BS人机对战版的,开始还下了点AI算法看看的: 但是写到最后,都写成了通讯版本的对战了,只因中间不小心看到了WCF的相关内容,顺便加了进来; 最后就定局了,反正新手实例,能加多点内容就加多点了. 关于原始初衷,后期再补上了.       好了,先上几个附加索引: 1:Silverlight+WCF 新手实例 象棋 在线演示 2:Silverlight+WCF 简单部署问题集 3:Silverlight4 ListBox bug 4:Silv

赶紧接着上一节:Silverlight+WCF 新手实例 象棋 主界面-实时聊天区(二十五) 这节我们实现上节没实现的纠结的进出房间的消息提示 我们清楚的知道,我们每个区都是一个用户控件,我们的在线用户进出时,得到聊天区域显示信息,这就涉及到用户控件之间的消息传递了。 在线用户区说:反正我是

赶紧接着上一节:Silverlight+WCF 新手实例 象棋 主界面-实时聊天区(二十五)  这节我们实现上节没实现的纠结的进出房间的消息提示 我们清楚的知道,我们每个区都是一个用户控件,我们的在线用户进出时,得到聊天区域显示信息,这就涉及到用户控件之间的消息传递了. 在线用户区说:反正我是没权直接去实时聊天区写东西,没办法,找中介委托设置一下了. 那谁是中介呢?委托谁呢?当然是衣食父母Index.xaml了,是它撑着整个家庭的. 好了,知道中介了,那我就签个委托书了: 回到OnlineUse

Silverlight+WCF 新手实例 象棋 房间状态更新(二十)

在线演示地址:Silverlight+WCF 新手实例 象棋 在线演示   这节开始,标题里就去掉"回归WCF通讯应用"几字了.   上节我们成功实现了进入房间,服务端也收到用户进入房间的请求了,这节,我们服务端收到进入房间请求后,通知在房间大门外的人更新房间状态. 我们要增加一个回调方法,ICallBack接口那,忘记的人回去看看WCF通讯那几篇(十四到十七节). 方法如下,以前说过了,回调的方法是给客户端实现的,服务端只管调就行了: using System.ServiceMode

Silverlight+WCF 新手实例 象棋 主界面-在线用户区(二十四)

在线演示地址:Silverlight+WCF 新手实例 象棋 在线演示 演示已更新到此节介绍:Silverlight+WCF 新手实例 象棋 介绍III(二十三)   这节我们来实现在线用户区的显示,把上两节介绍那张图再弄来,看在线用户区是哪块:   一眼扫过看到了,是第四区,现在开始了,还是上次下棋区域一样的逻辑,往Index.xaml里拉一个Board控件,然后后台写两行代码代码一下. 当然了,得新建一个用户控件:就叫:OnlineUser.xaml,好,空白的在线用户建完了,下面还是两步实

Silverlight+WCF 新手实例 象棋 主界面-棋谱-获取列表(三十八)

在线演示地址:Silverlight+WCF 新手实例 象棋 在线演示   在Silverlight+WCF 新手实例 象棋 主界面-棋谱-布局写谱(三十六)中,我们完成下棋双方的棋谱显示,这节,我们为观众增加棋子列表: 观众进入房间后,第一时间当然也要获取棋步列表了,不然进来干麻呢?你当这是聊天室啊,光聊天不看棋.   首先,当然是要在服务端添加一个获取棋步列表的接口方法了: WCF服务端,IService.cs:  /// <summary>     /// 服务端方法接口 by 路过秋天