上一节我们讲到对EF(EntityFramework)的初步封装,任何事情都不可能一蹴而就,通过大量的实际项目的实战,也发现了其中的各种问题。在这一章中,我们对上一章的EF_Helper_DG进行优化,主要优化点如下:

1.由DB实体单例模式改为从缓存中获取;

2.加入服务器缓存,协助查询,提升查询性能;

3.优化CUD操作方法的执行方式;

下面直接展示新的EF_Helper_DG 代码:

 using LinqKit; //AsExpandable() in linqkit.dll
 using System;
 using System.Linq;
 using System.Linq.Expressions;
 using System.Data.Entity;
 using System.Transactions;
 using QX_Frame.Helper_DG.Configs;

 namespace QX_Frame.Helper_DG
 {
     /*  time:   2016-10-30 15:26:05
         author: qixiao
     */
     /// <summary>
     /// EntityFramework CodeFirst Helper
     /// </summary>
     /// <typeparam name="Db">DbContext</typeparam>
     public abstract class EF_Helper_DG<Db> where Db : DbContext
     {
         /*the singleton Db */
         //private volatile static Db db = null;   //volatile find Db in memory not in cache

         #region The Singleton to new DBEntity_DG

         //private static readonly object lockHelper = new object();
         //static EF_Helper_DG()
         //{
         //    if (db == null)
         //    {
         //        lock (lockHelper)
         //        {
         //            if (db == null)
         //                db = System.Activator.CreateInstance<Db>();
         //        }
         //    }

         //    //close the Validate of EF OnSaveEnabled
         //    db.Configuration.ValidateOnSaveEnabled = false;
         //}

         #endregion

         #region get current dbContext
         public static DbContext GetCurrentDbContext()
         {
             //method 1 : CallContext 该方法有有时候第一次访问不到的bug
             //CallContext:是线程内部唯一的独用的数据槽(一块内存空间)
             //Db dbContext = CallContext.GetData("DbContext") as Db;
             //if (dbContext == null)  //线程在内存中没有此上下文
             //{
             //    //create a dbContext to memory if dbContext has not exist
             //    dbContext = System.Activator.CreateInstance<Db>();
             //    CallContext.SetData("DbContext", dbContext);
             //}

             //method 2 :
             Db dbContext = Cache_Helper_DG.Cache_Get("dbContext") as Db;
             if (dbContext == null)
             {
                 //create a dbContext to memory if dbContext has not exist
                 dbContext = System.Activator.CreateInstance<Db>();
                 Cache_Helper_DG.Cache_Add("dbContext",dbContext);
             }
             return dbContext;
         }
         #endregion

         #region Cache Strategy

         /// <summary>
         /// edit data cache must update
         /// </summary>
         public static void CacheChanges<T>()
         {
             if (QX_Frame_Helper_DG_Config.Cache_IsCache)
             {
                 Cache_Helper_DG.Cache_Delete(nameof(T));
             }
         }

         /// <summary>
         /// query cache
         /// </summary>
         /// <typeparam name="T"></typeparam>
         /// <returns></returns>
         public static IQueryable<T> GetIQuerybleByCache<T>() where T : class
         {
             if (QX_Frame_Helper_DG_Config.Cache_IsCache)
             {
                 IQueryable<T> iqueryable = Cache_Helper_DG.Cache_Get(nameof(T)) as IQueryable<T>;
                 if (iqueryable == null)
                 {
                     DbContext db = GetCurrentDbContext();
                     iqueryable = db.Set<T>().AsExpandable();
                     Cache_Helper_DG.Cache_Add(nameof(T), iqueryable, null, DateTime.Now.AddMinutes(QX_Frame_Helper_DG_Config.Cache_CacheExpirationTime_Minutes), TimeSpan.Zero);
                 }
                 return iqueryable;
             }
             else
             {
                 DbContext db = GetCurrentDbContext();
                 return db.Set<T>().AsExpandable();
             }
         }

         #endregion

         #region Add 

         public static Boolean Add<T>(T entity) where T : class
         {
             DbContext db = GetCurrentDbContext();
             CacheChanges<T>();
             db.Entry<T>(entity).State = EntityState.Added;
             ;
         }
         public static Boolean Add<T>(T entity, out T outEntity) where T : class
         {
             DbContext db = GetCurrentDbContext();
             CacheChanges<T>();
             db.Entry<T>(entity).State = EntityState.Added;
             outEntity = entity;
             ;
         }
         public static Boolean Add<T>(IQueryable<T> entities) where T : class
         {
             DbContext db = GetCurrentDbContext();
             CacheChanges<T>();
             db.Set<T>().AddRange(entities);
             ;
         }

         #endregion

         #region Update

         public static Boolean Update<T>(T entity) where T : class
         {
             DbContext db = GetCurrentDbContext();
             CacheChanges<T>();
             if (db.Entry<T>(entity).State == EntityState.Detached)
             {
                 db.Set<T>().Attach(entity);
                 db.Entry<T>(entity).State = EntityState.Modified;
             }
             else
             {
                 db.SaveChanges();
                 return true;
             }
             ;
         }
         public static Boolean Update<T>(T entity, out T outEntity) where T : class
         {
             DbContext db = GetCurrentDbContext();
             CacheChanges<T>();
             outEntity = entity;
             if (db.Entry<T>(entity).State == EntityState.Detached)
             {
                 db.Set<T>().Attach(entity);
                 db.Entry<T>(entity).State = EntityState.Modified;
             }
             else
             {
                 db.SaveChanges();
                 return true;
             }
             ;
         }
         #endregion

         #region Delete

         public static Boolean Delete<T>(T entity) where T : class
         {
             DbContext db = GetCurrentDbContext();
             CacheChanges<T>();
             db.Set<T>().Attach(entity);
             db.Entry<T>(entity).State = EntityState.Deleted;
             ;
         }
         public static Boolean Delete<T>(IQueryable<T> entities) where T : class
         {
             DbContext db = GetCurrentDbContext();
             CacheChanges<T>();
             db.Set<T>().RemoveRange(entities);
             ;
         }
         public static Boolean Delete<T>(Expression<Func<T, bool>> deleteWhere) where T : class
         {
             DbContext db = GetCurrentDbContext();
             CacheChanges<T>();
             IQueryable<T> entitys = GetIQuerybleByCache<T>().Where(deleteWhere);
             /**
              * change code 2017-5-6 11:11:19 qixiao
              * entitys.ForEach(m => db.Entry<T>(m).State = EntityState.Deleted);
              **/
             entitys.ForEachAsync(m => db.Entry<T>(m).State = EntityState.Deleted);
             ;
         }
         #endregion

         #region Select 

         public static Boolean Exist<T>(Expression<Func<T, Boolean>> selectWhere) where T : class

         {
             return GetIQuerybleByCache<T>().Where(selectWhere).FirstOrDefault<T>() == null ? false : true;
         }
         public static T selectSingle<T>(Expression<Func<T, Boolean>> selectWhere) where T : class
         {
             return GetIQuerybleByCache<T>().Where(selectWhere).FirstOrDefault<T>();
         }
         public static IQueryable<T> selectAll<T>() where T : class
         {
             return GetIQuerybleByCache<T>();
         }
         public static IQueryable<T> selectAll<T>(out int Count) where T : class
         {
             Count = GetIQuerybleByCache<T>().Count();
             return GetIQuerybleByCache<T>();
         }
         public static IQueryable<T> selectAll<T, TKey>(Expression<Func<T, TKey>> orderBy, Boolean isDESC = false) where T : class
         {
             if (isDESC)
                 return GetIQuerybleByCache<T>().OrderByDescending(orderBy);
             else
                 return GetIQuerybleByCache<T>().OrderBy(orderBy);
         }
         public static IQueryable<T> selectAll<T, TKey>(Expression<Func<T, TKey>> orderBy, out int Count, Boolean isDESC = false) where T : class
         {
             Count = GetIQuerybleByCache<T>().Count();
             if (isDESC)
                 return GetIQuerybleByCache<T>().OrderByDescending(orderBy);
             else
                 return GetIQuerybleByCache<T>().OrderBy(orderBy);
         }
         public static IQueryable<T> selectAll<T>(Expression<Func<T, Boolean>> selectWhere) where T : class
         {
             return GetIQuerybleByCache<T>().Where(selectWhere);
         }
         public static IQueryable<T> selectAll<T>(Expression<Func<T, Boolean>> selectWhere, out int Count) where T : class
         {
             var IQueryable = GetIQuerybleByCache<T>().Where(selectWhere);
             Count = IQueryable.Count();
             return IQueryable;
         }
         public static IQueryable<T> selectAll<T, TKey>(Expression<Func<T, TKey>> orderBy, Expression<Func<T, Boolean>> selectWhere, Boolean isDESC = false) where T : class
         {
             if (isDESC)
                 return GetIQuerybleByCache<T>().Where(selectWhere).OrderByDescending(orderBy);
             else
                 return GetIQuerybleByCache<T>().Where(selectWhere).OrderBy(orderBy);
         }
         public static IQueryable<T> selectAll<T, TKey>(Expression<Func<T, TKey>> orderBy, Expression<Func<T, Boolean>> selectWhere, out int Count, Boolean isDESC = false) where T : class
         {
             var IQueryable = GetIQuerybleByCache<T>().Where(selectWhere);
             Count = IQueryable.Count();
             if (isDESC)
                 return IQueryable.OrderByDescending(orderBy);
             else
                 return IQueryable.OrderBy(orderBy);
         }

         public static IQueryable<T> selectAllPaging<T, TKey>(int pageIndex, int pageSize, Expression<Func<T, TKey>> orderBy, Boolean isDESC = false) where T : class
         {
             var IQueryable = GetIQuerybleByCache<T>();
             if (isDESC)
                  <  ?  : pageIndex - ) * (pageSize <  ?  : pageSize)).Take(pageSize <  ?  : pageSize);
             else
                  <  ?  : pageIndex - ) * (pageSize <  ?  : pageSize)).Take(pageSize <  ?  : pageSize);
         }
         public static IQueryable<T> selectAllPaging<T, TKey>(int pageIndex, int pageSize, Expression<Func<T, TKey>> orderBy, out int Count, Boolean isDESC = false) where T : class
         {
             var IQueryable = GetIQuerybleByCache<T>();
             Count = IQueryable.Count();
             if (isDESC)
                  <  ?  : pageIndex - ) * (pageSize <  ?  : pageSize)).Take(pageSize <  ?  : pageSize);
             else
                  <  ?  : pageIndex - ) * (pageSize <  ?  : pageSize)).Take(pageSize <  ?  : pageSize);
         }
         public static IQueryable<T> selectAllPaging<T, TKey>(int pageIndex, int pageSize, Expression<Func<T, TKey>> orderBy, Expression<Func<T, Boolean>> selectWhere, Boolean isDESC = false) where T : class
         {
             var IQueryable = GetIQuerybleByCache<T>().Where(selectWhere);
             if (isDESC)
                  <  ?  : pageIndex - ) * (pageSize <  ?  : pageSize)).Take(pageSize <  ?  : pageSize);
             else
                  <  ?  : pageIndex - ) * (pageSize <  ?  : pageSize)).Take(pageSize <  ?  : pageSize);
         }
         public static IQueryable<T> selectAllPaging<T, TKey>(int pageIndex, int pageSize, Expression<Func<T, TKey>> orderBy, Expression<Func<T, Boolean>> selectWhere, out int Count, Boolean isDESC = false) where T : class
         {
             var IQueryable = GetIQuerybleByCache<T>().Where(selectWhere);
             Count = IQueryable.Count();
             if (isDESC)
                  <  ?  : pageIndex - ) * (pageSize <  ?  : pageSize)).Take(pageSize <  ?  : pageSize);
             else
                  <  ?  : pageIndex - ) * (pageSize <  ?  : pageSize)).Take(pageSize <  ?  : pageSize);
         }

         #endregion

         #region Transaction
         public static void Transaction(Action action)
         {
             using (TransactionScope trans = new TransactionScope())
             {
                 action();
                 trans.Complete();
             }
         }
         #endregion

         #region ExecuteSqlCommand

         public static void ExecuteSqlCommand(string sqlCommand)
         {
             DbContext db = GetCurrentDbContext();
             db.Database.ExecuteSqlCommand(sqlCommand);
         }
         public static void ExecuteSqlCommand(string sqlCommand, params object[] parameters)
         {
             DbContext db = GetCurrentDbContext();
             db.Database.ExecuteSqlCommand(sqlCommand, parameters);
         }

         #endregion
     }
 }

