hibernate 关系映射 annotation 版

关于mappedBy

双向关联是单向关联的扩展,只要一方写了配置(通常是 关联维护方即有外键的那方),另一方就可以通过 mappedBy 去找配置,不需要2边同时写.

 

关于@JoinColumn

@JoinColumn(name="外键名字段名") 可以用来指定外键,双向关联里的外键是不需要写 private 和 get / set 的.这个由hibernate自动维护.你表里设计了外键,但没用JoinColumn指定.hibernate会无视掉并以自己的命名方式想象一个外键,假如你数据库里的外键和这个想象外键对不上,报错.

 

也就是注意 @JoinColumn(name="外键名字段名")  注解写在哪里的问题.如果外键是自己的.@JoinColumn(name="外键名字段名")也写在自己这边,外键再写成成员变量就会报错.

 

怕漏写,总之外键的 private 和 get / set 是没报错就当成员变量写上.报错了去掉就是了.

 

关于 cascade 我大体是这么认为的.还有待验证.

注意这个是数组,所以 cascade={} .

先简单说明下,  A 类里有一个方法 getB() 用来获取 B 的对象

假如 getB() 方法上写了 cascade 注解,说明 A 的更删之类的操作会影响数据表内B的数据.

关联关系对这个没任何影响.总之就是A里获取B的方法上写上 cascade ,则 A 的操作就能影响数据库内具体的 B 数据.

多对多关系映射是通过中间表来维护关联,这个由hibernate自动维护中间表,不要写 cascade .

 

关于 fetch=FetchType.LAZY 我大体也是这么认为的.还有待验证.

先简单说明下,  A 类里有一个方法 getB() 用来获取 B 的对象

假如 getB() 方法上写了 fetch 注解,说明 A 类实体化后,除非使用 A.getB() 这个方法.否则不会立即从数据库里读取 B 的数据.

fetch=FetchType.EAGER 是默认值,就是立即读取.假如要立即读取B的数据.不写 fetch 就是了.

关联关系对这个没任何影响.

 



一对一关联

@OneToOne

一对一的主键关联实在看不出意义何在.反倒是获取实例的时候可能多N多没用的数据,所以不深究了...至于外键关联,直接看下面的一对多,多对一.

 


 

一对多,多对一关联

表字段说明

OneObject 表:

one_id 主键

 

ManyObject 表:

many_id 主键

one_id 外键,与OneObject 的主键关联



 

单向一对多

OneObject:

@OneToMany
@JoinColumn(name="one_id")

private Set<ManyObject> mb = new HashSet<ManyObject>();

 

ManyObject:

private int one_id; //这个就是和 @JoinColumn(name="one_id")  对应的 one_id .

不需要注解.

 


 

单向多对一

OneObject:

不需要注解.

 

ManyObject:

@ManyToOne
@JoinColumn(name="one_id")   //只需要在ManyObject指定外键为one_id.它就会去找 OneObject 的主键来对应.

private OneObject one;

 


 

双向一对多,双向多对一

OneObject:

@OneToMany(mappedBy="one") //告诉hibernate关联去找 one .

private Set<ManyObject> mb = new HashSet<ManyObject>();

 

ManyObject:

@ManyToOne
@JoinColumn(name="one_id")  //注意,在这里指定外键名后,不要在 写 private int one_id; 和它的get / set 方法了.这个改由hibernate维护.写上是要报错的.

private OneObject one;  //这个对应的就是 @OneToMany(mappedBy="one") 的 one.

 


 

多对多关联

表字段说明

ManyA 表:

a_id 主键

 

ManyB 表:

b_id 主键

 

A_B 表:(保存映射关系的中间表)

t_Aid 外键,与ManyA 的主键关联

t_Bid 外键,与ManyB 的主键关联

 


 

单向多对多

ManyA:

@ManyToMany
@JoinTable(
            name="A_B", //中间表名
            joinColumns={@JoinColumn(name="t_Aid")}, //设置ManyA自己在中间表的对应外键
            inverseJoinColumns={@JoinColumn(name="t_Bid") //设置对方(ManyB)在中间表的对应外键
        }
)

private Set<ManyB> mb = new HashSet<ManyB>();

 

ManyB:

不需要注解.

 


 

双向多对多

ManyA:

@ManyToMany
@JoinTable(
            name="A_B", //中间表名
            joinColumns={@JoinColumn(name="t_Aid")}, //设置ManyA自己在中间表的对应外键
            inverseJoinColumns={@JoinColumn(name="t_Bid") //设置对方(ManyB)在中间表的对应外键
        }
)

private Set<ManyB> mb = new HashSet<ManyB>();

 

ManyB:

@ManyToMany(mappedBy="mb") //告诉hibernate维护这个关联去找另一个对象的注解就可以了.不需要重复写.

private Set<ManyA> ma = new HashSet<ManyA>();

 

注:以上注释,ManyA的更删会影响到中间表A_B.但ManyB的更删不会影响到.假如2边都需要影响A_B表,则ManyB也写注解.注意 joinColumns 以及 inverseJoinColumns 的值对应不同.(其实就是两边都做成单向多对多)

ManyB:

@ManyToMany
@JoinTable(
            name="A_B",
            joinColumns={@JoinColumn(name="t_Bid")},
            inverseJoinColumns={@JoinColumn(name="t_Aid")
        }
 )

private Set<ManyA> ma = new HashSet<ManyA>();

你可能感兴趣的:(java,spring,Hibernate,annotation,Web)