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()); }}
最终的运行如下图所示效果: