hibernate(十)双向关联关系的CRUD
一、保存
1、
假设一个group有多个user,一个user只属于一个group,当保存user对象到数据库中时可以
User u = new User(); u.setName("u1"); Group g = new Group(); g.setName("g1"); u.setGroup(g); Session s = sessionFactory.getCurrentSession(); s.beginTransaction(); s.save(g);// 默认hibernate不会帮我们保存 s.save(u); s.getTransaction().commit();
需要注意hibernate默认不会帮我们保存group对象,应该设置一下。
@ManyToOne有个属性cascade(级联),CascadeType它有
(1)ALL:所有的操作时对当前对象所有级联的对象都会被hibernate操作
(2)MERGE:调用merge()方法时(合并的时候)会进行级联 (MERGE=save+update)
(3)PERSIST:调用persist()方法时(存储的情况下)会进行级联
(4)PEFRESH: 当一个session从数据库中load了一个对象和它级联的对象到内存中,而另外一个session在数据库中对这个对象进行了修改,则hibernate会对第一个sesson load的对象级联的对象刷新操作 (A里面需要读B改过之后的数据)
(5)REMOVE: 删除的情况下会进行级联
(Cascade的属性指明做什么操作的时候关联对象是绑定在一起的)
测试:将User类的getGroup()方法上的@ManyToOne注解修改为@ManyToOne(cascade=(CascadeType.ALL)),然后注释掉上面代码块中s.save(g),运行程序可以发现group会被hibernate自动保存到数据库中。
2、
上面我们存user,关联的group hibernate会帮我们存进去,如果我们存group,关联的user会不会帮我们存进去呢?答:不会
User u1 = new User(); u1.setName("u1"); User u2 = new User(); u1.setName("u2"); User u3 = new User(); u1.setName("u3"); Group g = new Group(); g.setName("g1"); g.getUsers().add(u1); g.getUsers().add(u2); g.getUsers().add(u3); Session s = sessionFactory.getCurrentSession(); s.beginTransaction(); s.save(g);//这里如果不设置的话,u1,u2,u3不会自动帮我们存进去 s.getTransaction.commit();
如果想让hibernate帮我们存入group的关联对象需要设置一下:将Group类中的getUsers()方法上的注解@OneToMany(mappedBy="group")修改为@OneToMany(mappedBy="group",cascade=(CascadeType.ALL)),注:如果只这么设置的话,u1,u2,u3存入数据库中后group_id都为NULL,因为u1,u2,u3对象中的group属性是null,所以需要在代码块中加u1.setGroup(g);u2.setGroup(g);u3.setGroup(g)。
规律:(1)多对一关系从多的一方操作比较简单;(2)如果关联关系是双向的,则需要设好导航(mappedBy);(3)双向关系在程序中要设定双向关联
二、读
1、get方式
我们先从数据库中get一个User出来
Session s = sessionFactory.getCurrentSession(); s.beginTransaction(); // 当我们从数据库中将这个对象取出来的时候不管设不设cascade都会把关联的Group对象取出来 User u = (User)s.get(User.class , 1); s.getTransaction().commit;
我们从数据库中get一个Group出来,不管设不设cascade不会把关联的user取出来,cascade只在存储的时候有效,读取需要再另外设置:
将Group类中的getUsers()方法上注解@OneToMany(mappedBy="group",cascade=(CascadeType.ALL))修改为:@OneToMany(mappedBy="group",cascade=(CascadeType.ALL),fetch=FetchType.EAGER),然后就可以在代码块中写:
Group g = (Group)s.get(Group.class,1);// 如果不加上fetch则不会将关联的User取出来 for(User u : g.getUsers()) { System.out.println(u.getName()); }
User类中getGroup()方法上注解@ManyToOne默认FetchType是EAGER,因为
当我们从数据库中将这个user对象取出来的时候不管设不设cascade都会把关联的Group对象取出来
当我们将User类中注解@ManyToOne添加属性fetch=FetchType.LAZY时,当我们用到user关联的group时,才会取出来:
Session s = sessionFactory.getCurrentSession(); s.beginTransaction(); User u = (User)s.get(User.class , 1);// 这时不会取出group System.out.println(u.getGroup().getName());// 这时会取出group s.getTransaction().commit; //System.out.println(u.getGroup().getName());这时会报错no session,原因是session已经关闭了。如果设置为EAGER则不会报错。
FetchType是个枚举类型,有:
(1)EAGER:(渴望)Define that data must be eagerly fetched
(2)LAZY:(懒惰)Define that data can be lazily fetched
2、load方式
三、更新
Session s = sessionFactory.getCurrentSession(); s.beginTransactio(); User u = (User)s.load(User.class , 1); u.setName("user"); u.getGroup().setName("g"); s.getTransaction().commit();//会发出两条update语句
这样写可以但是我们通常更新的是处于游离状态的对象,而当user处于游离状态时session已经关闭了没法保存。
Session s = sessionFactory.getCurrentSession(); s.beginTransaction(); User u = (User)s.get(User.class,1); s.getTransaction().commit(); u.setName("user"); u.getGroup().setName("group"); Session s2 = sessionFactory.getCurrentSession(); s2.beginTransaction(); s2.update(u); // 当User类中的getGroup()方法注解@ManyToOne添加属性cascade=CascadeType.ALL时,会更新group,否则当执行update方法时不会update Group s2.getTransaction().commit();
四、删除
1、删除一个User
Session s = sessionFactory.getCurrentSession(); s.beginTransaction(); User u = (User)s.load(User.class,1); s.delete(u); // 这时候会把u删除,然后会把u关联的group对象关联的所有user删除 s.getTransaction().commit();
如果不想把group删除可以:
u.setGroup(null); s.delete(u);
还可以使用HQL:
Session s = sessionFactory.getCurrentSession(); s.beginTransaction(); s.createQuery("delete from User u where u.id=1"); s.getTransaction().commit();
2、删除Group
Session s = sessionFactory.getCurrentSession(); s.beginTransaction(); Group g = (Group)s.load(Group.class,1); s.delete(g); // 这时候会把g删除会把g对象关联的所有user删除(当然已经设置了cascade=all) s.getTransaction().commit();
如果不想删除所有user,可以把user的group_id设成null。