2

我需要在其中两个中使用非 PK 列关联一些表。表和 FK 设置如下:

在此处输入图像描述

因此,可以通过 MessageUserGroups 表将消息分配给许多用户和组。此外,可以通过 UserGroups 表将一个用户分配给多个组。

由于遗留原因,用户和组中的 AdGuid 字段(唯一标识符)不是主键。但是,我想使用 MessageUserGroups 中的 UserId 和 GroupId 字段通过其 AdGuid 字段将消息与用户和组相关联。

此外,我想在 User 上有一个 PostedMessages 属性,它将 Message.AuthorId 与 User.AdGuid 相关联,并在 User 和 Group 上有 Messages 属性,它通过 MessageUserGroups 与消息相关。

我该如何向 Fluent NHibernate 解释这一点?

目前的映射看起来像:

public MessageMap()
{
    Id(m => m.Id);
    References(m => m.Author).Nullable().ForeignKey("FK_Messages_Author").Column("AuthorId").Fetch.Join().PropertyRef(u => u.AdGuid);

    HasManyToMany<users.user>(m => m.Users)
        .Cascade.All()
        .ParentKeyColumn("MessageId")
        .ChildKeyColumn("UserId")
        .Table("MessageUserGroups")
        .FetchType.Join();

    HasManyToMany<users.group>(m => m.Groups)
        .Cascade.All()
        .ParentKeyColumn("MessageId")
        .ChildKeyColumn("GroupId")
        .Table("MessageUserGroups")
        .FetchType.Join();
}
public UserMap()
{
    Id(u => u.Id);
    Map(u => u.AdGuid);

    HasManyToMany<staff.Message>(u => u.Messages)
        .Cascade.All()
        .ParentKeyColumn("AdGuid")
        .ChildKeyColumn("UserId")
        .Inverse()
        .LazyLoad()
        .Table("MessageUserGroup")

    HasMany<staff.Message>(u => u.PostedMessages)
        .Cascade.All()
        .KeyColumn("AuthorId")
        .Inverse()
        .LazyLoad()
        .Table("Messages")

    HasManyToMany<Group>(u => u.Groups)
        .Cascade.All()
        .Not.LazyLoad()
        .Cascade.All()
        .ParentKeyColumn("UserID")
        .ChildKeyColumn("GroupID")
        .Table("UserGroups")
}
public GroupMap()
{
    Id(g => g.Id);
    Map(g => g.AdGuid);

    HasManyToMany<staff.Message>(g => g.Messages)
        .Cascade.All()
        .ParentKeyColumn("AdGuid")
        .ChildKeyColumn("GroupId")
        .Inverse()
        .LazyLoad()
        .Table("MessageUserGroup")

    HasManyToMany<User>(g => g.Users)
        .Inverse()
        .Cascade.All()
        .ParentKeyColumn("GroupId")
        .ChildKeyColumn("UserId")
        .Table("UserGroups")
}

有点长,我知道。我可以使用 NH 成功查询消息。但是,当延迟加载 User 或 Group 的 Messages 属性时,NH 会生成以下 SQL(为简洁起见,删除了额外的属性):

SELECT users0_.MessageId as MessageId1_, users0_.UserId as UserId1_, user1_.Id
as id38_0_, user1_.AdGuid as guid38_0_ FROM MessageUserGroups users0_
LEFT OUTER JOIN Users user1_ on users0_.UserId=user1_.Id WHERE users0_.MessageId=@p0

该联接无效,因为它试图将 uniqueidentifer MessageUserGroups.UserId 与 bigint User.Id 进行比较。

同样,当延迟加载 User.PostedMessages 时,会生成以下 SQL(同样,缩写):

SELECT postedmess0_.AuthorId as AuthorId1_, postedmess0_.Id as Id1_, postedmess0_.Id as Id43_0_, postedmess0_.AuthorId as AuthorId43_0_
FROM Staff.Messages postedmess0_
WHERE postedmess0_.AuthorId=@p0

并且@p0设置为 89,这是我正在测试的用户的 ID,但 AuthorId 需要是唯一标识符。

在这两个之后,我得到一个SqlException:Operand type clash: uniqueidentifier is incompatible with bigint这是意料之中的。

看起来(流利的)NHibernate 总是想加入 PK,即使文档建议指定 ParentKeyColumn 和 ChildKeyColumn 应该允许我指定我喜欢的任何列。这真的可能吗?我误读了吗?

4

1 回答 1

1

ParentKeyColumn and ChildKeyColumn specify the column names in the link table between and not the columns of the entity-tables joined. What you need is PropertyRef and ChildPropertyRef on the hasmanytomany to specify the properties to join to

于 2012-08-19T20:25:39.693 回答