团长大人的学习笔记——Hibernate

Hibernate开发流程

  • 导入Maven依赖(还有数据库驱动)
<dependencies>
    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-core</artifactId>
        <version>5.2.10.Final</version>
    </dependency>
    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-entitymanager</artifactId>
        <version>5.2.10.Final</version>
    </dependency>
    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-osgi</artifactId>
        <version>5.2.10.Final</version>
    </dependency>
    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-c3p0</artifactId>
        <version>5.2.10.Final</version>
    </dependency>
    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-proxool</artifactId>
        <version>5.2.10.Final</version>
    </dependency>
    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-infinispan</artifactId>
        <version>5.2.10.Final</version>
    </dependency>
    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-ehcache</artifactId>
        <version>5.2.10.Final</version>
    </dependency>
</dependencies>
复制代码
  • 建立核心配置文件和Mapper的xml文件里java

    • 配置Xml头在核心包(core)里的dtd文件里找到,以下
    <!DOCTYPE hibernate-configuration PUBLIC
    	"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
    	"http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd"><
    复制代码
    <!DOCTYPE hibernate-mapping PUBLIC 
        "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
    复制代码
    • 建立核心配置文件
    <!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="hibernate.connection.driver_class">com.mysql.cj.jdbc.Driver</property>
            <property name="connection.url">jdbc:mysql://localhost:3306/money2?userSSL=true&amp;useUnicode=true&amp;characterEncoding=UTF8&amp;serverTimezone=GMT&amp;allowMultiQueries=true</property>
            <property name="connection.username">root</property>
            <property name="connection.password">root</property>
            <!--配置SQL方言-->
            <property name="hibernate.dialect">org.hibernate.dialect.MySQL57Dialect</property>
            <!--自动建立表(通常不须要)-->
            <property name="hbm2ddl.auto">update</property>
            <!--显示SQL-->
            <property name="show_sql">true</property>
            <property name="format_sql">true</property>
            <!--配置数据实体-->
    
            <!--注册mapper-->
            <mapping resource="mapping/AccUser.hbm.xml"/>
        </session-factory>
    </hibernate-configuration>
    复制代码
    • 建立实体类和映射文件
    <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
    <hibernate-mapping>
        <class name="com.libi.entity.AccUser" table="acc_user">
            <!--对象标识OID-->
            <id name="id" column="id">
                <generator class="identity"/>
            </id>
            <!--属性映射-->
            <property name="userName" column="user_name"/>
            <property name="password" column="password"/>
            <property name="createTime" column="create_time"/>
        </class>
    </hibernate-mapping>
    复制代码
    • 在核心配置文件的<session-factory>标签里注册mapping
    <!--注册mapper-->
    <mapping resource="mapping/AccUser.hbm.xml"/>
    复制代码
  • 添加主类进行测试mysql

public class Application {
    public static void main(String[] args) {
        Configuration configuration = new Configuration().configure("hibernate.cfg.xml");
        SessionFactory sessionFactory = configuration.buildSessionFactory();
        Session session = sessionFactory.openSession();
        Transaction transaction = session.getTransaction();
        transaction.begin();
        AccUser user = new AccUser();
        user.setUserName("hibernate");
        user.setPassword("1233333");
        user.setCreateTime(System.currentTimeMillis());
        session.save(user);
        transaction.commit();
        session.close();
        sessionFactory.close();
    }
}
复制代码

Hibernate主键生成策略

  • 配置位置:在Mapping配置文件里配置
<class name="com.libi.entity.AccUser" table="acc_user">
    <!--对象标识OID-->
    <id name="id" column="id">
        <generator class="具体策略"/>
    </id>
</class>
复制代码
  • 生成策略
    • identity:采用数据库底层的自增加列(MySQL)
    • sequence:采用数据库底层的自增加列(Oracle)
    • native:根据数据库选择identity,sequence或hilo中的一个
    • increment:采用Hibernate本身维护的自增加(先查询max,再+1)
    • uuid:采用UUID做为惟一字符串(id字段必须是string类型)
    • assigned:须要明确赋值(开发者维护)