上述代码为当前版本的EF_Helper_DG代码,当然欢迎所有的同行们优化补充哦~

ORM框架 EF - code first 的封装 优化一的更多相关文章

  1. ORM框架 EF - code first 的封装

    Code first 是Microsoft Entity Framework中的一种模式,CodeFirst不会有可视化的界面来进行拖动编辑DataBase-Entity,但会以一个类来进行对数据表关 ...

  2. ORM框架EF

    应用程序和数据库采用Tcp协议通讯 ORM框架有: NHibernate ,Dapper ,Mybatis 底层是 ADO.Net 好处: 1.面向对象 2.没有sql减少学习成本,快速开发 3.编译 ...

  3. MVC系列学习(二)-初步了解ORM框架-EF

    1.新建 一个控制台项目 2.添加一个数据项 a.选择数据库 注:数据库中的表如下: b.选择EF版本 c.选择表 3.初步了解EF框架 看到了多了一个以 edmx后缀的文件 在edmx文件上,右击打 ...

  4. ORM框架Hibernate (一) 对DAO封装和抽象

    说明 前面已经给大家介绍了Struts这个框架,Struts是对Web项目的表示层进行了封装,而Hibernate是对Web项目中DAO层进行封装,也即是.NET中我们常用到的D层封装,即对访问数据库 ...

  5. 抛弃EF,20分构建一个属于自己的ORM框架

    Poiuyt_cyc 博客园首页新随笔联系订阅管理随笔 - 11  文章 - 0  评论 - 111 抛弃EF,20分构建一个属于自己的ORM框架 相信EF大家都不陌生了,因为数据库表跟程序实体是一一 ...

  6. .Net 常用ORM框架对比:EF Core、FreeSql、SqlSuger

    前言: 最近由于工作需要,需要选用一种ORM框架,也因此对EF Core.FreeSql.SqlSuger作简单对比.个人认为各有有优势,存在即合理,不然早就被淘汰了是吧,所以如何选择因人而议.因项目 ...

  7. 【EF 4】ORM框架及其流行产品之一EF介绍

    导读:跳进了多租户切换数据库的坑,那么就继续走下去吧.在我们的项目中,是运用EF实现对数据库的操作,那么EF其实是.NET系统中,基于ORM框架的一个产品实现.在java那边,则有Hibernate和 ...

  8. webapi框架搭建-数据访问ef code first

    webapi框架搭建系列博客 为什么用ef? 我相信很多博友和我一样都有这种“选择困难症”,我曾经有,现在也有,这是技术人的一个通病——总想用“更完美”的方式去实现,导致在技术选择上犹豫不决,或总是推 ...

  9. EF、Dapper、NHibernate等ORM框架的比较及优缺点

    什么是ORM? ORM的全称是Object Relational Mapping,即对象关系映射.它的实现思想就是将关系数据库中表的数据映射成为对象,以对象的形式展现,这样开发人员就可以把对数据库的操 ...

