宿主iis部署wcf

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

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

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


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

宿主iis部署wcf

再新建一个wcf服务库

宿主iis部署wcf

此时解决方案结构如下:

宿主iis部署wcf


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

宿主iis部署wcf

打开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配置工具,方便我们的设置

宿主iis部署wcf

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

宿主iis部署wcf

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

宿主iis部署wcf

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

宿主iis部署wcf

终结点可以选择不填

宿主iis部署wcf

最终获得配置如下:

宿主iis部署wcf

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

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

宿主iis部署wcf

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

宿主iis部署wcf

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

宿主iis部署wcf


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

宿主iis部署wcf

clientaccesspolicy.xml内容如下:

宿主iis部署wcf
<?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>
宿主iis部署wcf

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

IService1.cs:

宿主iis部署wcf
namespace WcfServiceLibrary1
{
// 注意: 使用“重构”菜单上的“重命名”命令,可以同时更改代码和配置文件中的接口名“IService1”。
[ServiceContract]
public interface IService1
{
[OperationContract]
string SayHello(); // TODO: 在此添加您的服务操作
}
}
宿主iis部署wcf

Service1.cs:

宿主iis部署wcf
namespace WcfServiceLibrary1
{
// 注意: 使用“重构”菜单上的“重命名”命令,可以同时更改代码和配置文件中的类名“Service1”。
public class Service1 : IService1
{
#region IService1 成员 public string SayHello()
{
return "Hello WCF";
} #endregion
}
}
宿主iis部署wcf

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

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

宿主iis部署wcf

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

宿主iis部署wcf

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

宿主iis部署wcf


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

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

宿主iis部署wcf

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

宿主iis部署wcf

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

宿主iis部署wcf

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

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

宿主iis部署wcf

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

宿主iis部署wcf
        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;
}
宿主iis部署wcf

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

宿主iis部署wcf

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

宿主iis部署wcf


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

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

宿主iis部署wcf

宿主iis部署wcf

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

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

宿主iis部署wcf

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

宿主iis部署wcf

asp.net版本4.0

宿主iis部署wcf

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

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

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

宿主iis部署wcf

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

宿主iis部署wcf

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

宿主iis部署wcf
消息: 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)
宿主iis部署wcf

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

宿主iis部署wcf

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

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

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

宿主iis部署wcf

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

宿主iis部署wcf

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

宿主iis部署wcf
        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();
}
宿主iis部署wcf

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

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

宿主iis部署wcf


上一篇:【阿里云产品公测】利用PTS服务优化网站数据库读写性能


下一篇:.NET-记一次架构优化实战与方案-梳理篇