Hibernate入门

慕课网Hibernate学习笔记

开发环境

  1. eclipse mars
  2. hibernate toos

最简单的例子

  1. 创建java Project

  2. 导入相关依赖

  • 导入hibernate-re...\lib\required中的jar包
  • 导入mysql驱动
  • 导入Junit4包
  1. 创建hibernate.cfg.xml配置文件
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-configuration PUBLIC
        "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
    <session-factory>
        <property name="connection.user">root</property>
        <property name="connection.password">root</property>
        <property name="connection.driver_class">com.mysql.jdbc.Driver</property>
        <property name="connection.url">jdbc:mysql://localhost:3306/hibernate?useUnicode=true&characterEncoding=UTF-8</property>
        <property name="dialect">org.hibernate.dialect.MySQLDialect</property>
        
        <property name="show_sql">true</property>
        <property name="format_sql">true</property>
        <property name="hbm2ddl.auto">update</property>
    </session-factory>
</hibernate-configuration>
  1. 创建持久化类(javabean)
  • 公有
  • 提供不带参数的的构造方法
  • 属性私有
  • 有setter和getter
public class Student {
    private int id;
    private String name;
    private String sex;
    private int age;

    public Student() {
    }

    public Student(int id, String name, String sex, int age) {
        this.id = id;
        this.name = name;
        this.sex = sex;
        this.age = age;
    }
    public int getId() {
        return id;
    }

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

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getSex() {
        return sex;
    }

    public void setSex(String sex) {
        this.sex = sex;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "Student [id=" + id + ", name=" + name + ", sex=" + sex + ", age=" + age + "]";
    }

}

  1. 创建对象关系映射文件(*.hbm.xml)
    <?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<!-- Generated 2016-10-2 16:35:18 by Hibernate Tools 3.5.0.Final -->
<hibernate-mapping>
    <class name="Student" table="STUDENT">
        <id name="id" type="int">
            <column name="ID" />
            <generator class="assigned" />
        </id>
        <property name="name" type="java.lang.String">
            <column name="NAME" />
        </property>
        <property name="sex" type="java.lang.String">
            <column name="SEX" />
        </property>
        <property name="age" type="int">
            <column name="AGE" />
        </property>
    </class>
</hibernate-mapping>

  1. 将映射添加到hibernate.cfg.xml
 <mapping resource="Student.hbm.xml"/>
  1. 创建JUnit测试类
public class StudentTest {
    
    private SessionFactory sessionFactory;
    private Session session;
    private Transaction transaction;
    
    @Before 
    public void init(){
        StandardServiceRegistry registry = new StandardServiceRegistryBuilder().configure("hibernate.cfg.xml").build();
        sessionFactory = new MetadataSources(registry).buildMetadata().buildSessionFactory();
        session = sessionFactory.openSession();
        transaction = session.beginTransaction();
    }
    
    @After
    public void destroy(){
        transaction.commit();
        session.close();
        sessionFactory.close();
    }
    
    @Test
    public void testSaveStudent() {
        Student student = new Student(2, "老王", "男", 13);
        session.save(student);
    }
}

  1. 运行测试即可将数据存入数据库

Hibernate相关介绍

hibernat.cfg.xml属性介绍

  1. hibernat.show_sql:(true|false)是否输出sql语句到控制台

  2. hibernat.format_sql: (true|false)是否对输出的sql语句进行排版

  3. hbm2ddl.auto:(create|update|create-drop|va|idate)
    create每次插入都会删除原来的表,update会更新原来的表

  4. hibernate.default_schema:(hibernate)设置默认数据库,会在sql语句中的表名前加入前缀

  5. hibernate.dialect:(org.hibernate.dialect.MySQLDialect)数据库方言,可针对特定数据库进行优化

  6. hibernate.current_session_contex_class:(thread|jta):
    用于指定getCurrentSession是获取本地事务还是全局事务

session

  • SessionFactory一个系统一般只创建一个

  • openSession每次获取的事务时不同事务

  • getCurrentSession获取的是同一个事务,会自动关闭

  • 通过session进行增删改查

  • 操作前需开启transaction,完成后提交transaction

*.hbm.xml配置文件介绍

<hibernate-mapping
schema="schema_name"
catalog="catalog_name"//目录mysql不支持
default_cascade="cascade_style"//级联风格
default_access="field|property|ClassName"//访问策略
default-lazy="true|false"//加载策略
package="packageName"
>
<id>
<generator class="assigned|native|increment"/>
</id>
//组件属性
<component name="address" class=""> 
    <property name="" column></property>