Hibernate的自动模式(生成映射文件和实体类)

  • 在IDEA里配置Database(右面maven上面的按钮)
  • 添加Hibernate插件的支持

HQL

  • HQL = Hibernate Query Language,主要用面向对象的思惟来编写SQL。下面的示例就能够查询全部的User,而且实现分页
public List<AccUser> selectAllUser() {
    Configuration configuration = new Configuration().configure("hibernate.cfg.xml");
    SessionFactory sessionFactory = configuration.buildSessionFactory();
    Session session = sessionFactory.openSession();
    Query<AccUser> query = session.createQuery("select au from AccUser au", AccUser.class);
    query.setFirstResult(0);
    query.setMaxResults(10);
    return query.getResultList();
}
复制代码
  • 条件查询sql

    • 索引占位(下面的例子是查询Id是传入数字的AccUser)
    public AccUser selectUserById(Long uid) {
        Configuration configuration = new Configuration().configure("hibernate.cfg.xml");
        SessionFactory sessionFactory = configuration.buildSessionFactory();
        Session session = sessionFactory.openSession();
        Query<AccUser> query = session.createQuery("select au from AccUser au where id = ?", AccUser.class);
        //传入占位的ID实现查询
        query.setParameter(0, uid);
        //获取单个结果
        return query.uniqueResult();
    }
    复制代码
    • 命名占位(只须要在上例的基础上修改hql语句和传入参数的语句)
    ...
    Query<AccUser> query = session.createQuery("select au from AccUser au where id = :id", AccUser.class);
    ...
    query.setParameter("id", uid);
    ...
    复制代码
  • 模糊查询数据库

    • 使用百分号包起来作字符串部分匹配"%查询内容%"
  • HQL更新和删除(相似SQL更新和删除)数组

  • HQL查询部分字段缓存

    • 只查询一个字段,在后面加上这个字段的类型
    Query userName = session.createQuery("select userName from AccUser where id = 1",String.class);
    复制代码
    • 不知道这是什么类型或者查询两个以上的字段,使用Object数组(Hibernate会用Object数组的方式返回给你)
    Query userName = session.createQuery("select userName, password from AccUser where id = 1");
    复制代码

解决传递当前Session问题

  • 保存到本地线程bash

    • 在核心配置xml里配置容许保存到本地线程
    <property name="hibernate.current_session_context_class">thread</property>
    复制代码
    • 在线程里获取session (使用这种session必须开启事务而且不用关闭session,如今这个session会和这个线程共存亡)
    Session currentSession = sessionFactory.getCurrentSession();
    复制代码

级联查询

  • Hibernate能够实现不经过写SQL或HQL语句拿到字段里外键约束对应的记录的映射对象(关联信息默认是懒加载的)
  • 具体作法能够在IDEA的逆向生成工具生成

延迟检索

  • 至关我使用这个对象时才会查询数据库
    • 使用session.get(class)是当即检索(先查session缓存再查数据库)
    • 使用session.load(calss)时延迟检索(先查session缓存再查二级缓存最后查数据库)
  • 关联表的延迟与当即(在<many-to-noe>默标签里添加lazy属性,认是延迟lazy = true

实体类的状态

  • 瞬时态:没有主键,没有被session管理(实体类刚刚被建立)
  • 持久态:有主键,被session管理(实体类被session作save、get等操做,如今实体类被session管理)
  • 游离态:有主键,可是没有被session管理(session被关闭)

缓存

  • 一级缓存:
    • 缓存在session中,若是修改了缓存内容(持久态的对象被修改)而且提交了事务以后,Hibernate会进行快照对比,如过不一致,会同步到数据库
    • 能够调用clear方法清空全部一级缓存
  • 二级缓存
相关文章
相关标签/搜索