package com.baidu.test;

import java.util.ArrayList;
import java.util.LinkedHashSet;
import java.util.List;

import org.hibernate.Query;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.Transaction;
import org.hibernate.cfg.Configuration;
import org.hibernate.service.ServiceRegistry;
import org.hibernate.service.ServiceRegistryBuilder;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;

import com.baidu.leftJoin.Department;
import com.baidu.leftJoin.Employee;

public class TestHQL_LeftJoin {
    
    private SessionFactory sessionFactory;
    private Session session;
    private Transaction transaction;
    
    
    @Before
    public void init(){
        Configuration configuration = new Configuration().configure();
        ServiceRegistry serviceRegistry = new ServiceRegistryBuilder()
                                        .applySettings(configuration.getProperties())
                                        .buildServiceRegistry();
        
        sessionFactory = configuration.buildSessionFactory(serviceRegistry);
        
        session = sessionFactory.openSession();
        transaction = session.beginTransaction();
    }
    @After
    public void destroy(){
        transaction.commit();
        session.close();
        sessionFactory.close();
        
    }
    
//  ~~~~~~~~~~~~~~~~~~~~~~~~~~下面的例子是 从 1 对  多   ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    
    /**
     *
     * 迫切左外连接: 特点是:如果左表有不满足条件的,也返回左表不满足条件
     *        1. LEFT JOIN FETCH 关键字表示迫切左外连接检索策略.
     *        2. list() 方法返回的集合中存放实体对象的引用, 每个 Department 对象关联的 Employee 集合都被初始化,
     *             存放所有关联的 Employee 的实体对象.
     *        3. 查询结果中可能会包含重复元素, 可以通过一个 HashSet 来过滤重复元素
     *
     *         去重:
     *             方法一:使用 distinct
     *                 String hql  = "SELECT DISTINCT d FROM  Department d LEFT JOIN FETCH d.emps ";
     *                Query query = session.createQuery(hql);
     *
     *                List<Department> depts = query.list();
     *                System.out.println(depts.size());
     *             
     *             方法二
     *                  String hql  = "FROM  Department d LEFT JOIN FETCH d.emps ";
     *                Query query = session.createQuery(hql);
     *
     *                List<Department> depts = query.list();
     *
     *                depts = new ArrayList<>(new LinkedHashSet(depts));
     *                System.out.println(depts.size());
     *                
     *                for(Department dept:depts){
     *                    System.out.println(dept.getName() + "--" + dept.getEmps().size() );
     *                }
     *
     *
     */
    @Test
    public void testLeftJoinFetch(){
//        String hql  = "SELECT DISTINCT d FROM  Department d LEFT JOIN FETCH d.emps ";
//        Query query = session.createQuery(hql);
//        
//        List<Department> depts = query.list();
//        System.out.println(depts.size());
//        
        
        
        String hql  = "FROM  Department d LEFT JOIN FETCH d.emps ";
        Query query = session.createQuery(hql);
        
        
        List<Department> depts = query.list();
        System.out.println(depts.size());
        
        depts = new ArrayList<>(new LinkedHashSet(depts));
        System.out.println(depts.size());
        
        for(Department dept:depts){
            System.out.println(dept.getName() + "--" + dept.getEmps().size() );
        }
    }
    
    
    /**
     * 左外连接:
     *        1. LEFT JOIN 关键字表示左外连接查询.
     *        2. list() 方法返回的集合中存放的是对象数组类型
     *        3. 根据配置文件来决定 Employee 集合的检索策略.
     *        4. 如果希望 list() 方法返回的集合中仅包含 Department 对象,
     *            可以在HQL 查询语句中使用 SELECT 关键字
     *        
     *        这样的语句查询的结果有重复:
     *            String hql = "FROM Department d LEFT JOIN d.emps";
     *            Query query = session.createQuery(hql);
     *        
     *            List<Object[]> results = query.list();
     *            System.out.println(results.size());
     *    
     *         去重:
     *             仅能使用  distinct 的方法去除重复
     *     
     *             String hql = "SELECT DISTINCT d FROM Department d LEFT JOIN d.emps";
     *             Query query = session.createQuery(hql);
     *
     *             List<Department> depts = query.list();
     *             System.out.println(depts.size());
     *                 
     *             for(Department dept:depts){
     *                 System.out.println(dept.getName() + dept.getEmps().size());
     *             }
     *
     */
    @Test
    public void testLeftJoin(){
        String hql = "SELECT DISTINCT d FROM Department d LEFT JOIN d.emps";
        Query query = session.createQuery(hql);
        
        List<Department> depts = query.list();
        System.out.println(depts.size());
        
        for(Department dept:depts){
            System.out.println(dept.getName() + dept.getEmps().size());
        }        
        
    }
    
