Hibernate单表映射

建立实体类

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

package yitian.data;

import java.util.Date;

public class User {
    private int id;
    private String username;
    private String password;
    private String nickname;
    private Gender gender;
    private String address;
    private Date birthday;
    private Date registerTime;

}

性别是一个枚举:

package yitian.data;


public enum Gender {
    FEMALE,
    MALE
}

建立映射

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

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

package yitian.data;


import org.hibernate.annotations.GenericGenerator;
import org.hibernate.annotations.NaturalId;

import javax.persistence.*;
import java.util.Date;
import java.util.Objects;

@Entity
@Table(name = "user")
public class User {
    private int id;
    private String username;
    private String password;
    private String nickname;
    private Gender gender;
    private String address;
    private Date birthday;
    private Date registerTime;

    public User() {
    }

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    public int getId() {

        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    @NaturalId
    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    @Column
    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    @Column
    public String getNickname() {
        return nickname;
    }

    public void setNickname(String nickname) {
        this.nickname = nickname;
    }

    @Column
    @Enumerated(EnumType.STRING)
    public Gender getGender() {
        return gender;
    }

    public void setGender(Gender gender) {
        this.gender = gender;
    }

    @Column
    public String getAddress() {
        return address;
    }

    public void setAddress(String address) {
        this.address = address;
    }

    @Column
    public Date getBirthday() {
        return birthday;
    }

    public void setBirthday(Date birthday) {
        this.birthday = birthday;
    }

    @Column(name = "register_time")
    @Temporal(TemporalType.DATE)
    public Date getRegisterTime() {
        return registerTime;
    }

    public void setRegisterTime(Date registerTime) {
        this.registerTime = registerTime;
    }

    @Override
    public boolean equals(Object o) {

        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        User user = (User) o;
        return id == user.id &&
                Objects.equals(username, user.username) &&
                Objects.equals(password, user.password) &&
                Objects.equals(nickname, user.nickname) &&
                gender == user.gender &&
                Objects.equals(address, user.address) &&
                Objects.equals(birthday, user.birthday) &&
                Objects.equals(registerTime, user.registerTime);
    }

    @Override
    public int hashCode() {
        return Objects.hash(id, username, password, nickname, gender, address, birthday, registerTime);
    }

    @Override
    public String toString() {
        return "User{" +
                "id=" + id +
                ", username='" + username + '\'' +
                ", password='" + password + '\'' +
                ", nickname='" + nickname + '\'' +
                ", gender=" + gender +
                ", address='" + address + '\'' +
                ", birthday=" + birthday +
                ", registerTime=" + registerTime +
                '}';
    }
}

使用到的注解如下:

  • 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中添加一句,表示我们使用注解配置了某个类。

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

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

CRUD

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

添加

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

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

    @Test
    public void testAdd() {
        User user = new User();
        user.setUsername("yitian");
        user.setPassword("12345678");
        user.setBirthday(new Date());
        user.setGender(Gender.MALE);
        user.setRegisterTime(new Date());

        try (Session session = factory.openSession()) {
            session.beginTransaction();
            session.save(user);
            session.getTransaction().commit();

        }

        try (Session session = factory.openSession()) {
            Query<User> query = session.createQuery("from User");
            List<User> users = query.list();
            Assert.assertNotNull(users);
        }
    }

查询

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

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

更新

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

    @Test
    public void testUpdate() {
        try (Session session = factory.openSession()) {
            session.beginTransaction();
            User u = session.bySimpleNaturalId(User.class).load("zhang3");
            u.setEmail("zhang3m@yitian.com");
            session.update(u);
            session.getTransaction().commit();
            u = session.bySimpleNaturalId(User.class).load("zhang3");
            Assert.assertEquals(u.getEmail(), "zhang3m@yitian.com");

        }
    }

删除

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

    @Test
    public void testDelete() {
        User user = new User();
        user.setUsername("wuli");
        user.setNickname("呜呜呜");
        try (Session session = factory.openSession()) {
            session.beginTransaction();
            session.save(user);
            session.getTransaction().commit();

            User u = session.bySimpleNaturalId(User.class).load("wuli");
            logger.debug(u.toString());
            Assert.assertNotNull(u);
            session.beginTransaction();
            session.delete(u);
            session.getTransaction().commit();
            u = session.bySimpleNaturalId(User.class).load("wuli");
            Assert.assertNull(u);
        }
    }

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

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,980评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,178评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,868评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,498评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,492评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,521评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,910评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,569评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,793评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,559评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,639评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,342评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,931评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,904评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,144评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,833评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,350评论 2 342

推荐阅读更多精彩内容