Hibernate学习笔记 单表映射

ゞ 浴缸里的玫瑰 2022-07-13 11:43 285阅读 0赞

建立实体类

配置好SessionFactory之后,我们就可以开始建立一对一的单表映射了。首先需要建立一个实体类,这里Getter、Setter、toString、equals等方法省略了。我们可以方便的使用IDEA或者Eclipse的生成代码功能轻松生成。

  1. package yitian.data;
  2. import java.util.Date;
  3. public class User {
  4. private int id;
  5. private String username;
  6. private String password;
  7. private String nickname;
  8. private Gender gender;
  9. private String address;
  10. private Date birthday;
  11. private Date registerTime;
  12. }

性别是一个枚举:

  1. package yitian.data;
  2. public enum Gender {
  3. FEMALE,
  4. MALE
  5. }

建立映射

有两种方法来建立实体类与数据库之间的映射。第一种方法是建立一个映射文件,映射文件名类似于Entity.hbm.xml,也是一个xml文件,其中定义了实体类和数据表之间的关系。这是一种传统方式,缺点是需要一个额外的配置文件,而且XML配置容易出错,如果放置位置不对的话还会导致找不到文件。所以现在第二种方式使用的更多。

现在更流行的方式是使用JPA注解。JPA是一个Java EE标准,定义了一组注解,将注解配置到实体类上,就可以建立实体类和数据表之间的映射。下面就是一个使用注解配置的实体类:

  1. package yitian.data;
  2. import org.hibernate.annotations.GenericGenerator;
  3. import org.hibernate.annotations.NaturalId;
  4. import javax.persistence.*;
  5. import java.util.Date;
  6. import java.util.Objects;
  7. @Entity
  8. @Table(name = "user")
  9. public class User {
  10. private int id;
  11. private String username;
  12. private String password;
  13. private String nickname;
  14. private Gender gender;
  15. private String address;
  16. private Date birthday;
  17. private Date registerTime;
  18. public User() {
  19. }
  20. @Id
  21. @GeneratedValue(strategy = GenerationType.IDENTITY)
  22. public int getId() {
  23. return id;
  24. }
  25. public void setId(int id) {
  26. this.id = id;
  27. }
  28. @NaturalId
  29. public String getUsername() {
  30. return username;
  31. }
  32. public void setUsername(String username) {
  33. this.username = username;
  34. }
  35. @Column
  36. public String getPassword() {
  37. return password;
  38. }
  39. public void setPassword(String password) {
  40. this.password = password;
  41. }
  42. @Column
  43. public String getNickname() {
  44. return nickname;
  45. }
  46. public void setNickname(String nickname) {
  47. this.nickname = nickname;
  48. }
  49. @Column
  50. @Enumerated(EnumType.STRING)
  51. public Gender getGender() {
  52. return gender;
  53. }
  54. public void setGender(Gender gender) {
  55. this.gender = gender;
  56. }
  57. @Column
  58. public String getAddress() {
  59. return address;
  60. }
  61. public void setAddress(String address) {
  62. this.address = address;
  63. }
  64. @Column
  65. public Date getBirthday() {
  66. return birthday;
  67. }
  68. public void setBirthday(Date birthday) {
  69. this.birthday = birthday;
  70. }
  71. @Column(name = "register_time")
  72. @Temporal(TemporalType.DATE)
  73. public Date getRegisterTime() {
  74. return registerTime;
  75. }
  76. public void setRegisterTime(Date registerTime) {
  77. this.registerTime = registerTime;
  78. }
  79. @Override
  80. public boolean equals(Object o) {
  81. if (this == o) return true;
  82. if (o == null || getClass() != o.getClass()) return false;
  83. User user = (User) o;
  84. return id == user.id &&
  85. Objects.equals(username, user.username) &&
  86. Objects.equals(password, user.password) &&
  87. Objects.equals(nickname, user.nickname) &&
  88. gender == user.gender &&
  89. Objects.equals(address, user.address) &&
  90. Objects.equals(birthday, user.birthday) &&
  91. Objects.equals(registerTime, user.registerTime);
  92. }
  93. @Override
  94. public int hashCode() {
  95. return Objects.hash(id, username, password, nickname, gender, address, birthday, registerTime);
  96. }
  97. @Override
  98. public String toString() {
  99. return "User{" +
  100. "id=" + id +
  101. ", username='" + username + '\'' +
  102. ", password='" + password + '\'' +
  103. ", nickname='" + nickname + '\'' +
  104. ", gender=" + gender +
  105. ", address='" + address + '\'' +
  106. ", birthday=" + birthday +
  107. ", registerTime=" + registerTime +
  108. '}';
  109. }
  110. }

