Silverlight同步(Synchronous)调用WCF服务

转载 2011年01月11日 09:10:00

 Silverlight的RIA应用中访问远端的WebService或WCF服务,都是通过异步线程模式调用的。在某些情况下我们的调用是需要同步进行,虽然Silverlight没有内置同步线程模式调用远端服务接口,但是我们可以通过多线程的处理来伪装出同步调用的实现。在.NET Framework的多线程编程中提供了丰富的线程接口,其中AutoResetEvent和ManualResetEvent在多线程编码中最为常用,本文将介绍如何通过AutoResetEvent的线程等待特性实现Silverlight同步调用远端WCF服务。

一、定义WCF服务

  为了演示同步调用WCF服务的实现,提供一个简单的WCF服务接口,完成返回一本图书基本信息,WCF服务接口定义如下:

[ServiceContract]
public interface IDataService
{
    [OperationContract]
    Book GetBook();
}

public class Book
{
    public int ID { get; set; }
    public string Name { get; set; }
    public string Author { get; set; }
    public double Price { get; set; }
}
  接口提供一个返回图书基本信息的方法,包括图书编好,图书名,图书作者以及图书价格。接口具体的实现如下代码:

public class DataService : IDataService
{
    public Book GetBook()
    {
        return new Book
        {
            ID = 1001,
            Name = "《三国演义》",
            Author = "罗贯中",
            Price = 89.50
        };
    }
}
   如上提供可正常运行的WCF服务接口,在需要调用接口的地方通过WEB引用既可生成该服务的客户端代理对象。

二、基于MVVM模式的视图模型

  MVVM模式的核心为INotifyPropertyChanged接口,对于实体模型对象和UI控件元素间提供了完善的同步更新特性。为了方便界面元素同步更新,这里引入了MVVP模式的简单应用。

public class ViewModelBase : INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;

    protected void RaisePropertyChangedEvent(string propertyName)
    {
        var handler = PropertyChanged;
        if (handler != null)
            handler(this, new PropertyChangedEventArgs(propertyName));
    }
}
  还需要对应于服务接口中的Book对象定义一个ViewModel对象,详细如下代码所示:

public class BookViewModel : ViewModelBase
{
    private int iD;
    /// <summary>
    /// 图书ID
    /// </summary>
    public int ID
    {
        get { return iD; }
        set
        {
            iD = value;
            RaisePropertyChangedEvent("ID");
        }
    }

    private string name;
    /// <summary>
    /// 图书名称
    /// </summary>
    public string Name
    {
        get { return name; }
        set
        {
            name = value;
            RaisePropertyChangedEvent("Name");
        }
    }

    private string author;
    /// <summary>
    /// 图书作者
    /// </summary>
    public string Author
    {
        get { return author; }
        set
        {
            author = value;
            RaisePropertyChangedEvent("Author");
        }
    }

    private double price;
    /// <summary>
    /// 图书价格
    /// </summary>
    public double Price
    {
        get { return price; }
        set
        {
            price = value;
            RaisePropertyChangedEvent("Price");
        }
    }
}
三、基于AutoResetEvent的同步实现

   利用AutoResetEvent的线程等待特性,可以折中实现Silverlight同步调用远端WCF服务。其原理就是在Silverlight发起异步调用远端WCF的时候进行线程阻塞,比记录异步调用远端WCF服务接口的完成事件,当异步调用完成后就终止线程阻塞,从而获取状态事件对象中或得调用远程接口所返回的结果。由于视图模型对象实现了INotifyPropertyChanged接口能够及时的更新界面元素,以此间接的就实现了同步方式调用。

public class AsyncCallStatus<T>
{
    public AsyncCallStatus()
    {

    }

    public T CompletedEventArgs { get; set; }
}
public class BookFacade
{
    private AutoResetEvent autoResetEvent = new AutoResetEvent(false);

    public void GetBook(BookViewModel viewModel)
    {
        if (viewModel == null)
        {
            throw new ArgumentNullException("viewModel", "参数不能为空。");
        }

        DataService.DataServiceClient client = new DataService.DataServiceClient();
        client.GetBookCompleted += client_GetBookCompleted;

        var status = new AsyncCallStatus<GetBookCompletedEventArgs>();
        client.GetBookAsync(status);
        //阻塞线程
        autoResetEvent.WaitOne();

        if (status.CompletedEventArgs.Error != null)
        {
            throw status.CompletedEventArgs.Error;
        }
        var book = status.CompletedEventArgs.Result;
        viewModel.ID = book.ID;
        viewModel.Name = book.Name;
        viewModel.Author = book.Author;
        viewModel.Price = book.Price;
    }

    private void client_GetBookCompleted(object sender, GetBookCompletedEventArgs e)
    {
        var status = e.UserState as AsyncCallStatus<GetBookCompletedEventArgs>;

        status.CompletedEventArgs = e;
        //终止线程阻塞
        autoResetEvent.Set();
    }
}
四、Silverlight前端调用

  Siverlight前端就简单布局一个表单作为数据呈现界面,其代码如下:

