文档章节

NHibernate 多对多映射的数据更新

李朝强
 李朝强
发布于 2016/02/17 12:33
字数 939
阅读 110
收藏 1

#程序员薪资揭榜#你做程序员几年了?月薪多少?发量还在么?>>>

最近在用 NHibernate 做多对多更新时突然发现 NHibernate 更新的策略很差, 对多对多关系的更新居然是先全部删除再插入全部数据, 感觉非常奇怪, 现在还原如下:

原来的实体类关系如下:

public class User { public virtual int Id { get; set; } public virtual string Name { get; set; } public virtual ICollection<Role> Roles { get; set; } public User() {
        Roles = new HashSet<Role>();
    }
} public class Role { public virtual int Id { get; set; } public virtual string Name { get; set; } public virtual ICollection<User> Users { get; set; } public Role() {
        Users = new HashSet<User>();
    }

}

即一个用户可以有多个角色, 一个角色也可以有多个人, 典型的多对多关系, 对应的映射代码如下:

public class UserMapping : ClassMapping<User> { public UserMapping() {
        Table("[User]");

        Id(m => m.Id, map => { map.Column("[Id]"); map.Type(NHibernateUtil.Int32); map.Generator(Generators.Identity);
        });

        Property(m => m.Name, map => { map.Column("[Name]"); map.Type(NHibernateUtil.String);
        });

        Bag(
            m => m.Roles, map => { map.Table("[User_Role]"); map.Key(k => { k.Column("[UserId]"); });
            },
            rel => {
                rel.ManyToMany(map => { map.Class(typeof(Role)); map.Column("[RoleId]");
                });
            }
        );
    }
} public class RoleMapping : ClassMapping<Role> { public RoleMapping() {
        Table("[Role]");

        Id(m => m.Id, map => { map.Column("[Id]"); map.Type(NHibernateUtil.Int32); map.Generator(Generators.Identity);
        });

        Property(m => m.Name, map => { map.Column("[Name]"); map.Type(NHibernateUtil.String);
        });


        Bag(
            m => m.Users, map => { map.Table("[User_Role]"); map.Key(k => { k.Column("[RoleId]"); }); map.Inverse(true);
            },
            rel => {
                rel.ManyToMany(map => { map.Class(typeof(User)); map.Column("[UserId]");
                });
            }
        );

    }
}

数据库关系图如下:

当向用户添加或删除角色是, 发现更新的效率特别低, 代码如下:

using (var session = sessionFactory.OpenSession()) { var user = session.Query<User>().First(); var firstRole = user.Roles.First();
    user.Roles.Remove(firstRole);
    session.Update(user); var roleCount = session.Query<Role>().Count(); var role = new Role { Name = "Role " + (roleCount + 1) };
    session.Save(role);

    user.Roles.Add(role);
    session.Update(user);

    session.Update(user);
    session.Flush();
}

上面的代码是将用户的第一个角色删除, 再添加一个新的角色, NHibernate 生成的 SQL 语句如下(仅包含对关系表User_Role的操作):

DELETE FROM [User_Role] WHERE [UserId] = @p0;@p0 = 1 [Type: Int32 (0)] INSERT INTO [User_Role]  ([UserId], [RoleId]) VALUES (@p0, @p1);@p0 = 1 [Type: Int32 (0)], @p1 = 2 [Type: Int32 (0)] INSERT INTO [User_Role]  ([UserId], [RoleId]) VALUES (@p0, @p1);@p0 = 1 [Type: Int32 (0)], @p1 = 7 [Type: Int32 (0)] INSERT INTO [User_Role]  ([UserId], [RoleId]) VALUES (@p0, @p1);@p0 = 1 [Type: Int32 (0)], @p1 = 6 [Type: Int32 (0)] INSERT INTO [User_Role]  ([UserId], [RoleId]) VALUES (@p0, @p1);@p0 = 1 [Type: Int32 (0)], @p1 = 10 [Type: Int32 (0)]

居然是先将属于该用户的全部角色删除, 再添加一份新的进来, 完全无法接受, 反过来思考觉得肯定是自己的问题, 经过一番搜索 (Google), 发现 StackOverflow 上也有人问类似的问题, 并且最终在 NHibernate Tip: Use set for many-to-many associations 发现了解决方案, 将多对多的映射的bag改为用set, 问题终于得到了解决, 改过后的映射如下:

Set(
    m => m.Roles, map => { map.Table("[User_Role]"); map.Key(k => { k.Column("[UserId]"); });
    },
    rel => {
        rel.ManyToMany(map => { map.Class(typeof(Role)); map.Column("[RoleId]");
        });
    }
);

将UserMapping和RoleMapping中多对多映射全部改为Set之后, 上面的测试代码生成的 SQL 如下:

DELETE FROM [User_Role] WHERE [UserId] = @p0 AND [RoleId] = @p1;@p0 = 1 [Type: Int32 (0)], @p1 = 8 [Type: Int32 (0)] INSERT INTO [User_Role]  ([UserId], [RoleId]) VALUES (@p0, @p1);@p0 = 1 [Type: Int32 (0)], @p1 = 9 [Type: Int32 (0)]

