本次实验环境:spring boot + spring data jpa + hibernate
本次实验用例:
用户:
id
userName
password
info
用户信息
id
nickName
sex
用户类中有信息类的成员属性,信息类中没有,一个用户对应一个信息,一个信息对应一个用户。
存在用户就一定存在信息,存在信息就一定存在用户。实际上存在用户可以不存在对应的信息。但这是错误的业务逻辑状态。
#User类:
@Entity
@Table(name="t_121_user")
public class User {
@Id
@GeneratedValue
private Long id;
private String userName;
private String password;
@OneToOne(cascade=CascadeType.ALL)
@JoinColumn(name="info_id", unique=true)
private Info info = new Info();
//getter与setter省略
}
#Info类:
@Entity
@Table(name="t_121_info")
public class Info {
@Id
@GeneratedValue
private Long id;
private String nickName;
private String sex;
//getter与setter省略
}
#生成表
CRUD:
- 添加User,因为User与Info共生,所以应该同时添加User和Info的记录,这就是为什么我在User类中的info使用饿汉式的方式直接赋值。
可以直接添加info,但是必须再将该新的info记录的id赋值给user记录的info_id来进行关联。
User user = new User();
user.setUserName("张三");
user.setPassword("123");
userDao.saveAndFlush(user);
因为设置了cascade,会同时生成两条记录,且关系正确,如果需要在添加时对info对象的属性赋值,可以通过给user的info成员变量赋值的方式,无需单独save info。
Info info = new Info();
info.setNickName("zhangsan");
info.setSex("male");
User user = new User();
user.setUserName("张三");
user.setPassword("123");
user.setInfo(info);
userDao.saveAndFlush(user);
- 更新user,假如现在我要修改密码,这一举动只涉及到user类,与user和info之间的关系无关,业务核心是通过id来更新password字段。我们可以创建新的user对象,然后setId()和setPassword(),通过saveAndFlush来达到需求。
然而此时user中有两个属性使用的是默认值,userName和info,数据库中相应字段同样会被更新成null。
为了防止这种情况的发生,我们必须把不需要更新的属性设置成数据库中的原来的值。所以在更新前应该先查询。
User user = userDao.findOne(1L);
user.setPassword("456");
userDao.saveAndFlush(user); - 更新info,可以通过user来更新,也可以通过info自身的id来更新。同样通过user来更新时需保证user自身的字段值不改变。
- 删除user,通过user的id直接删除即可,并且级联删除对应的info记录。
userDao.delete(1L)
- 删除info,直接删除,此时会失败,user表中的info_id正指向该记录,无法删除,解决的方法是先断掉这个关系。
User user = userDao.findOne(1L);
user.setInfo(null);
userDao.saveAndFlush(user)
当然user和info是共生的,断掉关系删掉info之后,还是需要删掉对应的user,否则就是脏数据。