public class RepositoryBase<T> : IRepository<T> where T : class

  RepositoryBase 是IRepository的一个 实例,泛型的作用是为特定的模型层提供标准的增删查改操作。

protected XltAppDataContext DataContext;

  一个继承了DbContext的类(数据库上下文)↓↓↓↓↓↓↓↓↓

public class XltAppDataContext : DbContext

  新增一个实体

        public object Add(T entity)
{
DataContext.Set<T>().Add(entity);
return entity;
}

  创建一个实体

        public T Create()
{
return DataContext.Set<T>().Create();
}

  Create 和 Add  有什么区别呢,以下摘自MSDN?

  Create:Creates a new instance of an entity for the type of this set. Note that this instance is NOT added or attached to the set. The instance returned will be a proxy if the underlying context is configured to create proxies and the entity type meets the requirements for creating a proxy.

  Add:Adds the given entity to the context underlying the set in the Added state such that it will be inserted into the database when SaveChanges is called.

  今天我们来 最后分析一下这个基础 方法:

  

public T CloneWithDynamicProxiesType(T modelObject)

  这个方法说的是“克隆,连带着代理类型”,下面我们来 分析一下这个方法的语句,今天不做深入分析。

        public T CloneWithDynamicProxiesType(T modelObject)
{
T dataModelProxy = DataContext.Set<T>().Create();
Type dynamicType = typeof(T).GetType();
var properties = dataModelProxy.GetType().GetProperties();
//TODO: consider a better way to filter out all 'virual' properties of the model class
foreach (var property in properties.Where(p => !p.PropertyType.Namespace.StartsWith("HP.XLT")))
{
try
{
//TODO: consider a better way to filter out all 'virual' properties of the model class
var genArgType = property.PropertyType.GetGenericArguments().FirstOrDefault();
if (genArgType== null || !genArgType.Namespace.StartsWith("HP.XLT"))
{
var value = property.GetValue(modelObject, null); if (value == null)
{
continue;
}
property.SetValue(dataModelProxy, value);
}
}
catch (Exception)
{
throw;
}
}
return dataModelProxy;
}

分析如下:

  下面这句话的意思应该是从DataContext中的固定类型中创建 一个实例,但是 这个实例是和具体的数据库无关的,也就是没有Attach or Set 到具体的表。

T dataModelProxy = DataContext.Set<T>().Create();

  这个就是从泛型类中获取动态类型。

Type dynamicType = typeof(T).GetType();

 下面是有关介绍:

  • typeof takes a type name (which you specify at compile time).
  • GetType gets the runtime type of an instance.

  相当于先得到这个类型的名字,然后再得到类型的实例,一个是编译时,一个是 运行时。

  

  下面的一句话,相当于利用反射从Type中获取Properties(属性)。

var properties = dataModelProxy.GetType().GetProperties();

  下面 是一个foreach循环,它的意思是首先过滤 掉命名 空间 不以HP.XLT开头的Property

foreach (var property in properties.Where(p => !p.PropertyType.Namespace.StartsWith("HP.XLT")))

  具体的关于GetProperties的介绍请参考Stackoverflow中的这个问题

今天 写到这里 ,明天继续 ------于2015/12/1日记录。

  又一天开始了,哈哈,今天继续。。。。

  我们 注意到了上面的GetProperties其实是得到了这个东西,也就是PropertyInfo:

public PropertyInfo[] GetProperties();

  下面的语句:

var genArgType = property.PropertyType.GetGenericArguments().FirstOrDefault();

  其实 本质上是得到一个Type,而这个Type是“每个”Property的Type,而GetGenericArguments()这个方法 是 得到了Type的一个数组,形如Type [].我们 可以通过如下的方法来使用,这是我写的:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace ConsoleApplicationCSharp
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine("-------NonGenericType---------");
IsGenericType(typeof(NonGenericType));
Console.WriteLine("----------GenericType---------");
IsGenericType(typeof(GenericType<>));
Console.ReadLine();
} public static void IsGenericType(Type t)
{
if (t.IsGenericType)
{
// If this is a generic type, display the type arguments.
//
Type[] typeArguments = t.GetGenericArguments(); Console.WriteLine("\tList type arguments ({0}):",
typeArguments.Length); foreach (Type tParam in typeArguments)
{
// If this is a type parameter, display its
// position.
//
if (tParam.IsGenericParameter)
{
Console.WriteLine("\t\t{0}\t(unassigned - parameter position {1})",
tParam,
tParam.GenericParameterPosition);
}
else
{
Console.WriteLine("\t\t{0}", tParam);
}
}
}
}
} public class NonGenericType
{
public string Name { get; set; }
public int Age { get; set; }
} public class GenericType<T> where T : class
{
public string Name { get; set; }
public int Age { get; set; } public void Add(T t)
{ } } }

  结果如下,我们 可以 看出只有使用了泛型类的,才会进方法,具体的参考上面的代码:

  下面 这句话 的意思 是,如果 没有参数或者参数的所属的类的命名空间不以XXX开头 的 时候,那么就,其实这个项目 所有的命名 空间都是有HP.XLT的,所以这句话的意思就是“找不到”相对应的参数。

 if (genArgType== null || !genArgType.Namespace.StartsWith("HP.XLT"))

  下面一句话,我觉得还是MSDN上面说得最好

  Returns the property value of a specified object with optional index values for indexed properties.

  意思就是返回特定的属性值---对于特定的对象:

var value = property.GetValue(modelObject, null);

  下面展示了如果VALUE不为空,那么就把这个值赋给dataModelProxy,而这个dataModelProxy也是我们最终要 返回的实例,因为这里是一个基础方法,所以返回T。

 if (value == null)
{
continue;
}
property.SetValue(dataModelProxy, value);

  下面的代码,循环添加:  

        public virtual IEnumerable<T> Add(IEnumerable<T> entities)
{
foreach (var entity in entities)
{
Add(entity);
} return entities;
}

  其中上面的方法 用到了单个 添加 ,代码 如下 :

        public object Add(T entity)
{
DataContext.Set<T>().Add(entity);
return entity;
}

  有些人可能对Add方法为什么要返回object有疑问,不能返回void吗?这里我说说我的理解 ,有2个 方面:

  • 已过期 :框架要考虑的是全部,如果你 返回的是一个VOID,那么如果想用这个实体,那么必然要重新去得到这个实体,这样浪费了时间和空间,所以返回VOID在 框架层面上是不好的。
  • 更新:返回VOID比 返回 OBJEECT要好,原因 很简单 ,没用到过。
  • 另一个问题是返回object好还是T好 ,可以明确的说,返回T好,因为T是一个泛型类,指向的是一个具体的类型,而如果 返回object,我还要进行强制类型转换才能得到 具体的实体。

  剩下的类似的方法如下:

  

        public T Create()
{
return DataContext.Set<T>().Create();
} public T Get(object id)
{
return DataContext.Set<T>().Find(id);
}

    

  今天 写到这里 ,明天继续 ------于2015/12/2日记录。

  下面 的一个方法是 从DbContext中过滤数据

        public IEnumerable<T> FilterContext(System.Linq.Expressions.Expression<Func<T, bool>> predicate)
{
return DataContext.Set<T>().Local.Where(predicate.Compile());
}

  上面的代码 有几个知识点:

  1.Express<Func<T,bool>> 的用法

  2.Local永远不会去访问 数据库里的东西 ,它的目的是 把东西放在本地 缓存,这句话永远 用在查询 语句已经被 执行以后。

  下面的方式 是关于"Filter"的一个最重要的方法

        public IQueryable<T> Filter(System.Linq.Expressions.Expression<Func<T, bool>> expression, string includePath)
{
DbQuery<T> result = null; if (!string.IsNullOrEmpty(includePath))
{
var includePathArray = includePath.Split(';'); foreach (var singlePath in includePathArray)
{
if (!string.IsNullOrEmpty(singlePath))
{
result = result == null ? DataContext.Set<T>().Include(singlePath) : result.Include(singlePath.Trim());
}
}
}
else
{
result = DataContext.Set<T>();
} return expression == null ? result.AsQueryable() : result.Where(expression).AsQueryable();
}

  它是通过lambda表达式去进行过滤,DbQuery是一个 非泛型的Linq  to Entity实例对于DbContext来说。

