项目用的持久化层是hibernate 2.1.6.
前不久出现一个错误,简单描述一下:
现有3个对象:Party,TParty和Individual,其中Individual是Party的子类,Party和TParty各自独立,两个对象都映射到表T_Party。
当独立执行Individual person = (Individual)session.load(Individual.class, id)时,系统正确,.
而在同一thread下(OpenSessionInView),先session.find("from TParty"),再Individual person = (Individual)session.load(Individual.class, id)系统报错ClassCastException,经查此时系统返回的是TParty对象。
初步断定是hibernate问题,参看hiberante 2.1.7没有觉的有什么问题(因为自己电脑本地有这个版本,事实证明这是个错误)。折腾了一天,没有发现问题所在,最后只好下载了hibernate2.1.6,立刻发现问题所在。
hibernate在load对象时,会通过getEntity(key)查看是否加载过,而Key对象几个主要方法如下:
private
Key(Serializable id, Serializable identifierSpace, Class clazz,
boolean
batchLoadable)
{
if
(id
==
null
)
throw
new
AssertionFailure(
"
null identifier
"
);
this
.identifier
=
id;
this
.identifierSpace
=
identifierSpace;
this
.clazz
=
clazz;
this
.isBatchLoadable
=
batchLoadable;
}
/** */
/**
* Construct a unique identifier for an entity class instance
*/
public
Key(Serializable id, ClassPersister p)
{
this
( id, p.getIdentifierSpace(), p.getMappedClass(), p.isBatchLoadable() );
}
public
boolean
equals(Object other)
{
Key otherKey
=
(Key) other;
return
otherKey.identifierSpace.equals(
this
.identifierSpace)
&&
otherKey.identifier.equals(
this
.identifier);
}
public
int
hashCode()
{
int
result
=
17
;
result
=
37
*
result
+
identifierSpace.hashCode();
result
=
37
*
result
+
identifier.hashCode();
这个Key在hiberante几个版本都一样
但在ClassPersister在2.1.6和2.1.7却有不同:
hibernate 2.1.7中在AbstractEntityPersister中
public
Serializable getIdentifierSpace()
{
return
rootClassName;
}
但在hibernate 2.1.6 中的EntityPersister
public
Serializable getIdentifierSpace()
{
return
qualifiedTableName;
}
问题就出在这里。赶紧把hibernate从2.1.6升级到2.1.8。