艾伟_转载:数据库设计与Linq增强使用

最近对数据库的设计有些想法,貌似一般数据都有些通用字段

    public interface IData
    {
        /// 
        /// 数据ID标识
        /// 
        decimal ID { get; set; }
        /// 
        /// 更新时间
        /// 
        DateTime UpdateTime { get; set; }
        /// 
        /// 数据状态
        /// 
        int State { get; set; }
        /// 
        /// 创建时间
        /// 
        DateTime CreateTime { get; set; }
    }

其中ID是自增长主键(SQL,Oracle环境可以是Sequence生成的ID)

UpdateTime是最后一次更新时间

CreateTime是创建时间

State是数据状态(本来的设想里没有,看了这个文章觉得状态字段实在太需要了。。)

类型如下:

 

数据库就这样了。。有什么用呢~?继续看。。

在这个的基础上,可以抽象出一个 IData 接口:

IData 接口

    public interface IData
    {
        /// 
        /// 数据ID标识
        /// 
        decimal ID { get; set; }
        /// 
        /// 更新时间
        /// 
        DateTime UpdateTime { get; set; }
        /// 
        /// 数据状态
        /// 
        int State { get; set; }
        /// 
        /// 创建时间
        /// 
        DateTime CreateTime { get; set; }
    }

 

因为数据库的表字段跟IData属性成员是一致的,直接对Linq生成的实体类进行接口签名:

IData 接口签名

签名放在

一般没人用的Linq设计器cs文件里面了。。好处是方便跟着那个大坨的Linq文件走。。

 

这个文件右键dbml文件 -> 查看代码就出现了,默认是不出现的(因为很惹人讨厌,之前不小心弄出来了我都会把它删掉。。现在用上了。。)

当然了,上面的借口签名不是一个一个写出来的,直接CodeSmith很简单的就出来了 

接下来,针对IData进行扩展:

数据对象扩展
    /// 
    /// 数据对象扩展
    /// 
    public static class DataExtension
    {
        /// 
        /// 保存或更新
        /// 
        /// 要保存或更新的数据
        /// 操作结果
        public static bool SaveOrUpdate(this IData data)
        {
            bool success = false;
            CIIDesignerDataContext db = new CIIDesignerDataContext();
            // todo: save or update 
            if (data.ID < 1)
            {
                // todo: save
                data.CreateTime = DateTime.Now;
                data.UpdateTime = DateTime.Now;
                db.GetTable(data.GetType()).InsertOnSubmit(data);
                
            }
            else
            {
                // todo: update
                data.UpdateTime = DateTime.Now;
                db.GetTable(data.GetType()).Attach(data);
                db.SubmitChanges();
            }

            try
            {
                db.SubmitChanges();
                success = true;
            }
            catch
            {
                success = false;
            }
            return success;
        }

        /// 
        /// 删除数据
        /// 
        /// 要删除的数据
        /// 操作结果
        public static bool Delete(this IData data)
        {
            bool success = false;
            CIIDesignerDataContext db = new CIIDesignerDataContext();
            db.GetTable(data.GetType()).Attach(data);
            db.GetTable(data.GetType()).DeleteOnSubmit(data);
            try
            {
                db.SubmitChanges();
                success = true;
            }
            catch (Exception)
            {
                success = false;
            }
            return success;
        }
    }

这样下来每个实体类可以直接增删改:(不知道这样用处大不大。。)

 

好了,现在我们的Linq实体类可以直接增删改,不用关心Linq的DataContext了,充血充的更厉害了……

但是查询的时候还是没办法彻底摆脱Linq的DataContext,再来~

不知道命名,就赶新潮也叫Repository(其实还是DataContext的范畴)

数据存储池

    /// 
    /// 数据存储池
    /// 
    /// 
    public class Repository<T> where T : class, IData
    {
        CIIDesignerDataContext db = new CIIDesignerDataContext();

        /// 
        /// 创建数据对象
        /// 
        /// 
        public T Create()
        {
            return (T)Activator.CreateInstance(typeof(T));            
        }

        /// 
        /// 根据主键获取数据
        /// 
        /// 主键ID
        /// 
        public T FindByID(decimal ID)
        {
            return db.GetTable<T>().FirstOrDefault(c => c.ID == ID);
        }

        /// 
        /// 获取数据Query
        /// 
        /// 
        public IQueryable<T> GetQuery()
        {
            return db.GetTable<T>().AsQueryable();
        }

        /// 
        /// 查询数据
        /// 
        /// 
        public IEnumerable<T> Query(Func<T,bool> Expression)
        {
            return db.GetTable<T>().Where(Expression);
        }

        /// 
        /// 查询数据
        /// 
        /// 动态Linq查询
        /// 
        public IEnumerable<T> GetQuery(string dLinq)
        {
            // todo: DynamicLinq 查询
            return null;
        }
        
    }

 

这样查询的功能全部被Repository接管了~(实现动态linq的Query方法后,可以直接用字符串查询)

基本上我们彻底离开DataContext了,用起来大概是这样:

TestClass

当然,对Repository还有还有很多事情可以继续做,让它成为一个真正的Repository

======

以上的做法,具体用处大不大还不知道

不过,不涉及到复杂跨表查询,是可以脱离Linq的DataContext了

而且使用中处理的实体类都是实现了IData接口的数据对象,辅以扩展方法,代码通用性得到很大提高

个人觉得还蛮有意义……

时间: 2024-08-16 05:23:39

艾伟_转载:数据库设计与Linq增强使用的相关文章

数据库设计与Linq增强使用