DbQuery<T> result = null;

  上面 这句话 先 判断IncludePath是否为空,至于Include是 干啥的呢?Include当中 包含了“属性的名称”,表示包含了此属性的所有的结果集。比如

public ClassA Ainstance{get;set;} //这里面,属性的名称就是Ainstance

  注意ClassA是这里的一个 导航属性,也就是不能为基本数据类型,比如int,string什么的。

  下面 最后来 总结 一下 这个方法:

  下面再来看看一个特殊的方法:

        public void Update(T entity)
{
DataContext.Entry<T>(entity).State = EntityState.Modified;
}

  这里的Entry,State是表示什么意思呢?下面的图可以揭开迷雾:

  其实不同的State对应了不同的状态,可以有不同的CRUD状态,上半部分的代码 和 下半部分的代码 其实 是 一个意思。

  下面介绍最后一个方法:

        public void Detach(T entity)
{
try //Catch 'invalid operation exception' - most likely cause: object is not attached
{
((IObjectContextAdapter)DataContext).ObjectContext.Detach(entity);
}
catch (InvalidOperationException)
{ }
}

  这么 写的目的是:不能有2个相同的KEY值在ObjectStateManager中,所以 需要把旧的实体从ORM中分离出去,这样才可以更新数据。

  大家 可以按转到定义按钮,去看看DbContext,得到如下结果,这证明了Dbcontext可以和IObjectContextAdapter进行强制 类型转换:

  MSDN上对于 ObjectContext.Detach的介绍相对 来说 比较 简单:从objectContext中移除相应的object.

  好了,这个文件讲解完毕----于2015-12-3日

