我正在使用NHibernate 3.2,我在映射集合时遇到了问题。
查询用户帐户后......
UserAccount userAccount = (from u in Session.Query<UserAccount>()
where u.Username == username
select u).SingleOrDefault();
... Role
属性包含空集合,而roles
字段包含实际项目:
这是NH 3.2中的错误还是什么?我很确定它确实适用于3.1。
我认为代理行为可能有问题,特别是处理Roles
属性的延迟加载的覆盖。但是,我不知道如何检查生成的代理类(使用IL反编译器),因为据我所知它在运行时只存在于内存中。
编辑1
为了帮助我了解代理内部发生了什么,我刚刚发布了这个问题:Is there a way to decompile the proxy classes that are generated by NHibernate?。我认为这个问题应该是一个问题,因为它可能在许多其他情况下都有用。
编辑2
好吧,我设法反编译了动态代理类。这是管理Roles
属性的延迟加载的方法:
public override IEnumerable<Role> get_Roles()
{
IInterceptor interceptor = this.Interceptor;
if (interceptor == null)
{
throw new NotImplementedException();
}
object[] args = new object[0];
InvocationInfo info = new InvocationInfo(
this, (MethodInfo)methodof(UserAccount.get_Roles),
null, new Type[0], args);
args = info.Arguments;
return (IEnumerable<Role>)interceptor.Intercept(info);
}
我认为这里并没有什么特别之处。
编辑3
在调试拦截器时(部分显示在下面),我注意到methodName == "get_Roles"
调用它时TargetInstance
属性(在这种情况下为UserAccount
) ,它的roles
字段是一个空集合。在访问代理实例上的Roles
属性之前,代理的role
字段 具有已填充的集合。
public class DefaultDynamicLazyFieldInterceptor
: IFieldInterceptorAccessor, Proxy.DynamicProxy.IInterceptor
{
...
public object Intercept(InvocationInfo info)
{
var methodName = info.TargetMethod.Name;
if (FieldInterceptor != null)
{
...
检查InvocationInfo
实例时,proxy
和Target
个实例执行会有一个roles
字段,其中包含已填充的集合。< / p>
编辑4
我认为我遇到了NH Issue 2772 - Lazy-collection not loaded when a property is Lazy-loaded。
结束编辑
以下是实体类:
public class UserAccount : Entity
{
...
private IList<Role> roles;
public virtual IEnumerable<Role> Roles
{
get
{
return roles;
}
}
...
}
public class Role : Entity
{
...
}
以下是UserAccount映射的HBM映射文件的一部分:
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2" default-access="property"
auto-import="false" default-cascade="none" default-lazy="true">
<class xmlns="urn:nhibernate-mapping-2.2" dynamic-insert="true"
dynamic-update="true" schema="[MySchema]" mutable="true"
name="MyNamespace.UserAccount, MyAssembly, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null"
table="UserAccounts">
...
<bag access="nosetter.camelcase" cascade="none" name="Roles"
schema="[MySchema]" table="UserAccounts_Roles" mutable="true">
<key>
<column name="UserAccountId" />
</key>
<many-to-many class="MyNamespace.Role, MyAssembly, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null">
<column name="RoleId" />
</many-to-many>
</bag>
...
</class>
</hibernate-mapping>
这是HBM的一部分,它指定了角色映射:
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2" default-access="property"
auto-import="false" default-cascade="none" default-lazy="true">
<class xmlns="urn:nhibernate-mapping-2.2" dynamic-insert="true"
dynamic-update="true" schema="[MySchema]" mutable="true"
name="MyNamespace.Role, MyAssembly, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null"
table="Roles">
...
</class>
</hibernate-mapping>
答案 0 :(得分:1)
尝试使用
bag access="field.camelcase"
而不是
bag access="nosetter.camelcase"
答案 1 :(得分:0)
我刚检查了2772问题,显然它已经修复,将在下一个主要版本的NHibernate(计划被称为4.0)中发布。因此,我将问题标记为已回答,尽管有点过早。