<Grid x:Name="LayoutRoot" Background="White">
    <Grid HorizontalAlignment="Left" Name="grid1" VerticalAlignment="Top" Width="300" Margin="20">
        <Grid.RowDefinitions>
            <RowDefinition Height="30"></RowDefinition>
            <RowDefinition Height="30"></RowDefinition>
            <RowDefinition Height="30"></RowDefinition>
            <RowDefinition Height="30"></RowDefinition>
            <RowDefinition Height="30"></RowDefinition>
        </Grid.RowDefinitions>
        <Grid.ColumnDefinitions>
            <ColumnDefinition Width="60"></ColumnDefinition>
            <ColumnDefinition Width="*"></ColumnDefinition>
        </Grid.ColumnDefinitions>
        <sdk:Label  HorizontalAlignment="Left" Content="图书编号:" VerticalAlignment="Center" Grid.Column="0" Grid.Row="0"/>
        <TextBox Text="{Binding ID}" Grid.Column="1" Grid.Row="0"></TextBox>
        <sdk:Label  HorizontalAlignment="Left" Content="图书名称:" VerticalAlignment="Center" Grid.Column="0" Grid.Row="1"/>
        <TextBox Text="{Binding Name}" Grid.Column="1" Grid.Row="1"></TextBox>
        <sdk:Label  HorizontalAlignment="Left" Content="图书作者:" VerticalAlignment="Center" Grid.Column="0" Grid.Row="2"/>
        <TextBox Text="{Binding Author}" Grid.Column="1" Grid.Row="2"></TextBox>
        <sdk:Label  HorizontalAlignment="Left" Content="图书价格:" VerticalAlignment="Center" Grid.Column="0" Grid.Row="3"/>
        <TextBox Text="{Binding Price}" Grid.Column="1" Grid.Row="3"></TextBox>
           
        <Button Content="查询" Grid.Column="1" Grid.Row="4" Width="60" Height="23" Click="Button_Click"></Button>
    </Grid>
</Grid>
   通过按钮执行调用WCF服务接口查询图书信息,按钮事件直接使用上面所写的图书门面类(BookFacade)的调用服务方法即可。

private void Button_Click(object sender, RoutedEventArgs e)
{
    try
    {
        ThreadPool.QueueUserWorkItem(delegate(object o)
        {
            BookViewModel viewModel = new BookViewModel();

            new BookFacade().GetBook(viewModel);

            Deployment.Current.Dispatcher.BeginInvoke(() => this.DataContext = viewModel);
        });
    }
    catch (Exception ex)
    {
        MessageBox.Show(ex.ToString());
    }
}
   最终的运行如下图所示效果:

  

silverlight学习之路(2)异步编程趋于同步 利用async和await调用wcf服务

首先非常感谢张伟文老师。
  • duyelang
  • duyelang
  • 2014年09月10日 14:12
  • 1050

SilverLight商业应用程序开发---学习笔记(2) WCF RIA服务

从服务器暴露数据 何为WCF RIA服务 RIA服务建立在WCF顶层,用于建立与服务器可沟通的数据驱动Silverlight应用程序.可以描述如下 数据为中心的设计模式 可提供高级数据管理,...
  • Eric_K1m
  • Eric_K1m
  • 2013年07月01日 14:30
  • 962

VS 2010中使用C#创建及调用WCF完整实例(Windows服务宿主)

一个傻瓜式的Windows服务宿主模式下的WCF的完整示例 只注重于实现,不讲理论...
  • meerio
  • meerio
  • 2015年10月26日 21:49
  • 5198

WCF 之 AJax前台调用WCF服务

调用WCF服务,我们一般都是中客户端的配置文件中配置好WCF服务的终结点,然后中后台代码中实例化WCF服务,然后调用其中的方法,今天给大家介绍一种,不需要配置终结点,直接中前台通过AJax方法调用WC...
  • u010786678
  • u010786678
  • 2015年09月29日 22:40
  • 2192

采用axis2方式Java客户端调用WCF服务端Web Service

首先确保WebService服务开启,如下图:
  • yangshijin1988
  • yangshijin1988
  • 2014年08月13日 22:28
  • 2485

Java客户端调用WCF服务

一.如何判断接口用的是WCF服务的打开别人给你的接口地址(WSDL) 里面有svcutil.exe 就是WCF二.在eclipse中安装axis2插件1.原因: 为什么安装axis2插件,为了...
  • u011008029
  • u011008029
  • 2016年05月04日 14:06
  • 2557

WCF之各种WCF引用方式

写在开头:本文内容来自 WCF全面解析中的一个经典例子,如果你已经看过了,那么可以忽略本文,本文旨在和大家分享不一样的WCF使用方法。 准备工作: 1.创建解决方案WCFService(当然名字可...
  • dyllove98
  • dyllove98
  • 2013年07月30日 18:54
  • 13000

一个WCF服务开发与调用的完整示例

开发工具:VS2008 开发语言:C# 开发内容:简单的权限管理系统 第一步、建立WCF服务库 点击确定,将建立一个WCF 服务库示例程序,自动生成一个包括IService1.cs和Ser...
  • sinat_15155817
  • sinat_15155817
  • 2017年02月24日 17:41
  • 182

Android/iOS访问wcf传递参数为实体对象的问题

简单记录一下使用实体对象作为参数的传递! 在服务端使用webservice时是没有问题的,但是当替换成wcf时就出现传递的参数无法序列化的问题! 服务端代码: Service1.svc namespa...
  • wuwo333
  • wuwo333
  • 2015年01月19日 14:57
  • 3095

Post方式调用wcf服务

我们平常在PC端调用WCF服务,只要知道WCF服务的地址,客户端直接添加引用服务就可以使用了,殊不知还有其他方式,其实,我们也可以 通过HTTP POST的方式调用WCF服务,这样就不用添加引用了,在...
  • zx13525079024
  • zx13525079024
  • 2014年05月26日 12:03
  • 10428
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Silverlight同步(Synchronous)调用WCF服务
举报原因:
原因补充:

(最多只允许输入30个字)