zoukankan      html  css  js  c++  java
  • 宿主iis部署wcf

    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寄宿网站放到服务器上,再次访问一下,问题解决,结果如下:


  • 相关阅读:
    php_sphinx安装使用
    编程实现自定义解决方案
    NetBeans 时事通讯(刊号 # 55 May 06, 2009)
    Java 上下文与依赖注入(JSR 299)[1]
    走进Java 7模块系统
    VC编程中关于新建的框架窗口的销毁的一个心得
    家常菜之豆豉蒸鸡翅
    用ntsd命令强制杀死进程
    NetBeans 时事通讯(刊号 # 56 May 21, 2009)
    NetBeans 时事通讯(刊号 # 56 May 21, 2009)
  • 原文地址:https://www.cnblogs.com/qq260250932/p/5330807.html
Copyright © 2011-2022 走看看