今天碰到一个外联结问题,总结解决的步骤:
1. 以前没注意left join on 问题,第一次使用的hql: table a left join table b on a.id = b.aid 出现异常,经过查询,发现hibernate 不能这样使用 做链接;
2. table a left join table b where a.id = b.aid 还有有问题,继续修改;
3. table a left join table b with a.id = b.aid 还有有问题,继续修改;
4. 经过多次查询资料,发现hibernate 不支持如上写的sql,只能在配置文件中设置many to one;
one :
<set name="conf" cascade="none" inverse="false">
<key column="id"></key>
<one-to-many class="com.dao.ConfDAO" not-found="ignore"/>
</set>
注:cascade:save-update、delete、all-delete-orphan、all、none 这些值;不知道的话,自己去查;
inverse:属性只有两个值"true"和"false"。"true"表示将关系维护的权力交给对方,"false"表示不交出维护权力(默认值)。 not-found="ignore":用来指定引用的外键不存在时将如何处理:exception(默认)抛出异常 、ignore 忽略many : <many-to-one name="department" class="com.polycom.ccm.shared.dao.DepartmentDAO" column="department" not-found="ignore" insert="false" update="false"/>
注:many中insert update 不影响one,这边不能配置delete;
5. hql:我这边还是不能with关键字,不知道为什么;
如下是我查询正确的数据hql:from ConfDAO as conf LEFT JOIN conf.de as de where 1=1 and (conf.deId = 0 or de.id = conf.deId)
参考资料:
数据库中的表一般都是相互关联的,它们通过foreign key产生关系。
定义foreign key约束时可以指定三种引用行为:delete cascade、delete set null、delete no action,默认是delete on action。它们的含义是:
1、delete cascade : 删除主表的同时也删除子表有关的记录
这个行为适合主从表关系较为紧密的情况,比如菜单和子菜单。当主表的记录不存在时,从表的数据已经没有意义,存在也是多余,所以当删除主表时,从表相关记录也一同删除。
2、delete set null : 删除主表时将子表外键设置为NULL
这个行为适合主从表关系不是相当密切的情况,比如角色和用户,一个角色可以对应多个用户(一个用户也可以有多重角色),不能因为删除了某个角色,而把属于这个角色的用户都删除了,因为这些用户可能还和其他表有很大关系。
3、delete no action: 不做任何操作
这个情况下,如果主表的某个记录已经被引用,删除这条记录会失败,主要提醒用户注意数据完整性。
以上3种情况在hibernate下如果实现呢?下面以多对一关联举例。
假设新闻与用户存在多对一关联
1、删除用户的同时也删除他发表的新闻
News映射文件:
<many-to-one name="editer" class="Account">
</many-to-one>
Account映射文件:
<set name="news"
</set>
2、删除用户时不删除他发表的新闻
其实新闻表除了保存用户ID外,我建议还保存用户的名字,多了一点数据冗余,但是在查询新闻时不用外连接用户表,而且在删除新闻时,虽然外键置NULL,但用户名还在,我们还能知道这新闻是谁发布的。
<set name="news"
</set>
注意以上的inverser=false
3、删除用户时不执行额外操作
只要在<set>中设置 cascade="none"
注意
以上情况只是一对多关系,对于多于多等关系还是有些区别的,比如我通过中间表映射的多对多关系,在没有设置cascade的情况下,删除记录时,也会主动删除中间表中的记录,如:
menu映射文件
<set name="roles" lazy="true" table="MenuRole" >
</set>
删除菜单时,会主动从中间表menurole删除记录