微型项目实践(9):页面的数据访问策略

简介:

上一篇中,我们完成了数据库的访问,今天我们来看看系统设计的最后一部分——UI层。加入了UI层之后,系统设计会变成这个样子:

system_design

这也就是系统最终的结构图。这个图上新添加的两个项目,一个是UI,一个是WebSite。其中前者依赖于业务逻辑和数据访问,提供统一的界面处理,而WebSite仅包含Aspx页面。需要注意的是,上图中箭头表示依赖或调用,而这个关系是具有传递性的,比如UI依赖于Business,而Business依赖于Common,则UI自然就依赖于Common。

将UI和WebSite分写在两个项目中,是我个人的一个习惯。其中UI主要包含页面基类和Helper类,还有数据库的获取方法,对于多种界面(比如既提供Web也提供WinForm)的系统,这一层次的抽象是必要的,而对于只提供Web访问的系统,把UI层放在WebSite中的App_Code文件夹下也无不可。

UI层中,目前就有两个核心类,一个是所有页面的基类,另一个是数据库生成类,其类结构图如下:

UI

其中DatabaseGateWay的作用就是读取Web.Config中数据库链接节点的连接字符串,并调用DataAccess层的方法,构造新的数据库:

   1:  namespace DongBlog.UI
   2:  {
   3:      /// <summary>
   4:      /// 数据库
   5:      /// </summary>
   6:      public class DatabaseGateWay
   7:      {
   8:          private const string DatabaseConnectionConfigurationName = 
"DongBlogDatabaseConnectionString";
   9:   
  10:          /// <summary>
  11:          /// 取得新的数据库
  12:          /// </summary>
  13:          /// <returns>新数据库</returns>
  14:          public static IDatabase GetNewDatabase()
  15:          {
  16:              return Database.New(ConfigurationManager
.ConnectionStrings[DatabaseGateWay.DatabaseConnectionConfigurationName]
.ConnectionString);
  17:          }
  18:      }
  19:  }

该类的GetNewDatabase()静态方法,是类型安全的,也是唯一使用到了DataAccess层具体实现的地方,系统的其它部分都只了解IDatabase而不了解其实现,更换不同的DataAccess只需要修改此方法。

PageBase类是所有Aspx页的基类,该类在该系统中只提供了数据库访问功能,即一个页面对应一个数据库访问,代码如下:

  11:  namespace DongBlog.UI
  12:  {
  13:      /// <summary>
  14:      /// 所有页面的基类
  15:      /// </summary>
  16:      public class PageBase : Page
  17:      {
  18:          private IDatabase _Database = DatabaseGateWay.GetNewDatabase();
  19:   
  20:          /// <summary>
  21:          /// 取得数据库访问
  22:          /// </summary>
  23:          protected IDatabase Database
  24:          {
  25:              get { return _Database; }
  26:          }
  27:      }
  28:  }

一个页面对应一个IDatabase,对于Linq来说,就是一个页面对应一个DataContext,这保证页面生存周期中的所有业务实体都来源或依附于同一个DataContext,避免了跨DataContext传递实体的问题。在Linq中,一个DataContext产生的Entity交由另一个DataContext中使用是一件非常麻烦的事情,必须保证实体必须使用Attach方法附加到新的DataContext上,如果不附加,则新的DataContext会认为该Entity是new出来的,这会导致再数据库中插入一条新的记录,而不是与现存记录建立关联,这个Bug很难调试,因为不会显示任何错误。我们举个例子说明这个问题。

我们经常会在用户登录的时候,在Session中储存登陆的用户:

User user = GetLoginUser(username, password);
if (user != null)
    Session["CurrentUser"] = user;
else
    Response.Redirect("Login.aspx");

这样以后我们就可以用Session["CurrentUser"]取得当前用户了。当用户新建一篇日志时,我们会使用类似下面的代码:

Blog blog = new Blog();
blog.Creator = Session["CurrentUser"] as User;
blog.Save(Database);

问题来了:上面两个代码通常来说不是发生在一个页面中,也就是说,从数据库中读取该user的DataContext早已不存在了,而新的、用于存储Blog的DataContext会认为这个blog相关的Creator是一个“新”的User,所以会在数据库中插入一条新的User记录,而不是和原来的User记录建立关联!为了避免这个情况,我们必须用Attach方法将Session["CurrentUser"]附加到新的DataContext中,上面的代码要这么改:

Database.Users.Attach(Session["CurrentUser"] as User);
Blog blog = new Blog();
blog.Creator = Session["CurrentUser"];
blog.Save(Database);

但是假设User还有关联的实体,比如Role、Power之类的东西,就需要全部附加上!否则就会插入新的关联记录!这几乎是不可能的,所以我们要保证每一个页面周期中所有的实体来源于同一个DataContext,对于上面这个例子,解决办法是不保存User,而保存UserID:

User user = GetLoginUser(username, password);
if (user != null)
    Session["CurrentUserID"] = user.ID;
else
    Response.Redirect("Login.aspx");

这样我们就可以在添加日志时这样写:

Blog blog = new Blog();
blog.Creator = Database.Users.GetByID(Convert.ToInt32(Session["CurrentUserID"]));
blog.Save(Database);

进一步,我们可以在PageBase中加载User,考虑在PageBase中存在以下代码:

private User _CurrentUser;
public User CurrentUser
{
    get
    {
        if(Session["CurrentUserID"] == null)
            return null;

        if(_CurrentUser == null)
            _CurrentUser = Database.Users.GetByID(Convert.ToInt32(Session["CurrentUserID"]));
        return _CurrentUser;
    }
}

