Hibernate构架(四) 查询,连接池,二级缓存,session管理

1.查询

  1. Get/load主键查询
  1. 对象导航查询
  2. HQL查询, Hibernate Query language hibernate 提供的面向对象的查询语言。
  3. Criteria 查询, 完全面向对象的查询(Query By Criteria ,QBC)
  4. SQLQuery, 本地SQL查询
    缺点:不能跨数据库平台: 如果该了数据库,sql语句有肯能要改
    使用场景: 对于复杂sql,hql实现不了的情况,可以使用本地sql查询

1.1 HQL查询

public class App {
    
    private static SessionFactory sf;
    static {
        sf = new Configuration()
            .configure()
            .addClass(Dept.class)   
            .addClass(Employee.class)   // 测试时候使用
            .buildSessionFactory();
    }

    /*
     * 1)   Get/load主键查询
        2)  对象导航查询
        3)  HQL查询,  Hibernate Query language  hibernate 提供的面向对象的查询语言。
        4)  Criteria 查询,   完全面向对象的查询(Query By Criteria  ,QBC)
        5)  SQLQuery, 本地SQL查询

     */
    @Test
    public void all() {
        
        Session session = sf.openSession();
        session.beginTransaction();
        
        //1) 主键查询
//      Dept dept =  (Dept) session.get(Dept.class, 12);
//      Dept dept =  (Dept) session.load(Dept.class, 12);
        
        //2) 对象导航查询
//      Dept dept =  (Dept) session.get(Dept.class, 12);
//      System.out.println(dept.getDeptName());
//      System.out.println(dept.getEmps());
        
        // 3)   HQL查询
        // 注意:使用hql查询的时候 auto-import="true" 要设置true,
        //  如果是false,写hql的时候,要指定类的全名
//      Query q = session.createQuery("from Dept");
//      System.out.println(q.list());
        
        // a. 查询全部列
//      Query q = session.createQuery("from Dept");  //OK
//      Query q = session.createQuery("select * from Dept");  //NOK, 错误,不支持*
//      Query q = session.createQuery("select d from Dept d");  // OK
//      System.out.println(q.list());

        // b. 查询指定的列  【返回对象数据Object[] 】
//      Query q = session.createQuery("select d.deptId,d.deptName from Dept d");  
//      System.out.println(q.list());
        
        // c. 查询指定的列, 自动封装为对象  【必须要提供带参数构造器】
//      Query q = session.createQuery("select new Dept(d.deptId,d.deptName) from Dept d");  
//      System.out.println(q.list());
        
        // d. 条件查询: 一个条件/多个条件and or/between and/模糊查询
        // 条件查询: 占位符
//      Query q = session.createQuery("from Dept d where deptName=?");
//      q.setString(0, "财务部");
//      q.setParameter(0, "财务部");
//      System.out.println(q.list());
        
        // 条件查询: 命名参数
//      Query q = session.createQuery("from Dept d where deptId=:myId or deptName=:name");
//      q.setParameter("myId", 12);
//      q.setParameter("name", "财务部");
//      System.out.println(q.list());
        
        // 范围
//      Query q = session.createQuery("from Dept d where deptId between ? and ?");
//      q.setParameter(0, 1);
//      q.setParameter(1, 20);
//      System.out.println(q.list());
        
        // 模糊
//      Query q = session.createQuery("from Dept d where deptName like ?");
//      q.setString(0, "%部%");
//      System.out.println(q.list());
        

        // e. 聚合函数统计
//      Query q = session.createQuery("select count(*) from Dept");
//      Long num = (Long) q.uniqueResult();
//      System.out.println(num);
        
        // f. 分组查询
        //-- 统计t_employee表中,每个部门的人数
        //数据库写法:SELECT dept_id,COUNT(*) FROM t_employee GROUP BY dept_id;
        // HQL写法
//      Query q = session.createQuery("select e.dept, count(*) from Employee e group by e.dept");
//      System.out.println(q.list());
        
    
        
        
        session.getTransaction().commit();
        session.close();
    }
    