最近对数据库的设计有些想法,貌似一般数据都有些通用字段 public interface IData    {        ///         /// 数据ID标识        ///         decimal ID { get; set; }        ///         /// 更新时间        ///         DateTime UpdateTime { get; set; }        ///         /// 数据状态        /// 

艾伟_转载:基于.NET平台的Windows编程实战(二)—— 需求分析与数据库设计

本系列文章导航 基于.NET平台的Windows编程实战(一)--前言 基于.NET平台的Windows编程实战(二)-- 需求分析与数据库设计 基于.NET平台的Windows编程实战(四)-- 数据库操作类的编写 基于.NET平台的Windows编程实战(五)-- 问卷管理功能的实现 基于.NET平台的Windows编程实战(六)-- 题目管理功能的实现 大家都知道一个系统的成败与否关键在于其所做的需求分析是否到位,数据库的设计是否合理.因为本系列文章的目的是在于提高大家对.NET Wind

艾伟_转载:排行榜数据库设计与分析——为什么实时排行不可行?

很多网游中都有排行榜,这里就专门讨论一下这个排行榜背后的数据库设计.一开始我觉得这是一个基本的数据库设计问题.只需要有一个实体,没有实体间的关系,没有复杂的逻辑.网络上也搜索不到太多关于这类设计的问题,好像根本不值得为其写个文章.但是在公司专门做了一个月的排行榜数据库设计.才发现问题根本没有看上去那么简单.甚至一篇文章都难以讲明白.不知自己误入歧途了,还是这个问题的确就是很复杂的.所以写个文章讲给大家,或许能有人一语道破. 一开始听到要设计一个排行榜,觉得很简单,一个外键加一个分数列,排名不保存

艾伟_转载:基于.NET平台的Windows编程实战(四)—— 数据库操作类的编写

本系列文章导航 基于.NET平台的Windows编程实战(一)--前言 基于.NET平台的Windows编程实战(二)-- 需求分析与数据库设计 基于.NET平台的Windows编程实战(四)-- 数据库操作类的编写 基于.NET平台的Windows编程实战(五)-- 问卷管理功能的实现 基于.NET平台的Windows编程实战(六)-- 题目管理功能的实现 大家都知道本系统的正常运行少不了数据库操作这一块,且其在本系统中具有决定性作用,可以说没有它的操作系统将无法运行,故在本节课程中,专门把针

艾伟_转载:基于.NET平台的Windows编程实战(五)—— 问卷管理功能的实现

本系列文章导航 基于.NET平台的Windows编程实战(一)--前言 基于.NET平台的Windows编程实战(二)-- 需求分析与数据库设计 基于.NET平台的Windows编程实战(四)-- 数据库操作类的编写 基于.NET平台的Windows编程实战(五)-- 问卷管理功能的实现 基于.NET平台的Windows编程实战(六)-- 题目管理功能的实现 首先,为了使我们的界面更加便于操作及布局,我们引入第三开源控件DockPanel[当然也可以不引入控件而直接进行开发],你可以从这里:Do

艾伟_转载:基于.NET平台的Windows编程实战(六)—— 题目管理功能的实现

本系列文章导航 基于.NET平台的Windows编程实战(一)--前言 基于.NET平台的Windows编程实战(二)-- 需求分析与数据库设计 基于.NET平台的Windows编程实战(四)-- 数据库操作类的编写 基于.NET平台的Windows编程实战(五)-- 问卷管理功能的实现 基于.NET平台的Windows编程实战(六)-- 题目管理功能的实现 申明:本系列课程是专为新手们写来入门练习用的,目的是想通过一个完整的问卷调查管理系统的案例开发来让新手们了解.加深或是熟悉软件项目的开发流

艾伟_转载:基于.NET平台的Windows编程实战(一)——前言

本系列文章导航 基于.NET平台的Windows编程实战(一)--前言 基于.NET平台的Windows编程实战(二)-- 需求分析与数据库设计 基于.NET平台的Windows编程实战(四)-- 数据库操作类的编写 基于.NET平台的Windows编程实战(五)-- 问卷管理功能的实现 基于.NET平台的Windows编程实战(六)-- 题目管理功能的实现 前言:本系列文章是一个关于.NET Windows编程的入门实战教程.通过一个完整的真实案例:问卷调查管理系统,全面展示了基于.NET平台

艾伟_转载:使用LINQ to SQL更新数据库(上):问题重重

在学习LINQ时,我几乎被一个困难所击倒,这就是你从标题中看到的更新数据库的操作.下面我就一步步带你走入这泥潭,请准备好砖头和口水,Follow me. 从最简单的情况入手 我们以Northwind数据库为例,当需要修改一个产品的ProductName时,可以在客户端直接写下这样的代码: // List 0NorthwindDataContext db = new NorthwindDataContext(); Product product = db.Products.Single(p =>

艾伟_转载:使用LINQ to SQL更新数据库(中):几种解决方案

在前一篇文章中,我提出了在使用LINQ to SQL进行更新操作时可能会遇到的几种问题.其实这并不是我一个人遇到的问题,当我在互联网上寻找答案时,我发现很多人都对这个话题发表过类似文章.但另我无法满足的是,他们尽管提出了问题,却没有进行详细的剖析,只给出了解决方案(如添加RowVersion列.去除关联等),但却没有说明为什么必须这么做.这也是我写上篇的初衷,希望通过对LINQ to SQL源代码的分析,来一步一步找出解决问题的办法.本文将对这些方法一一进行讨论. 方案一:重新赋值 在Terry