    /**
     * 迫切内连接: 特点是:不返回左表不满足条件
     *        INNER JOIN FETCH 关键字表示迫切内连接, 也可以省略 INNER 关键字
     *        list() 方法返回的集合中存放 Department 对象的引用, 每个 Department
     *                对象的 Employee 集合都被初始化, 存放所有关联的 Employee 对象
     *
     * 内连接:
     *        INNER JOIN 关键字表示内连接, 也可以省略 INNER 关键字
     *        list() 方法的集合中存放的每个元素对应查询结果的一条记录, 每个元素都是对象数组类型
     *        如果希望 list() 方法的返回的集合仅包含 Department  对象, 可以在 HQL 查询语句中使用 SELECT 关键字
     *
     *
     *
     */
    @Test
    public void testInnerJoinFetch(){
        //String hql  = "SELECT DISTINCT d FROM  Department d LEFT JOIN FETCH d.emps ";
        String hql  = "FROM  Department d INNER JOIN FETCH  d.emps ";
        Query query = session.createQuery(hql);
        
        
        List<Department> depts = query.list();
        depts = new ArrayList<>(new LinkedHashSet(depts));
        System.out.println(depts.size());
        
        for(Department dept:depts){
            System.out.println(dept.getName() + "--" + dept.getEmps().size() );
        }
    }
    
    
//  ~~~~~~~~~~~~~~~~~~~~~~~~~~下面的例子是 从多 对  1   ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    
    @Test
    public void testLeftJoinFetch2(){
        String hql = "FROM Employee e LEFT JOIN FETCH e.dept";
        Query query = session.createQuery(hql);
        
        List<Employee> emps = query.list();
        System.out.println(emps.size());
        
        for(Employee emp:emps){
            System.out.println(emp + " -- " + emp.getDept());
        }
        
    }
}