    // g. 连接查询
    @Test
    public void join() {
        
        Session session = sf.openSession();
        session.beginTransaction();
        
        //1) 内连接   【映射已经配置好了关系,关联的时候,直接写对象的属性即可】
//      Query q = session.createQuery("from Dept d inner join d.emps");
        
        //2) 左外连接
//      Query q = session.createQuery("from Dept d left join d.emps");

        //3) 右外连接
        Query q = session.createQuery("from Employee e right join e.dept");
        q.list();
        
        session.getTransaction().commit();
        session.close();
    }
    
    // g. 连接查询 - 迫切连接
    @Test
    public void fetch() {
        
        Session session = sf.openSession();
        session.beginTransaction();
        
        //1) 迫切内连接    【使用fetch, 会把右表的数据,填充到左表对象中!】
//      Query q = session.createQuery("from Dept d inner join fetch d.emps");
//      q.list();
        
        //2) 迫切左外连接
        Query q = session.createQuery("from Dept d left join fetch d.emps");
        q.list();
        
        session.getTransaction().commit();
        session.close();
    }
    
    // HQL查询优化
    @Test
    public void hql_other() {
        Session session = sf.openSession();
        session.beginTransaction();
        // HQL写死
//      Query q = session.createQuery("from Dept d where deptId < 10 ");
        
        // HQL 放到映射文件中
        Query q = session.getNamedQuery("getAllDept");
        q.setParameter(0, 10);
        System.out.println(q.list());
        
        session.getTransaction().commit();
        session.close();
    }
}

1.2 Criteria 查询