这样我们就可以使用以下的代码建立日志了:

Blog blog = new Blog();
blog.Creator = CurrentUser;
blog.Save(Database);

关于PageBase的用处还有很多,包括但不限于以下这些:

  1. 构造Database。
  2. 构造当前User。
  3. 判断当前用户是否具有该页面的访问权限。
  4. 根据当前用户定制页面:加入不同的Css等。

另外,基于PageBase的扩展方法,将对所有页面有效,这点常常用于执行JavaScript。

下一篇中,我们开始编写我们第一个页面。

代码下载

本文转自冬冬博客园博客,原文链接:http://www.cnblogs.com/yuandong/archive/2008/05/17/1201508.html ,如需转载请自行联系原作者
相关文章
|
8月前
|
存储 算法 Android开发
提升安卓应用性能的五大实用策略
【4月更文挑战第5天】在快速发展的数字时代,用户对移动应用的性能要求越来越高。对于安卓开发者而言,优化应用性能不仅是提升用户体验的关键,也是增强应用竞争力的必要手段。本文将深入探讨五种实用的策略,帮助开发者有效提升安卓应用的性能。这些策略涵盖了从代码优化到资源管理等多个方面,旨在为开发者提供全面的指导和建议。通过实践这些策略,开发者可以显著减少应用的内存消耗、提高响应速度,并最终交付给用户一个更加流畅和高效的应用体验。
|
12天前
|
存储 消息中间件 前端开发
工厂人员定位管理系统架构设计:构建一个高效、可扩展的人员精确定位
本文将深入探讨工厂人员定位管理系统的架构设计,详细解析前端展示层、后端服务层、数据库设计、通信协议选择等关键环节,并探讨如何通过微服务架构实现系统的可扩展性和稳定性。
44 10
|
8月前
|
缓存 监控 Android开发
提升安卓应用性能的五大关键策略
【4月更文挑战第30天】 在竞争激烈的应用市场中,卓越的性能是确保用户留存和应用成功的核心因素。本文将详细阐述五种提高安卓应用性能的有效技术策略。这些策略包括优化内存使用、减少网络请求延迟、多线程与并发处理、UI渲染优化以及电池效率改进。通过深入分析每项技术的原理及其在实际开发中的应用,旨在帮助开发者构建更快速、流畅且响应敏捷的安卓应用。
|
5月前
|
UED 存储 数据管理
深度解析 Uno Platform 离线状态处理技巧:从网络检测到本地存储同步,全方位提升跨平台应用在无网环境下的用户体验与数据管理策略
【8月更文挑战第31天】处理离线状态下的用户体验是现代应用开发的关键。本文通过在线笔记应用案例,介绍如何使用 Uno Platform 优雅地应对离线状态。首先,利用 `NetworkInformation` 类检测网络状态;其次,使用 SQLite 实现离线存储;然后,在网络恢复时同步数据;最后,通过 UI 反馈提升用户体验。
133 0
|
8月前
|
算法 开发工具 Android开发
提升安卓应用性能的五大策略
【2月更文挑战第16天】在竞争激烈的应用市场中,性能优越的安卓应用更能吸引和保留用户。本文将深入探讨五种有效的策略,帮助开发者优化安卓应用性能,包括代码优化、内存管理、多线程应用、使用最新的安卓SDK以及利用硬件加速特性。
|
8月前
|
机器人 TensorFlow 算法框架/工具
量化交易机器人系统开发详细策略/需求步骤/逻辑方案/源码设计
auto nhwc_data = nhwc_Tensor->host<float>(); auto nhwc_size = nhwc_Tensor->size(); ::memcpy(nhwc_data, image.data, nhwc_size);
|
存储 监控 算法
解密上网行为管理:数据流分析算法的引入与优势
今天,我们将一起来谈谈数据流分析算法,这项看似高深莫测的技术是如何在上网行为管理中大放异彩的。首先,让我们来了解一下,什么是数据流分析算法?简而言之,这是一种用于处理大量数据的方法,它允许我们在数据流经过时实时监控、分析和提取有用信息。这一技术的应用领域之一就是上网行为管理。
193 1
|
数据库
易搭工作流引擎用是什么开源 还是阿里自研产品,零代码平台场景页面映射数据库表是动态创建,采用什么框架处理,怎么让系统产生高并发能力。易搭权限有没有了解,求解。
易搭工作流引擎用是什么开源 还是阿里自研产品,零代码平台场景页面映射数据库表是动态创建,采用什么框架处理,怎么让系统产生高并发能力。易搭权限有没有了解,求解。
|
存储 安全 数据安全/隐私保护
实时云渲染应用之虚拟仿真项目的四大优势
实时云渲染方案中,虚拟仿真程序存储在服务器上,程序执行时占用的是服务器的显卡和CPU资源,用户侧机器无特殊硬件要求,只要能正常的观看1080P的视频即可。因为云渲染方案中,虚拟仿真程序存在服务器上,执行过程也在服务器,因此产生的数据也是在服务器上,数据泄露风险更低,安全系数更高。从用户侧的角度来说,点量实时云渲染方案相比于传统的方式,学生无需安装任何程序和客户端,学生通过网页,就可以打开服务器上的虚拟仿真程序进行操作学习。效果和本地安装几乎无任何差别,摆脱了对于电脑硬件参数的高性能要求,可以做到即点即用。
84 0