使用到的注解如下:

  • Entity和Table注解用来标记实体类。默认情况下表名和实体类名相同,如果希望自定义表名使用Table注解并设置相应的name属性。
  • Id注解标明了主键。GeneratedValue表示自动生成主键,生成策略默认为AUTO,表示由Hibernate根据数据库类型自动选择合适的策略。如果希望在不同数据库之间有更好的可移植性,就选择默认策略。
  • NaturalId指定实体类的自然主键。一般情况下推荐使用无意义的数字作为主键,但是有时候类似于用户名、书籍编号这样的属性也具有主键的意义。这时候我们将这些属性定义为自然主键。同样可以向主键使用,更为方便。
  • Column将对应的实体类属性映射到数据表的列上,可以添加name参数自定义数据表的列名。
  • Temporal表示这一列数据是一个日期类型,具体的日期类型由TemporalType枚举表示,有DATE、TIME、TIMESTAMP三个类型,表示三种日期类型。
  • Enumerated用来映射枚举类型,可以使用EnumType枚举定义如何映射枚举,如果是EnumType.ORDINAL,那么就会使用枚举对应的数字,如果使用EnumType.STRING,就会使用枚举对应的名称。

注解即可以放在字段上,也可以放在方法上。如果放在字段上,Hibernate就会使用字段对数据进行操作。如果放在方法上,Hibernate就使用方法操作数据。不论哪种都是可以的,但是需要注意保持一致性,不能混用。Hibernate检索的依据是Id注解的位置,如果Id放在了方法上而其他注解放在了字段上,Hibernate就会忽略字段上的所有注解,反之亦然。推荐将注解放在方法上,因为如果方法中进行了额外的操作,我们就可以保证这些操作能够正确的执行。

上面这个实体类基本上涵盖了常用的类型映射和方法。我们做一个简单的小程序完全够用了。不过仅仅添加注解还是不够的,我们需要向hibernate.cfg.xml中添加一句,表示我们使用注解配置了某个类。

  1. <mapping class="yitian.data.User"/>

这样,我们的Hibernate的单表映射就配置完成了。

CRUD

CRUD也就是英文的增删查改的意思,我们配置好单表映射之后,就可以利用Hibernate提供的方法方便的操作数据了。

添加

Hibernate支持事务,因此在添加数据之前需要新建一个事务,添加完数据之后需要提交事务。这一点在代码中很清楚。如果事务失败,可以使用rollback()方法回滚事务,返回到操作之前的状态。

添加数据使用save方法。这个方法很简单,直接看代码就可以了。

  1. @Test
  2. public void testAdd() {
  3. User user = new User();
  4. user.setUsername("yitian");
  5. user.setPassword("12345678");
  6. user.setBirthday(new Date());
  7. user.setGender(Gender.MALE);
  8. user.setRegisterTime(new Date());
  9. try (Session session = factory.openSession()) {
  10. session.beginTransaction();
  11. session.save(user);
  12. session.getTransaction().commit();
  13. }
  14. try (Session session = factory.openSession()) {
  15. Query<User> query = session.createQuery("from User");
  16. List<User> users = query.list();
  17. Assert.assertNotNull(users);
  18. }
  19. }

查询

Hibernate支持多种查询操作。这里仅仅用了最简单的HQL语句,查询性别为男的所有人。详细的查询方法会在后面说明。

  1. @Test
  2. public void testQuery() {
  3. try (Session session = factory.openSession()) {
  4. Query<User> findAllMan = session.createQuery("from User u where u.gender='MALE'");
  5. List<User> users = findAllMan.list();
  6. Assert.assertNotNull(users);
  7. logger.debug(users.toString());
  8. }
  9. }

更新

更新使用update()方法,首先需要取出一个对象,修改属性之后在使用update方法更新到数据库中。不要忘了提交事务。

  1. @Test
  2. public void testUpdate() {
  3. try (Session session = factory.openSession()) {
  4. session.beginTransaction();
  5. User u = session.bySimpleNaturalId(User.class).load("zhang3");
  6. u.setEmail("zhang3m@yitian.com");
  7. session.update(u);
  8. session.getTransaction().commit();
  9. u = session.bySimpleNaturalId(User.class).load("zhang3");
  10. Assert.assertEquals(u.getEmail(), "zhang3m@yitian.com");
  11. }
  12. }

删除

删除方法是delete(),传输要删除的对象,然后提交事务即可。

  1. @Test
  2. public void testDelete() {
  3. User user = new User();
  4. user.setUsername("wuli");
  5. user.setNickname("呜呜呜");
  6. try (Session session = factory.openSession()) {
  7. session.beginTransaction();
  8. session.save(user);
  9. session.getTransaction().commit();
  10. User u = session.bySimpleNaturalId(User.class).load("wuli");
  11. logger.debug(u.toString());
  12. Assert.assertNotNull(u);
  13. session.beginTransaction();
  14. session.delete(u);
  15. session.getTransaction().commit();
  16. u = session.bySimpleNaturalId(User.class).load("wuli");
  17. Assert.assertNull(u);
  18. }
  19. }

这里还使用到了bySimpleNaturalId方法。如果实体类中只定义了一个自然主键,就可以使用这个方法然后使用load方法获取自然主键对应的对象。

以上就是Hibernate单表映射的简单例子。代码没有给完全,主要看一看Hibernate的运行模式即可。看到这里,大家应该对Hibernate有了大致的了解。

发表评论

表情:
评论列表 (有 0 条评论,285人围观)

还没有评论,来说两句吧...

相关阅读

    相关 Hibernate关系映射之一对多映射

    一、基本概述 在表中的一对多,是使用外键关联,通过一张表的一个键另一个表的外键来建立一多关系;而在类中表示为一个类中有一个集合属性包含对方类的很多对象,而在另一个类中,只包