随机推荐

  1. 用shell制作IP脚本

    vim  ip.sh #!/bin/bashread -p "eth:" eread -p "ip:" ip1read -p "netmask:&qu ...

  2. idea/eclipse下Maven工程集成web服务(tomcat、jetty)

     idea/eclipse下Maven工程集成web服务 转载请注明出处:http://www.cnblogs.com/funnyzpc/p/8093554.html 应用服务器最常用的一般有这哥仨: ...

  3. C(8)

    C语言位运算与文件 本章引言: 在不知不觉中我们的C高速入门系列已经慢慢地接近尾声了,而在这一节中,我们会对 C语言中的位运算和文件进行解析,相信这两章对于一些人来说是陌生的,由于非常多 老师都会跳过 ...

  4. php之str_replace具体解释

    str_replace (PHP 4, PHP 5) str_replace - Replace all occurrences of the search string with the repla ...

  5. SVN 的一些操作

    SVN是一个非常好的代码管理工具. 之前一直是使用现有的SVN环境,认为非常方便. 可是近期要自己搭建一个SVN环境来托管自己的代码,并且代码放在独立的server上,所以决定自己搭建一个SVN环境. ...

  6. Git(一)之基本操作详解

    前言 在这段时间的工作中一直都是用Git作为版本控制,只是知道简单的几个命令,没有去了解它的内部原理.所以周末有时间来系统学习. 现在的公司基本上都是用Git作为版本控制,当然也有SVN的. Git是 ...

  7. AJAX跨域问题解决---后台解决

    对于ajax请求数据,经常出现一个坑,防不胜防.今天突然找到一个很好的解决办法,直接在后台设置资源共享就可以了. 代码为:response.raw().setHeader("Access-C ...

  8. 前端MVC Vue2学习总结(一)——MVC与vue2概要、模板、数据绑定与综合示例

    一.前端MVC概要 1.1.库与框架的区别 框架是一个软件的半成品,在全局范围内给了大的约束.库是工具,在单点上给我们提供功能.框架是依赖库的.Vue是框架而jQuery则是库. 1.2.AMD与CM ...

  9. Jsp制作验证码

    验证码 验证码(CAPTCHA)是"Completely Automated Public Turing test to tell Computers and Humans Apart&qu ...

  10. String、StringBuilder和StringBuffer类

    */ .hljs { display: block; overflow-x: auto; padding: 0.5em; color: #333; background: #f8f8f8; } .hl ...