RepositoryBase文件解析的更多相关文章

  1. CocosStudio文件解析工具CsdAnalysis

    起因 因为工作需要,所以需要使用CocosStudio来制作界面动画什么的.做完了发现需要找里边对象的时候会有很长一串代码,感觉不是很爽.之前写OC代码的时候可以吧程序中的对象指针跟编辑器中的对象相对 ...

  2. 通过正则表达式实现简单xml文件解析

    这是我通过正则表达式实现的xml文件解析工具,有些XHTML文件中包含特殊符号,暂时还无法正常使用. 设计思路:常见的xml文件都是单根树结构,工具的目的是通过递归的方式将整个文档树装载进一个Node ...

  3. 八、Android学习第七天——XML文件解析方法(转)

    (转自:http://wenku.baidu.com/view/af39b3164431b90d6c85c72f.html) 八.Android学习第七天——XML文件解析方法 XML文件:exten ...

  4. phpcms V9 首页模板文件解析

    在了解了<phpcms V9 URL访问解析>之后,我们已经知道首页最终执行的是content模块下index控制器的init方法. 下面, 我们逐步分析过程如下: 第一.首页默认执行的是 ...

  5. (转)AVI文件格式解析+AVI文件解析工具

    AVI文件解析工具下载地址:http://download.csdn.net/detail/zjq634359531/7556659 AVI(Audio Video Interleaved的缩写)是一 ...

  6. itextSharp 附pdf文件解析

    一.PdfObject: pdf对象 ,有9种,对象是按照对象内涵来分的,如果按照对象的使用规则来说,对象又分为间接对象和直接对象.间接对象是PDF中最常用的对象,如前面对象集合里面的,所有对象都是间 ...

  7. 《热血传奇2》wix、wil文件解析Java实现

    在百度上搜索java+wil只有iteye上一篇有丁点儿内容,不过他说的是错的!或者说是不完整的,我个人认为我对于热血传奇客户端解析还是有一定研究的,请移步: <JMir——Java版热血传奇2 ...

  8. paper 37 : WINCE的BIB文件解析

    WINCE的BIB文件解析 BIB的全称为Binary Image Builder,在Wince编译过程中的最后MakeImage阶段会用到BIB文件,BIB文件的作用是指示构建系统如何构建二进制映像 ...

  9. 如何让你的Apache支持include文件解析和支持shtml的相关配置

    源地址:http://www.itokit.com/2011/0430/65992.html Apache支持include文件解析shtml首先要应该修改Apache配置文件httpd.conf . ...

随机推荐

  1. dotNET跨平台相关文档整理

    一直在从事C#开发的相关技术工作,从C# 1.0一路用到现在的C# 6.0, 通常情况下被局限于Windows平台,Mono项目把我们C#程序带到了Windows之外的平台,在工作之余花了很多时间在M ...

  2. 通俗易懂的来讲讲DOM

    DOM是所有前端开发每天打交道的东西,但是随着jQuery等库的出现,大大简化了DOM操作,导致大家慢慢的“遗忘”了它的本来面貌.不过,要想深入学习前端知识,对DOM的了解是不可或缺的,所以本文力图系 ...

  3. javascript之Object.defineProperty的奥妙

    直切主题 今天遇到一个这样的功能: 写一个函数,该函数传递两个参数,第一个参数为返回对象的总数据量,第二个参数为初始化对象的数据.如: var o = obj (4, {name: 'xu', age ...

  4. 算法与数据结构(十五) 归并排序(Swift 3.0版)

    上篇博客我们主要聊了堆排序的相关内容,本篇博客,我们就来聊一下归并排序的相关内容.归并排序主要用了分治法的思想,在归并排序中,将我们需要排序的数组进行拆分,将其拆分的足够小.当拆分的数组中只有一个元素 ...

  5. 总结30个CSS3选择器

    或许大家平时总是在用的选择器都是:#id  .class  以及标签选择器.可是这些还远远不够,为了在开发中更加得心应手,本文总结了30个CSS3选择器,希望对大家有所帮助. 1 *:通用选择器 ;; ...

  6. BPM合同管理解决方案分享

    一.方案概述合同是组织与组织间所订协议的法律 表现形式,体现着双方对于合作在法律和道德上的承诺.然而,大多数企业的合同管理都或多或少存在合同审批过程不规范.签订草率.审批权责不清.合同执行跟踪难.合同 ...

  7. Toast显示图文界面——Android开发之路1

    Toast的多种使用方法 Toast其实是一个功能特别强大的组件,不仅仅可以吐司一个文本内容,还可以吐司图片以及图文混排的界面.具体用法如下: 第一种:简单的纯文本内容的吐司: Toast.makeT ...

  8. 烂泥:VMWare Workation双网卡配置IP地址

    本文由ilanniweb提供友情赞助,首发于烂泥行天下 想要获得更多的文章,可以关注我的微信ilanniweb 前几天给一个客户做远程项目实施,客户那边的服务器是Windows OS的,我们这边的业务 ...

  9. [DS] 标记字段

    标记字段 代码中有时候有这种需求:需要一个公共访问的标记字段,以下称为标记字段. 下面是案例: 一个订单详情页面,如果页面在显示中,程序中其它地方需要访问这个"正在查看中"的订单信 ...

  10. Hyper-V上运行的Linux虚拟机验证是否安装了集成服务

    Hyper-V上运行的Linux虚拟机验证是否安装了集成服务 ps aux|grep "hv"root       311  0.0  0.0      0     0 ?     ...