nhibernate源码分析之五: 对象标识

王朝other·作者佚名  2006-01-09
窄屏简体版  字體: |||超大  

对象标识相当于数据表中的主键,在持久化中起着十分重要的作用,nhibernate通过对象标识来辨别两个持久对象是否相等。

在映射文件中,通过id属性来定义对象标识,内容如下:

<id name="orderId" type="Int32" unsaved-value="0" column="order_id">

<generator class=identity />

</id>

其中unsaved-value属性用来指明对象未持久化时的值,如果此值与未持久化的对象标识值不符,将无法save对象,generator用于指定标识对象的类型,常用的有identity, assigned等。

标识对象为实现IIdentitierGenerator接口的类,由IdentitierGeneratorFactory类根据映射文件的标识类型来创建,IIdentifierGenerator定义了Generate方法,用于产生对象标识。

1. 标识对象的建立

标识对象在持久化类AbstractEntityPersister中创建,通过它我们就可以对持久对象的标识进行操作了。

//*** AbstractEntityPersister.cs ***

public virtual IIdentifierGenerator IdentifierGenerator {

get {

if (idgen==null) {

throw new HibernateException("...");

}

return idgen;

}

}

idgen在构造函数中被赋值。

protected AbstractEntityPersister(PersistentClass model, ISessionFactoryImplementor factory) {

// ...

// GENERATOR

idgen = model.Identifier.CreateIdentifierGenerator(dialect);

useIdentityColumn = idgen is IdentityGenerator;

identitySelectString = useIdentityColumn ? dialect.IdentitySelectString : null;

// ...

}

其中model为PersistentClass或其子类,Identifier为Value类型的属性。

// *** Value.cs ***

public IIdentifierGenerator CreateIdentifierGenerator(Dialect.Dialect dialect) {

if ( uniqueIdentifierGenerator==null ) {

uniqueIdentifierGenerator = IdentifierGeneratorFactory.Create(identifierGeneratorStrategy, type, identifierGeneratorProperties, dialect);

}

return uniqueIdentifierGenerator;

}

//*** IdentitifierGeneratorFactory ***

public static IIdentifierGenerator Create(string strategy, IType type, IDictionary parms, Dialect.Dialect dialect) {

try {

System.Type clazz = (System.Type) idgenerators[strategy];

// ...

if (clazz==null) clazz = System.Type.GetType(strategy);

IIdentifierGenerator idgen = (IIdentifierGenerator) Activator.CreateInstance(clazz);

if (idgen is IConfigurable) ((IConfigurable) idgen).Configure(type, parms, dialect);

return idgen;

}

catch (Exception e) {

throw new MappingException("could not instantiate id generator", e);

}

}

Create方法通过标识对象类名来创建标识对象。

2. 标识对象在持久化中的使用

在会话和持久化操作一文,我曾提到当前会话会把要持久化的对象存储起来,直到调用Flush或关闭会话。存储持久对象的集合为entitiesByKey,这是一个Hashtable,它的key为一个Key对象, value为持久对象,Key对象简单的存储持久对象的id和IdentifierSpace。

在进行持久化操作时,nhibernate必须首先检查对象是否在entitiesByKey中,这由GetEntity方法完成,然后再根据对象是否在集合中作后续处理。

//*** SessionImpl.cs ***

public object GetEntity(Key key) {

return entitiesByKey[key];

}

下面来看看DoUpdate中的处理:

private void DoUpdate(object obj, object id) {

// ...

Key key = new Key(id, persister);

object old = GetEntity(key);

if (old==obj) {

throw new AssertionFailure("Hibernate has a bug in Update() ... or you are using an illegal id type");

}

else if ( old!=null ) {

throw new HibernateException("Another object was associated with this id ( the object with the given id was already loaded)");

);

// ...

AddEntity(key, obj);

AddEntry(obj, Status.Loaded, null, id, persister.GetVersion(obj), LockMode.None, true, persister);

// ...

}

如果首次对持久对象执行Update,此时old为空,操作顺利执行,并且对象被加入到集合中,

当再次调用Update时(在同一会话中,并且没有调用会导致Flush的操作),此时old不为空,将引发一个异常。

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
 
 
© 2005- 王朝網路 版權所有 導航