//4)    Criteria 查询,
    @Test
    public void criteria() {
        
        Session session = sf.openSession();
        session.beginTransaction();
        
        Criteria criteria = session.createCriteria(Employee.class);
        // 构建条件
        criteria.add(Restrictions.eq("empId", 12));
//      criteria.add(Restrictions.idEq(12));  // 主键查询
        
        System.out.println(criteria.list());
        
        
        session.getTransaction().commit();
        session.close();

1.3 SQL查询

// 5)   SQLQuery, 本地SQL查询
    // 不能跨数据库平台: 如果该了数据库,sql语句有肯能要改。
    @Test
    public void sql() {
        
        Session session = sf.openSession();
        session.beginTransaction();
        
        SQLQuery q = session.createSQLQuery("SELECT * FROM t_Dept limit 5;")
            .addEntity(Dept.class);  // 也可以自动封装
        System.out.println(q.list());
        
        session.getTransaction().commit();
        session.close();
    }

1.4 分页查询

// 分页查询
    @Test
    public void all() {
        
        Session session = sf.openSession();
        session.beginTransaction();
        
         Query q = session.createQuery("from Employee");
         
         // 从记录数
         ScrollableResults scroll = q.scroll();  // 得到滚动的结果集
         scroll.last();                         //  滚动到最后一行
         int totalCount = scroll.getRowNumber() + 1;// 得到滚到的记录数,即总记录数
         
         // 设置分页参数
         q.setFirstResult(0);
         q.setMaxResults(3);
         
         // 查询
         System.out.println(q.list());
         System.out.println("总记录数:" + totalCount);
        
        session.getTransaction().commit();
        session.close();
    }

2.连接池

Hibernate 自带的也有一个连接池,且对C3P0连接池也有支持

2.1 Hbm 自带连接池

只维护一个连接,比较简陋。
可以查看hibernate.properties文件查看连接池详细配置:

【Hbm对C3P0连接池支持, 核心类】
告诉hib使用的是哪一个连接池技术。
hibernate.connection.provider_class org.hibernate.connection.C3P0ConnectionProvider

2.2 Hbm中的C3P0

Hibernate.cfg.xml 中增加连接池相关配置:

<!-- 【连接池配置】 -->
        <!-- 配置连接驱动管理类 -->
        <property name="hibernate.connection.provider_class">org.hibernate.connection.C3P0ConnectionProvider</property>
        <!-- 配置连接池参数信息 -->
        <property name="hibernate.c3p0.min_size">2</property>
        <property name="hibernate.c3p0.max_size">4</property>
        <property name="hibernate.c3p0.timeout">5000</property>
        <property name="hibernate.c3p0.max_statements">10</property>
        <property name="hibernate.c3p0.idle_test_period">30000</property>
        <property name="hibernate.c3p0.acquire_increment">2</property>

3.二级缓存

Hibernate提供了基于应用程序级别的缓存, 可以跨多个session,即不同的session都可以访问缓存数据。 这个换存也叫二级缓存。
Hibernate提供的二级缓存有默认的实现,且是一种可插配的缓存框架!如果用户想用二级缓存,只需要在hibernate.cfg.xml中配置即可; 不想用,直接移除,不影响代码。
如果用户觉得hibernate提供的框架框架不好用,自己可以换其他的缓存框架或自己实现缓存框架都可以。

举例

<!--****************** 【二级缓存配置】****************** -->
        <!-- a.  开启二级缓存 -->
        <property name="hibernate.cache.use_second_level_cache">true</property>
        <!-- b. 指定使用哪一个缓存框架(默认提供的) -->
        <property name="hibernate.cache.provider_class">org.hibernate.cache.HashtableCacheProvider</property>
        <!-- 开启查询缓存 -->
        <property name="hibernate.cache.use_query_cache">true</property>
        <!-- c. 指定哪一些类,需要加入二级缓存 -->
        <class-cache usage="read-write" class="cn.itcast.b_second_cache.Dept"/>
        <class-cache usage="read-only" class="cn.itcast.b_second_cache.Employee"/>
        <!-- 集合缓存[集合缓存的元素对象,也加加入二级缓存] -->
        <collection-cache usage="read-write" collection="cn.itcast.b_second_cache.Dept.emps"/>

测试:

public class App {  
    private static SessionFactory sf;
    static {
        sf = new Configuration()
            .configure()
            .addClass(Dept.class)   
            .addClass(Employee.class)   // 测试时候使用
            .buildSessionFactory();
    }
    // 1. 测试二级缓存的使用
    // 没有/有用 二级缓存
    @Test
    public void testCache() {
        Session session1 = sf.openSession();
        session1.beginTransaction();
        // a. 查询一次
        Dept dept = (Dept) session1.get(Dept.class, 5);
        dept.getEmps().size();// 集合
        session1.getTransaction().commit();
        session1.close();
        
        System.out.println("------");
        
        // 第二个session
        Session session2 = sf.openSession();
        session2.beginTransaction();
        // a. 查询一次
        dept = (Dept) session2.get(Dept.class, 5);  // 二级缓存配置好; 这里不查询数据库
        dept.getEmps().size();
        
        session2.getTransaction().commit();
        session2.close();
    }
    
    
    @Test
    public void listCache() {
        Session session1 = sf.openSession();
        session1.beginTransaction();
        // HQL查询  【setCacheable  指定从二级缓存找,或者是放入二级缓存】
        Query q = session1.createQuery("from Dept").setCacheable(true);
        System.out.println(q.list());
        session1.getTransaction().commit();
        session1.close();
        
        
        Session session2 = sf.openSession();
        session2.beginTransaction();
        q = session2.createQuery("from Dept").setCacheable(true);
        System.out.println(q.list());  // 不查询数据库: 需要开启查询缓存
        session2.getTransaction().commit();
        session2.close();
    }
}

4.session管理

public class App_session {

    private static SessionFactory sf;
    static {
        sf = new Configuration()
            .configure()
            .addClass(Dept.class)   
            .addClass(Employee.class)   // 测试时候使用
            .buildSessionFactory();
    }
    
    @Test
    public void testSession() throws Exception {
        //openSession:  创建Session, 每次都会创建一个新的session
        Session session1 = sf.openSession();
        Session session2 = sf.openSession();
        System.out.println(session1 == session2);//false
        session1.close();
        session2.close();
        
        //getCurrentSession 创建或者获取session
        // 线程的方式创建session  
        // 一定要配置:<property name="hibernate.current_session_context_class">thread</property>
        Session session3 = sf.getCurrentSession();// 创建session,绑定到线程
        Session session4 = sf.getCurrentSession();// 从当前访问线程获取session
        System.out.println(session3 == session4);//true
        
        // 关闭 【以线程方式创建的session,可以不用关闭; 线程结束session自动关闭】
        //session3.close();
        //session4.close(); 报错,因为同一个session已经关闭了!
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容