ASP.NET2.0中数据源控件之异步数据访问(1)
点击次数:26 次 发布日期:2008-11-26 14:47:23 作者:源代码网
|
源代码网推荐 源代码网推荐 引言 源代码网推荐 源代码网推荐 在线程池中有固定不变的大量线程可用于服务请求,遗憾的是,该解决方案并非仅仅提高限制(还会增加线程占用资源以及 CPU 占用资源)。因此,当一个页面被阻止而等候另一个服务器时,它还在占用线程,因而可能会导致其他传入的请求在队列中等候更长的时间。这将导致对站点的访问变慢,并降低 CPU 的利用率。在 Visual Studio 2005 中,我们引入了异步页面,这使得控件能够定义它们希望异步完成的任务,即,无需阻止用来处理请求的线程。在此将不介绍异步页面本身的详细信息,Dmitry(英文)和 Fritz Onion(英文)中以前已经有所介绍。此处要介绍的是如何在数据源控件中利用此功能,使用加载项框架来实现异步数据源。 源代码网推荐 源代码网推荐 背景 源代码网推荐 源代码网推荐 在第 1 部分中,间接提到了 DataSourceView 类的有些古怪的设计: 源代码网推荐 源代码网推荐 public abstract class DataSourceView { 源代码网推荐 public virtual void Select(DataSourceSelectArguments arguments, 源代码网推荐 DataSourceViewSelectCallback callback); 源代码网推荐 protected abstract IEnumerable ExecuteSelect( 源代码网推荐 DataSourceSelectArguments arguments); 源代码网推荐 ... 源代码网推荐 } 源代码网推荐 源代码网推荐 您会注意到,公共 Select 方法实际上并不返回任何数据。而是接受一个回拨,并通过该回拨来返回数据。它只调用受保护的 ExecuteSelect(它始终执行同步数据访问)来检索要退还给数据绑定控件的数据。DataSourceView 类的默认实现实际上不会异步执行任何操作。原因在于,并不存在任何现成的异步数据源控件。但 OM 的设计确实允许实现异步数据访问,在这种设计下,数据在异步工作完成之后才可用。因此,我们就有了一个基于回拨的模型。 源代码网推荐 源代码网推荐 那些熟悉框架中的异步 API 的人会注意到缺少了异步模式:公共 Select、BeginSelect 和 EndSelect 方法,在这些方法中,数据绑定控件选择要调用哪些方法。但是,数据绑定控件并不能确定是选择同步 API 还是选择异步 API。此外,在数据绑定控件上添加属性也毫无作用。数据源控件封装了有关如何访问数据存储的详细信息,对数据存储的访问是同步发生还是异步发生应该根据数据源是否基于语义来决定或者根据自定义属性来决定。潜在的“bool PerformAsyncDataAccess”属性的正确位置适合于数据源控件本身。这还使得数据源控件可以使用一种方法来执行数据访问,即使多个数据绑定控件被绑定到同一个数据源。至此已多次解释了该体系结构所蕴涵的这些微妙的概念,但愿能阐明该设计。 源代码网推荐 源代码网推荐 关于异步任务,最后要注意的一点是:页面是否应该执行任何异步工作完全由页面开发人员最终决定(通过 Page 指令的 Async 属性)。因此,任何编写良好的数据源控件必须退化为根据需要来执行同步数据访问。 源代码网推荐 源代码网推荐 框架 源代码网推荐 源代码网推荐 在此框架中(在此系列结尾会用示例的剩余部分来演示这一点),已将 AsyncDataSource 和 AsyncDataSourceView 基类放在一起,这些基类可以用于实现能够执行异步数据访问的数据源控件。以下大概介绍了框架内容,以及有助于弄清楚其含义的一些注释: 源代码网推荐 源代码网推荐 public abstract class AsyncDataSourceControl : DataSourceControl, 源代码网推荐 IAsyncDataSource { 源代码网推荐 private bool _performAsyncDataAccess; 源代码网推荐 源代码网推荐 protected AsyncDataSourceControl() { 源代码网推荐 _performAsyncDataAccess = true; 源代码网推荐 } 源代码网推荐 源代码网推荐 public virtual bool PerformAsyncDataAccess { 源代码网推荐 get; set; 源代码网推荐 } 源代码网推荐 源代码网推荐 bool IAsyncDataSource.IsAsync { 源代码网推荐 get { return _performAsyncDataAccess && Page.IsAsync; } 源代码网推荐 } 源代码网推荐 } 源代码网推荐 源代码网推荐 public abstract class AsyncDataSourceView : DataSourceView { 源代码网推荐 源代码网推荐 protected abstract IAsyncResult BeginExecuteSelect( 源代码网推荐 DataSourceSelectArguments arguments, 源代码网推荐 AsyncCallback asyncCallback, 源代码网推荐 object asyncState); 源代码网推荐 源代码网推荐 protected abstract IEnumerable EndExecuteSelect( 源代码网推荐 IAsyncResult asyncResult); 源代码网推荐 源代码网推荐 protected override IEnumerable ExecuteSelect( 源代码网推荐 DataSourceSelectArguments arguments) { 源代码网推荐 //实现从 DataSourceView 中继承的 源代码网推荐 //抽象 ExecuteSelect 方法, 源代码网推荐 //方法是使用 BeginExecuteSelect 和 EndExecuteSelect, 源代码网推荐 //以便通过阻止来 源代码网推荐 //进行同步数据访问。 源代码网推荐 } 源代码网推荐 源代码网推荐 private IAsyncResult OnBeginSelect(object sender, 源代码网推荐 EventArgs e, AsyncCallback asyncCallback, 源代码网推荐 object extraData); 源代码网推荐 private void OnEndSelect(IAsyncResult asyncResult); 源代码网推荐 源代码网推荐 public override void Select(DataSourceSelectArguments arguments, 源代码网推荐 DataSourceViewSelectCallback callback) { 源代码网推荐 if (_owner.IsAsync) { 源代码网推荐 //使用 OnBeginSelect 和 OnEndSelect 源代码网推荐 //作为 BeginEventHandler 和 EndEventHandler 方法, 源代码网推荐 //来调用 Page.ReGISterAsyncTask, 源代码网推荐 //以指明需要 源代码网推荐 //进行异步工作。这些方法将依次 源代码网推荐 //调用特定的 源代码网推荐 //数据源实现,方法是调用 源代码网推荐 //已在此类中引入的 源代码网推荐 //抽象 BeginExecuteSelect 和 EndExecuteSelect 源代码网推荐 //方法。 源代码网推荐 } 源代码网推荐 else { 源代码网推荐 //执行同步数据访问 源代码网推荐 base.Select(arguments, callback); 源代码网推荐 } 源代码网推荐 } 源代码网推荐 ... 源代码网推荐 } 源代码网推荐 源代码网推荐 源代码网推荐 做人要厚道,请注明转自酷网动力(www.ASPCOOL.COM)。 源代码网推荐 源代码网供稿. |
