wojilu,wojilu系统的ORM代码解析-[源代码结构分析,ObjectBase基类分析]

wojilu系统有着完整的ORM系统,对于ORM系统,园子里面应该有很多很多高手了,我以前对于ORM系统也没有什么研究,借着学习wojilu系统,顺便学习了一下wojilu的ORM代码,已经一个ORM的实现过程。
首先来看一下wojilu系统的ORM部分的代码的位置:wojilu\ORM
ListORMSourcewojilu系统的ORM代码解析-[源代码结构分析,ObjectBase基类分析]wojilu

在学习wojilu的代码之前,我们先来想象一下他的代码大概是什么样子的。研究别人的代码之前,一定要自己先做一个预测,就像考试一样,自己先把题目做一遍,然后看答案,这样才能学习到东西,如果光看答案,自己失去了独立思考的机会,可能会被一个错误的答案误导。
ORM系统,就是将Object通过Mapping的方法放入RationShip的数据库中。将对象通过映射放入关系数据库中。他需要的东西有
1.一套对象基本类型到数据库类型的映射规则,例如将长整形int16,映射为数据库的长整形,将DateTime映射到DateTimeStamp,将String映射到varChar等等。
2.一套反射机制,通过反射可以遍历对象的内部所有公开的属性。
3.一套数据库自动访问系统,有一些基础的数据库访问方法,让开发者感觉不到在访问数据库。为了实现这个功能,必然有一个自动的Sql文生成系统。
大概就是这些东西了,这里是我的答案,接下来,我们看看wojilu的答案。当然,wojilu的答案并非标准答案,ORM系统,没有标准答案,也不需要标准答案,能够符合你的项目的答案就是好的答案。
在很多ORM系统里面,都有一个ObjectBase的类,这个类是万物之源,ORM都是围绕着这个万物之源展开的,只有将一个对象继承自ObjectBase,才能够享受到ORM的便利。换句话说,继承自ObjectBase的东西,都被纳入到了ORM的托管范围之内了。wojilu也有一个万物之源的类,这个类的源代码在wojilu\_wojilu\ObjectBase.cs
下面截取一部分代码:
1 namespace wojilu { 2 3 /// 4 /// 所有ORM中的领域模型都需要继承的基类 5 /// 6 /// 7 [Serializable] 8 public class ObjectBase : IEntity, IComparable where T : ObjectBase { 9 10 private int _id; 11 12 /// 13 /// 对象的 id 14 /// 15 public int Id { 16 get { return _id; } 17 set { this.setId( value ); _id = value; } 18 } 19 20 protected virtual void setId( int id ) { 21 } 22 23 /// 24 /// 查询所有数据 25 /// 26 /// 27 public static List findAll() { return db.findAll(); } 28 29 /// 30 /// 根据 id 查询对象 31 /// 32 ///
33 /// 34 public static T findById( int id ) { return db.findById( id ); } 35 36 /// 37 /// 统计所有的数据量 38 /// 39 /// 40 public static int count() { return db.count(); }
这里定义了ORM使用的多个方法的接口,也是整个ORM的基础,我们可以从分析这个类来分析整个wojiluORM系统。
这个类需要实现IEntity, IComparable两个接口。
第一个接口IEntity,实体接口:看看它的代码
每一个Orm对象,都具有一个ID属性,这个属性,如果关注wojilu系统的话,这个ID是缓存系统的关键字,同样ID的内容,都是从缓存中获取的。ID作为是否缓存命中的一个依据。
另外还有一个Get和SET的方法,获取或者设定某个属性的值。(并非通过反射,速度较快) 这句话的意思,暂时还不是很清楚,可能通过进一步的学习能够理解吧。
1 /// 2 /// 可以被 ORM 持久化的对象,都自动实现了本接口 3 /// 4 public interface IEntity { 5 6 /// 7 /// 每一个持久化对象,都具有一个 Id 属性 8 /// 9 int Id { get; set; } 10 11 /// 12 /// 获取属性的值(并非通过反射,速度较快) 13 /// 14 /// 属性名称
15 /// 16 Object get( String propertyName ); 17 18 /// 19 /// 设置属性的值(并非通过反射,速度较快) 20 /// 21 /// 属性名称
22 /// 属性的值
23 void set( String propertyName, Object propertyValue ); 24 25 /// 26 /// 包括对象的元数据,以及在对象查询的时候需要的额外信息,不常用 27 /// 28 //ObjectInfo state { get; set; } 29 } 30 31 32 33 }
具体的实现方法,源代码里面交代的很仔细了,截取部分代码:
1 /// 2 /// 根据属性名称获取属性的值 3 /// 4 /// 属性名称
5 /// 6 public Object get( String propertyName ) { 7 EntityInfo ei = getEntityInfo(); 8 if (propertyName.IndexOf( "." ) < 0) { 9 EntityPropertyInfo ep = ei.GetProperty( propertyName ); 10 if (ep == null) throw new Exception( String.Format( "property '{1}' of {0} is empty", ei.FullName, propertyName ) ); 11 return ep.GetValue( this ); 12 } 13 String[] arrItems = propertyName.Split( new char[] { '.' } ); 14 Object result = null; 15 ObjectBase obj = this; 16 for (int i = 0; i < arrItems.Length; i++) { 17 if (i < (arrItems.Length - 1)) { 18 obj = (ObjectBase)obj.get( arrItems[i] ); 19 } 20 else { 21 result = obj.get( arrItems[i] ); 22 } 23 } 24 return result; 25 } 26 27 /// 28 /// 设置属性的值 29 /// 30 /// 属性名称
31 /// 属性的值
32 public void set( String propertyName, Object propertyValue ) { 33 getEntityInfo().GetProperty( propertyName ).SetValue( this, propertyValue ); 34 }
IComparable接口应该是系统自带的接口,这个接口一般是用来自动排序的。NET框架中已经有自带的排序的方法了,但是两个对象,如何比较大小,以什么规则来比较大小,是通过这个IComparable对象的代码来实现的。
(MSDN:定义一种特定于类型的通用比较方法,值类型或类通过实现此方法对其实例进行排序。)
名称 说明
公共方法ListORMSourcewojilu系统的ORM代码解析-[源代码结构分析,ObjectBase基类分析]wojilu CompareTo 将当前实例与同一类型的另一个对象进行比较,并返回一个整数,该整数指示当前实例在排序顺序中的位置是位于另一个对象之前、之后还是与其位置相同。
一般的排序操作,都是通过sql的Orderby来实现的,这里Compare的具体作用,暂时不明确。
不过,我们先看看这个CompareTo方法的具体实现吧:
1 /// 2 /// 排序方法(根据Id大小排序) 3 /// 4 ///
5 /// 6 public virtual int CompareTo( Object obj ) { 7 return ((ObjectBase)obj).Id.CompareTo( Id ); 8 }
这里只是单纯通过比较ID来获得大小关系。在排序等操作中,ID作为默认的OrderBy关键字。
这里只是wojiluORM的开篇。更多的内容请参看本系列的其他文章。
wojilu系统的ORM介绍
ORM中启用数据库事务

Tags:  抽象基类 java基类 什么叫虚基类 虚基类 wojilu

延伸阅读

最新评论

发表评论