在 NHibernate 参考文档的 19.5. Understanding Collection performance 中这样描述:

Bags are the worst case. Since a bag permits duplicate element values and has no index column, no primary key may be defined. NHibernate has no way of distinguishing between duplicate rows. NHibernate resolves this problem by completely removing (in a single DELETE) and recreating the collection whenever it changes. This might be very inefficient.

不只是多对多, 如果你的集合需要更新, NHibernate 推荐的是:

19.5.2. Lists, maps, idbags and sets are the most efficient collections to update

然而 bags 也不是一无是处:

19.5.3. Bags and lists are the most efficient inverse collections

Just before you ditch bags forever, there is a particular case in which bags (and also lists) are much more performant than sets. For a collection with inverse="true" (the standard bidirectional one-to-many relationship idiom, for example) we can add elements to a bag or list without needing to initialize (fetch) the bag elements! This is because IList.Add() must always succeed for a bag or IList (unlike an ISet). This can make the following common code much faster.

Parent p = sess.Load(id); Child c = new Child(); c.Parent = p;
p.Children.Add(c); //no need to fetch the collection! sess.Flush();

由此可见,bag在多对多映射更新时性能较差, 如果不需要更新,则可以放心使用, 在需要更新时则set是更好的选择。

本文转载自:http://beginor.github.io/2015/03/17/nhibernate-many-to-many-update.html

李朝强
粉丝 93
博文 305
码字总数 155420
作品 0
郑州
产品经理
私信 提问
加载中

评论(0)

Visual Entity 下载列表(最新版为 1.8.0 ,已支持Postgre&DB2,即将支持 Sysbase ASE)

发展线路: 2.x 将支持的功能有: 1、增加重构功能。当修改类或属性名称,提示并进行重构以更新所有引用。 2、允许将自定义的特性放在属性窗口中,以方便编辑。 3、Web Services 代码生成。 ...

长平狐
2013/06/17
336
1
NHibernate3剖析:Mapping篇之集合映射基础(4):Map映射

本节内容 系列引入 Dictionary映射 案例分析 案例一:DictionaryOfElements 案例二:DictionaryOfComponents 案例三:DictionaryOfOneToMany 案例四:DictionaryOfManyToMany 案例五:Dicti...

YJingLee's Blog
2010/05/12
0
0
NHibernate3剖析:Mapping篇之集合映射基础(2):Bag映射

本节内容 系列引入 Bag映射 案例分析 案例一:BagOfElements 案例二:BagOfComponents 案例三:BagOfComponentsWithParent 案例四:BagOfNestedComponentsWithParent 案例五:BagOfOneToMan...

YJingLee's Blog
2010/05/10
0
0
耗时两月,NHibernate系列出炉

写在前面 这篇总结本来是昨天要写的,可昨天大学班长来视察工作,多喝了点,回来就倒头就睡了,也就把这篇总结的文章拖到了今天。 nhibernate系列从开始着手写,到现在前后耗费大概两个月的时...

wolfy
2014/12/02
0
0
NHibernate3剖析:Mapping篇之集合映射基础(3):List映射

本节内容 系列引入 List映射 案例分析 案例一:ListOfElements 案例二:ListOfComponents 案例三:ListOfOneToMany 案例四:ListOfManyToMany 结语 系列引入 NHibernate3.0剖析系列分别从Con...

YJingLee's Blog
2010/05/12
0
0

没有更多内容

加载失败,请刷新页面

加载更多

Flutter 使用Navigator进行局部跳转页面

老孟导读:Navigator组件使用的频率不是很高,但在一些场景下非常适用,比如局部表单多页填写、底部导航一直存在,每个tab各自导航场景。 Navigator 是管理路由的控件,通常情况下直接使用N...

老孟Flutter
今天
28
0
使用site-maven-plugin在github上搭建公有仓库

简介 Maven是我们在开发java程序中经常使用的构建工具,在团队合作开发过程中,如果我们想要将自己写好的jar包共享给别人使用,通常需要自己搭建maven仓库,然后将写好的jar包上传到maven仓库...

flydean
今天
25
1
Python笔记:爬取各国疫情仙去人数数据制作南丁格尔玫瑰图

我凌晨用Python爬虫实时从网上爬取各国在疫情中仙去人数的数据,然后仿人民日报制作成“南丁格尔玫瑰图” 英国37048;意大利32877;西班牙27117都对得上; 但美国、法国、巴西的跟百度大数据...

tengyulong
今天
26
0
PHP Session的用法

在 PHP 中,Session 是一种服务器端的机制,服务器使用一种散列表的结构(类似于 JSON)来保存信息。相比于保存在客户端的 Cookie,Session 将用户交互信息保存在了服务器端,使得同一个客户...

linuxprobe2020
今天
37
0
webpack.03-打包js同时生成html页面

在空文件夹初始化:CMD npm init -y cnpm install -D webpack webpack-cli 文件结构 src(文件夹)--->test.js console.log('hello webpack') src(文件夹)--->index.html <html> ......

_qq507570355
今天
24
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部