WCF学习笔记(4)——宿主iis部署wcf

本文将部署一个wcf+silverlight简单实例,以下是详细步骤:

(环境:服务端win2003,iis6.0,asp.net4.0;客户端winXP,iis5.1,vs2010,silverlight4.0)


 

1.新建一个解决方案,方案中新建一个网站(选择wcf服务)

  再新建一个wcf服务库

  此时解决方案结构如下:


2.删除WCFService.App_code自带的IService.cs,Service.cs,然后添加引用我们自定义的wcf服务库WcfServiceLibrary1

  打开WCFService宿主网站的Service.svc文件,内容如下:

<%@ ServiceHost Language="C#" Debug="true" Service="Service" CodeBehind="~/App_Code/Service.cs" %>

  由于我们删除了vs自动生成的文件,改用我们自定义的wcf服务库文件,所以将以上代码修改为:(删除掉CodeBehind,修改Service为我们自定义的wcf服务库中,默认生成的Service1服务契约

<%@ ServiceHost Language="C#" Debug="true" Service="WcfServiceLibrary1.Service1" %>

  接下来设置一下我们的WCFService宿主的配置文件Web.config,可以使用编辑WCF配置工具,方便我们的设置

  打开设置工具,首先新建一个服务

  找到我们添加引用过来的wcf服务(宿主网站引用的服务.dll一般在Bin文件夹下,如果你添加后没有看到它,试着重新生成以下解决方案

  选择我们的模式,由于本例后面将用silverlight程序和wcf通信,宿主也将选择iis,所以选择基于http的模式

  终结点可以选择不填

  最终获得配置如下:

  保存设置,我们将会得到自定生成后的Web.config

  当然,为了部署在我们iis之上,必须设置固定一下宿主网站的端口号,并且修改我们的虚拟路径

  查看我们是否创建服务成功,可以选择在浏览器中查看

  创建成功,我们将得到一个返回页面(记下这个服务的路径,它将在本例后面添加服务时使用到,本例中的路径:http://localhost:9090/Service.svc

 


3.由于我们独立出了wcf服务,将会遇到跨域问题,所以要添加一个跨域文件clientaccesspolicy.xml在宿主网站的根目录下由于我们将用silverlight程序来通信,所以只需要添加一个clientaccesspolicy.xml跨域文件

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>

4.在我们自定义的wcf服务库中,简单写一个wcf服务契约来完成我们的例子吧

IService1.cs:

namespace WcfServiceLibrary1
{
    // 注意: 使用“重构”菜单上的“重命名”命令,可以同时更改代码和配置文件中的接口名“IService1”。
    [ServiceContract]
    public interface IService1
    {
        [OperationContract]
        string SayHello();

        // TODO: 在此添加您的服务操作
    }
}

Service1.cs:

namespace WcfServiceLibrary1
{
    // 注意: 使用“重构”菜单上的“重命名”命令,可以同时更改代码和配置文件中的类名“Service1”。
    public class Service1 : IService1
    {
        #region IService1 成员

        public string SayHello()
        {
            return "Hello WCF";
        }

        #endregion
    }
}

5.接下来我们新建一个silverlight程序,用来和wcf服务通信吧

  在解决方案中,新建一个silverlight程序

  由于本例中,我们将分开独立的发布部署wcf服务和silverlight演示网站,所以注意选择新建的silverlight时,要新建一个Web项目安置,而不要选择安置在wcf服务的寄宿网站中(无需启用WCF RIA服务)

  此时得到解决方案结构如下:


6.然后我们来完善我们的silverlight程序来完成和wcf服务的通信吧

首先在silverlight程序添加服务引用

  这里的地址我们填写前面测试wcf服务布置成功与否时,返回成功页面上的地址,本例如下:

  确定添加后,得到解决方案结构如下:

  vs2010为我们自动生成了服务引用ServiceReference1,和配置文件ServiceReferences.ClientConfig(如果你足够仔细,一定发现了我们在地址中使用了localhost的本机地址,本机的地址满足我们在本机开发环境下正常调试即可,而我们将会在稍后部署到服务端服务器的iis上时,再做修改

  简单的设置一下本例的展示界面吧:

  一个button,一个textblock,简约却也清楚不是嘛,在button1的点击事件中,添加代码如下:

        private void button1_Click(object sender, RoutedEventArgs e)
        {
            //绑定模式为基本http模式
            Binding binding = new BasicHttpBinding();
            //终结点地址
            EndpointAddress endPoint = new EndpointAddress("http://localhost:9090/Service.svc");
            //设置终结点,包括模式和地址
            ServiceReference1.Service1Client client = new ServiceReference1.Service1Client(binding, endPoint);
            //设置调用异步调用完成返回成功的执行方法
            client.SayHelloCompleted += new EventHandler<ServiceReference1.SayHelloCompletedEventArgs>(client_SayHelloCompleted);
            //异步调用
            client.SayHelloAsync();
        }

        void client_SayHelloCompleted(object sender, ServiceReference1.SayHelloCompletedEventArgs e)
        {
            this.textBlock1.Text = e.Result;
        }

  点击button1,我们将发出和wcf服务的通信,等待wcf服务返回的结果,演示在textblock1上,重新生成一下解决方案,F5运行一下,看看效果吧:

 

  仔细的你一定会发现,同时起来的还有一个WCF服务主机的窗口,如果WCF服务通信有问题的话,你可以在这里查看异常


 

 7.OK,开心的看着我们的wcf+silverlight的程序已经调试成功了,觉得这还是挺简单的,但是别忘了,我们的项目还没有部署呢,离最后的上线还有一个重要的步骤,那就是在服务器上的部署

首先把我们的解决方案中的两个Web项目发布一下(一个wcf服务宿主Web,一个silverlight寄宿Web,因为winXP环境下,只能安装iis5.1(iis5.1 只能发布一个网站)所以我们每次只能发布一个,step by step吧

每次发布生成的文件(本例采用文件系统发布在本机的D:\www下),我们分别保存下来,然后放置到服务器上

ok,来看一下我们的服务端的iis配置吧

ServiceTest:安置我们的wcf服务宿主网站,注意端口选择我们之前约定好固定的9090

asp.net版本4.0

SilverlightTest:安置我们的Silverlight寄宿网站,同样的设置asp.net版本4.0iis6.0中,需设置端口,本例的Silverlight寄宿网站端口设8080,服务端ip为192.168.1.131,局域网内访问

ok,在客户端访问一下我们的成果吧

首先输入wcf服务的地址,查看返回结果

 

You have created a service!看来我们的wcf服务布置成功了,接着我们访问一下演示用的silverlight吧

如果rp不走运的话,会发现有返回的异常,内容如下:

消息: Unhandled Error in Silverlight Application 操作过程中出现异常,结果无效。有关异常的详细信息,请查看 InnerException。   位于 System.ComponentModel.AsyncCompletedEventArgs.RaiseExceptionIfNecessary()
   位于 SilverlightApplication1.ServiceReference1.SayHelloCompletedEventArgs.get_Result()
   位于 SilverlightApplication1.MainPage.client_SayHelloCompleted(Object sender, SayHelloCompletedEventArgs e)
   位于 SilverlightApplication1.ServiceReference1.Service1Client.OnSayHelloCompleted(Object state)

分析一下说明,问题出在silverlight调用wcf服务时出错但是你如果返回正常,那一定是因为你本机的9090端口wcf没关吧

问题出在silverlight调用wcf时,使用的服务地址当时我们用的是localhost,之前由于是在本机调试,服务端客户端在同一台主机,他们的地址都是localhost,所以问题没有出现,但是silverlight程序是需要客户端下载到本地去执行的,当我们试图访问放置在真正服务端上的服务时,当然只能去找本机端口的wcf服务,而我们真正的想法是让它去找服务器端口下的wcf服务

知道了问题所在,开始修改一下我们的silverlight程序吧

首先修改一下我们的silverlight程序引用的wcf服务的地址吧:

 

这次我们使用服务端的ip地址(本例中服务器ip为192.168.1.131)

同时修改一下我们button1按键事件:

        private void button1_Click(object sender, RoutedEventArgs e)
        {
            //绑定模式为基本http模式
            Binding binding = new BasicHttpBinding();
            //终结点地址
            //EndpointAddress endPoint = new EndpointAddress("http://localhost:9090/Service.svc");
            EndpointAddress endPoint = new EndpointAddress("http://192.168.1.131:9090/Service.svc");
            //设置终结点,包括模式和地址
            ServiceReference1.Service1Client client = new ServiceReference1.Service1Client(binding, endPoint);
            //设置调用异步调用完成返回成功的执行方法
            client.SayHelloCompleted += new EventHandler<ServiceReference1.SayHelloCompletedEventArgs>(client_SayHelloCompleted);
            //异步调用
            client.SayHelloAsync();
        }

这样silverlight程序在调用wcf服务时,就会去真正的服务端查找了(可以先把localhost地址注释掉,因为我们本机客户端的ip地址是不可以和服务端的ip相同的,我们还是会用到localhost来调试程序的

ok,重新生成一下我们的解决方案,将修改过后的silverlight寄宿网站放到服务器上,再次访问一下,问题解决,结果如下:


 

 

posted on 2013-01-24 12:57  影天  阅读(13705)  评论(6编辑  收藏  举报

导航