</component>

增删改查

  1. save
  2. update
  3. delete
  4. get //一次性获取
  5. load // 懒加载

一对多

  1. 实体类中设置多方的set

  2. 在一方的映射文件中设置

<set name="students" table="student" lazy="true" inverse="true" cascade="all">//inverse=true表示由多方维护关联关系(多方add元素后,插入时不会出现update)
//cascade="all|save-update|delete|none"
    <key column="gid"></key>
    <one-to-many class="student">
</set>

多对一

  1. 在实体类中设置一方的实例

  2. 在多方映射文件中设置

<many-to-one name="grade" class="Grade" conlumn = "gid"></many-to-one>

HQL语句

select xxx from xxx where xxx group by xxx having xxx order by xxx

使用方法

  1. 大小写敏感

  2. 简单例子

    @Test
    public void testQuery(){
        String hql = "from Student as s";
        List<Student> students = session.createQuery(hql).list();
        for(Student student : students){
            System.out.println(student);
        }
    }
  1. 返回Object[]
//只选择一个属性则返回object
    @Test
    public void testSelect(){
        String hql = "select s.id, s.name from Student as s";
        Query<Object[]> query = session.createQuery(hql);
        List<Object[]> list = query.list();
        for(Object[] objects : list){
            System.out.println(objects[0]);
            System.out.println(objects[1]);
        }
    }   
  1. 返回list
    @Test
    public void testSelectWithList(){
        String hql = "select new list(s.id, s.name) from Student as s";
        Query query = session.createQuery(hql);
        List<List> lists = query.list();
        for(List list : lists){
            System.out.println(list.get(0));
            System.out.println(list.get(1));
        }
    }
  1. 返回自定义类
//需要有相应的构造方法
String hql = "select new Student(s.id, s.name) from Student as s"
List<Student> lists = query.list();

  1. 没有指定构造器会调用默认构造器即无参构造函数

  2. 集合运算"from grade g where g.students is not empty";

  3. 获取单个对象

    @Test
    public void testGetOne(){
        String hql="from Student as s where s.id=1";
        Query query = session.createQuery(hql);
        Student student = (Student) query.uniqueResult();
        System.out.println(student.getName());
    }

Hibernate 缓存

一级缓存
  1. 与session相同的生命周期

  2. 示例代码

    @Test
    public void testCache(){
        Student student1 = session.get(Student.class, 1);
        //清除指定对象缓存
        session.evict(student1);
        //清除所有缓存
        session.clear();
        Student student2 = session.get(Student.class, 1);
        //list()不使用缓存
        String hql = "from Student";
        Query query = session.createQuery(hql);
        List<Student> students = query.list();
        //iterator 先从数据库获取所有id,判断缓存中是否有对应记录,若有则直接使用缓存
        Iterator iterator = query.iterate();
        while(iterator.hasNext()){
            Student student = (Student) iterator.next();
            System.out.println(student.getName());
        }
    }

Hibernate注解

类级别注解
  1. @Entity(name="tableName")
  2. @table(name="", catalog="", schema="")
  3. @Embeddable // 在嵌入类上添加(如Address类)
属性级别注解

可写在属性上也可以写在get方法上

  1. @Id
    //多个属性Id实体类需要实现serializable接口
    //字符串主键需要通过@Column指定长度

  2. @GeneratedValue(strategy=(), generator="")
    //主键生成策略

@GeneratedValue(generator="id")
@GenericGenerator(name="id", strategy="assigned")
  1. @Column(length="", name="", nullable="", insertable="", unique="", updateable="")
  2. @Embedded
    //嵌入类需要@Embeddable
  3. @EmbeddedId
    //嵌入类(需要@Embeddable,实现Serializable接口,有无参构造方法,覆盖equals,hashCode)作为主键
  4. @Transient
    //忽略某个字段,不映射到数据库
关联映射注解
  1. @OneToOne(cascade=CascadeType.ALl, mappedBy="id")
    //mappedBy设置主控放
    @JoinColumn(name="id", unique=true)

  2. @ManyTOOne(cascade={CascadeType.ALL} fetch=FetchType.EAGER)
    //fetch抓取策略默认积极
    @JoinColumn(name="gid", referenceColumnName="gid")

  3. @OneToMany(cascade={CascadeType.ALL} fetch=FetchType.LAZY)
    @JoinColumn(name="gid")

  4. @ManyToMany(mappedBy="")
    @JoinTable(name="",JoinColumns="",
    inverseJoinColumns={@JoinColumn(name="tid")})

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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

推荐阅读更多精彩内容