为何要使用WCF
在Silverlight中,网络在某些方面与.NET Framework中的栈相同。可以使用套接字类HttpRequest和WebClient,WCF。但只能用异步方法在网络上首发请求。而其UI线程不应阻塞。而所谓异步调用,就是在主线程进行其他操作的时候,还在进行下载请求,不用等待函数的返回值。在此AJAX就是典型的,不刷新页面调用HttpRequest来进行操作,就是异步了。
具体搭建
首先,新建一个WCF应用程序如下,命名为Services:
具体WCF的细节可参见Artech的我的WCF之旅(1):创建一个简单的WCF程序
这时,我们就要给WCF服务指定一个端口了,可以在Service工程上选择属性,指定一个如下的端口:
编译后,可在创建的Silverlight应用程序工程上右键,点击添加服务应用。
点击Discover找到Service1.svc点击ok即可,这时WCF的服务已添加到Silverlight中。
好了,然后在Silverlight应用程序中的MainPage.xaml.cs中添加如下代码:
public MainPage()
{
Binding binding = new BasicHttpBinding();
EndpointAddress endPoint = new EndpointAddress("http://localhost:49628/Service1.svc");
ServiceReference1.Service1Client client = new ServiceReference1.Service1Client(binding, endPoint);
client.GetDataCompleted += new EventHandler<ServiceReference1.GetDataCompletedEventArgs>(client_GetDataCompleted);
client.GetDataAsync(1);
// Required to initialize variables
InitializeComponent();
}
默认的wcf工程中,GetData(int value),我们传入一个1,返回的结果是一个字符串,"You entered: 1”。
private void client_GetDataCompleted(object sender, ServiceReference1.GetDataCompletedEventArgs e)
{
string ct = e.Result;
}
可以run了,但是会报如下错:
没有正确的跨域策略,这是我们可以在Service添加一个xml文件,命名为clientaccesspolicy.xml,在其中写入如下代码:
<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>s
这样问题就解决了。好了,来探究下为什么?
原来在客户端的浏览器内部运行的应用程序有一个重要的限制:客户端只能从控件来自的同一个服务器中访问网络服务。Silverlight网络栈会检查服务器上是否存在Silverlight Policy文件clientaccesspolicy.xml.如果这个文件不存在,则Silverlight也接受crossdomain.xml文件,而Adobe Flash也使用这个文件。有了第一个文件,就不会检查第二个文件了。
xml中的意思是说对于来自*(即所有)的客户端,把访问权限授予所有子路径。