Hibernate 迫切连接和普通连接的区别的更多相关文章

  1. Http 和TCP的关系,TCP长连接和短连接有什么区别?

    HTTP 协议即超文本传送协议(Hypertext Transfer Protocol ),是Web联网的基础,也是手机联网常用的协议之一,HTTP协议是建立在TCP协议之上的一种应用.由于HTTP在 ...

  2. UrlConnection连接和Socket连接的区别

    关于UrlConnection连接和Socket连接的区别,只知道其中的原理如下: 抽象一点的说,Socket只是一个供上层调用的抽象接口,隐躲了传输层协议的细节. urlconnection 基于H ...

  3. SQL Server中内连接和外连接的区别

    SQL Server中内连接和外连接的区别 假设一个数据库中有两张表,一张是学生表StudentInfo,一张是班级表ClassInfo,两张表之间用ClassId字段进行关联. 如果用内连接,正常的 ...

  4. TCP/IP系列——长连接与短连接的区别

    1 什么是长连接和短连接       三次握手和四次挥手   TCP区别于UDP最重要的特点是TCP必须建立在可靠的连接之上,连接的建立和释放就是握手和挥手的过程. 三次握手为连接的建立过程,握手失败 ...

  5. mysql(3)—— 内连接、外连接的区别

    先来看一下,内连接的语法: SELECT  XXX FROM XXX INNER JOIN XXX ON XXX; 这里 INNER 可以省略,在上一篇博客中我们对于笛卡尔积现象的研究中(http:/ ...

  6. sql左外连接和右外连接的区别例子转摘

    sql左外连接和右外连接的区别   两个表:A(id,name)数据:(1,张三)(2,李四)(3,王五)B(id,name)数据:(1,学生)(2,老师)(4,校长) 左连接结果:select A. ...

  7. (转载)http和socket之长连接和短连接区别

    TCP/IPTCP/IP是个协议组,可分为三个层次:网络层.传输层和应用层.在网络层有IP协议.ICMP协议.ARP协议.RARP协议和BOOTP协议.在传输层中有TCP协议与UDP协议.在应用层有: ...

  8. HTTP协议原理(长连接,短连接/ get,post区别等等)

    HTTP简介 HTTP协议是Hyper Text Transfer Protocol(超文本传输协议)的缩写,是用于从万维网(WWW:World Wide Web )服务器传输超文本到本地浏览器的传送 ...

  9. http和socket之长连接和短连接区别【转】

    转自:https://blog.csdn.net/mengyafei43/article/details/25195445 TCP/IP TCP/IP是个协议组,可分为三个层次:网络层.传输层和应用层 ...

  10. sql左外连接、右外连接、group by、distinct(区别)、intersect(交叉)、通配符、having

    连接条件可在FROM或WHERE子句中指定,建议在FROM子句中指定连接条件.WHERE和HAVING子句也可以包含搜索条件,以进一步筛选连接条件所选的行.             连接可分为以下几类 ...

随机推荐

  1. vim列编辑

    命令模式下:ctrl + v(我在gvim,win7中是ctrl +shift + q)进入列编辑模,选中要编辑的行(j 上,k下) 输入 “I” (大写的 I,光标定位到选中的第一行),输入要编辑的 ...

  2. javaEE开发中使用session同步和token机制来防止并发重复提交

    javaEE开发中使用session同步和token机制来防止并发重复提交 通常在普通的操作当中,我们不需要处理重复提交的,而且有很多方法来防止重复提交.比如在登陆过程中,通过使用redirect,可 ...

  3. js原生继承之——构造函数式继承实例

    <!DOCTYPE html><html lang="en"><head>    <meta charset="UTF-8&qu ...

  4. 关于自己封装Web前端框架的思考和探索

    一.引言 首先这些年关于前端技术层出不穷,从最早的只用js做简单验证,到现在发现好像大前端已经无所不能了的感觉.特别是为了降低前端开发复杂度,涌现了一大批 的MVC/MVVM模式的前端框架,不停了刷新 ...

  5. Bootstrap3网上api文档地址

    http://v3.bootcss.com/css/#forms http://www.ziqiangxuetang.com/bootstrap/bootstrap-forms.html 另附加fa字 ...

  6. Struts2 属性驱动、模型驱动、异常机制

    模型驱动使用单独的VO(值对象)来封装请求参数和处理结果,属性驱动则使用Action实例来封装请求参数和处理结果. 一.使用模型驱动 1.login.action采用模型驱动模式实现,采用模型驱动时必 ...

  7. 有关extdelete恢复测试

    客户意外rm掉了数据文件,导致数据库无法打开,由于没有完整的备份和归档,需要使用别的方法,而客户又关闭了数据库,导致无法使用文件描述符恢复,就要使用linux上别的方法了,现记录使用extundele ...

  8. AngularJS的五个超酷特性

    AngularJS是一个超棒的javascript框架,不单单对于开发人员来说非常有吸引力,对于UI设计师来说也同样出色.在这篇教程中,我们将简单的介绍AngularJS几个重量级必备特性,并且介绍它 ...

  9. 《RDLC部署》RDLC部署到IIS缺少DLL程序集

    1.错误:从vs生成网站部署到服务器后打开RDLC报表却提示缺少DLL程序集. 一般是缺少如下文件 1. Microsoft.ReportViewer.Common.dll 2.   Microsof ...

  10. EM算法 大白话讲解

    假设有一堆数据点,它是由两个线性模型产生的.公式如下: 模型参数为a,b,n:a为线性权值或斜率,b为常数偏置量,n为误差或者噪声. 一方面,假如我们被告知这两个模型的参数,则我们可以计算出损失. 对 ...