Hibernate进阶之如何使用Hql内连接,外连接以及自连接

Hibernate进阶之如何使用Hql内连接,外连接以及自连接

一、sql语句中的 内连接、自连接和外连接:

1、使用等值连接/内连接查询,查询客户姓名,订单编号,订单价格
等值连接/内连接:只能查询出符合条件的记录:
select c.name,o.orderno,o.price
from customers c,orders o
where c.id = o.customers_id; 

2、使用左外连接,按客户分组,查询每个客户的订单数:
select c.name,count(o.orderno)
from customers c left join orders o
on c.id = o.customers_id
group by c.name;

3、使用右外连接,按客户分组,查询每个客户的订单数:
select c.name,count(o.orderno)
from orders o right join customers c
on c.id = o.customers_id
group by c.name;

外连接:既能查询出符合条件的记录,同时不符合条件的记录也能查询出

4、等值连接语法:
select    字段
from      表名,表名 
where     等值连接条件;

外连接语法:
select    字段
from      表名1 left/right join 表名2
on        等值连接条件;
group by  字段

 

二、HQL实战:

1、使用等值连接查询,查询客户姓名,订单编号,订单价格:
    select c.name,o.orderno,o.price
    from Customer c join c.orderSet o
    where c.id = o.customer.id
2、使用左外连接,按客户分组,查询每个客户的姓名和订单数:
             select c.name,count(o.orderno)
             from Customer c left join c.orderSet o

             group  by c.name 

这里需要注意的是:左外链接中left join on后面跟着的条件省略不写,否则会报错,Hibernate会自动加上去,如果要添加条件可以使用with。

3、使用自连接,求出xx的老板是yy
    select a.name,b.name
    from  Emp a,Emp b 

    where a.mgr = b.id     

 

三、具体小案例:

首先写配置文件(CustomerOrder.hbm.xml,Emp.hbm.xml):

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC 
    "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
    "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="example.hql">
	<class name="Customer" table="customers">
		<id name="id" column="id" type="int">
			<generator class="native"></generator>
		</id>
		<property name="name" column="name" type="string"></property>
		<property name="age" column="age" type="int"></property>
		<!-- set标签用于映射单向一对多
			name表示单方的关联属性
			table表示多方对应表的名字
			key-cloumn表示多方对应表的外键
			one-to-many-class表示单方关联属性中的每个元素的类型
		 -->
		 <set name="orderSet" table="orders" cascade="all" inverse="true">
		 	<key column="customers_id"></key>
		 	<one-to-many class="Order"/>
		 </set>
	</class>
	<!-- 映射类的多方 -->
	<class name="Order" table="Orders">
		<id name="id" column="id">
			<generator class="native"></generator>
		</id>
		<property name="orderNo" column="orderNo"></property>
		<property name="price" column="price"></property>
		<many-to-one name="customer" column="customers_id"></many-to-one>
	</class>
	<query name="findCustomerByAge">
		<![CDATA[
			from Customer c where c.age>?
		]]>
	</query>
</hibernate-mapping>
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC 
    "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
    "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="example.hql">
	<class name="Emp" table="emps">
		<id name="id" column="id" type="int">
			<generator class="native"></generator>
		</id>
		<property name="name" column="name" ></property>
		<property name="age" column="sal" ></property>
		<property name="age" column="mgr" ></property>
	</class>
</hibernate-mapping>

接下来写实体类Customer,Order,Emp员工类:

Customer:

package example.hql;
 
import java.util.LinkedHashSet;
import java.util.Set;
 
/**
 * 客户(一方)
 * @author Administrator
 *
 */
public class Customer {
 
	private Integer id;//对应表的主键
	private String name;
	private Integer age;
	private Set<Order> orderSet=new LinkedHashSet<Order>();//关联属性
	public Customer() {
	}
	public Customer(String name, Integer age) {
		super();
		this.name = name;
		this.age = age;
	}
	public Integer getId() {
		return id;
	}
	public void setId(Integer id) {
		this.id = id;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public Integer getAge() {
		return age;
	}
	public void setAge(Integer age) {
		this.age = age;
	}
	public Set<Order> getOrderSet() {
		return orderSet;
	}
	public void setOrderSet(Set<Order> orderSet) {
		this.orderSet = orderSet;
	}
	
}

Order类:

package example.hql;
/**
 * 订单(多的一方)
 * @author Administrator
 *
 */
public class Order {
 
	private Integer id;
	private String orderNo;//订单编号
	private Integer price;//价格
	private Customer customer;//关联的属性
	
	
	public Order(String orderNo, Integer price, Customer customer) {
		super();
		this.orderNo = orderNo;
		this.price = price;
		this.customer = customer;
	}
	public Integer getId() {
		return id;
	}
	public void setId(Integer id) {
		this.id = id;
	}
	public Integer getPrice() {
		return price;
	}
	public void setPrice(Integer price) {
		this.price = price;
	}
	
	public String getOrderNo() {
		return orderNo;
	}
	public void setOrderNo(String orderNo) {
		this.orderNo = orderNo;
	}
	public Order() {
	}
	public Customer getCustomer() {
		return customer;
	}
	public void setCustomer(Customer customer) {
		this.customer = customer;
	}
	
}

Emp类:

package example.hql;
 
/**
 * 员工表
 * @author Administrator
 *
 */
public class Emp {
 
	private Integer id;
	private String name;
	private Integer sal;//薪水
	private Integer mgr;//直属领导编号
	public Integer getId() {
		return id;
	}
	public void setId(Integer id) {
		this.id = id;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public Integer getSal() {
		return sal;
	}
	public void setSal(Integer sal) {
		this.sal = sal;
	}
	public Integer getMgr() {
		return mgr;
	}
	public void setMgr(Integer mgr) {
		this.mgr = mgr;
	}
	public Emp() {
	}
	
	
}

最后实现内连接,外连接和自连接:

package example.hql;
 
import java.util.List;
 
import org.hibernate.Query;
import org.hibernate.Session;
import org.hibernate.Transaction;
import org.junit.Test;
 
import example.utils.HibernateUtils;
 
public class CustomerOrderDao2 {
 
	/**
	 * 使用等值连接查询,查询客户姓名,订单编号,订单价格
	 */
	@Test
	public void test01(){
		Session session=HibernateUtils.getSession();
		Transaction t=session.getTransaction();
		try{
			t.begin();
			String hql="select c.name,o.orderNo,o.price from Customer c join c.orderSet o where c.id=o.customer.id";
			Query query=session.createQuery(hql);
			List<Object[]> list= (List<Object[]>) query.list();
			//对象导航查询
			for(Object[] o:list){
				System.out.println(o[0]+","+o[1]);
			}
			t.commit();
		}catch (Exception e) {
			e.printStackTrace();
			t.rollback();
		}finally{
			HibernateUtils.closeSession();
		}
	}
	
	/**
	 *使用左外连接,按客户分组,查询每个客户的姓名和订单数
	 */
	@Test
	public void test02(){
		Session session=HibernateUtils.getSession();
		Transaction t=session.getTransaction();
		try{
			t.begin();
			String hql="select c.name,count(o.orderNo) from Customer c left join c.orderSet o group by c.name";
			Query query=session.createQuery(hql);
			List<Object[]> list= (List<Object[]>) query.list();
			//对象导航查询
			for(Object[] o:list){
				System.out.println(o[0]+","+o[1]);
			}
			t.commit();
		}catch (Exception e) {
			e.printStackTrace();
			t.rollback();
		}finally{
			HibernateUtils.closeSession();
		}
	}
	/**
	 *使用自连接,求出xx的老板是yy
	 */
	@Test
	public void test03(){
		Session session=HibernateUtils.getSession();
		Transaction t=session.getTransaction();
		try{
			t.begin();
			String hql="select e1.name,e2.name from Emp e1,Emp e2 where e1.mgr=e2.id";
			Query query=session.createQuery(hql);
			List<Object[]> list= (List<Object[]>) query.list();
			//对象导航查询
			for(Object[] o:list){
				System.out.println(o[0]+","+o[1]);
			}
			t.commit();
		}catch (Exception e) {
			e.printStackTrace();
			t.rollback();
		}finally{
			HibernateUtils.closeSession();
		}
	}
}

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/114712.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 软件开发模型总结归纳(瀑布模型、螺旋模型、迭代模型、增量模型、敏捷模型)

    软件开发模型总结归纳(瀑布模型、螺旋模型、迭代模型、增量模型、敏捷模型)文章目录 0 软件的生命周期 1 瀑布模型 2 螺旋模型 3 迭代模型 4 增量模型 5 敏捷模型 0 软件的生命周期 软件的生命周期是指从软件产品的设想开始到软件不在使用而结束的时间 软件的生命周期分为 6 个阶段 即需求分析 计划 设计 编码 测试 运行维护 1 瀑布模型 瀑布模型是最早出现的软件开发模型 是所有其他软件开发模型的基础框架 与软件的生命周期不同的是 它缺少了软

    2025年9月1日
    3
  • pycharm 编码怎么设置_pycharm编码格式

    pycharm 编码怎么设置_pycharm编码格式Python中默认的编码格式是ASCII格式,在没修改编码格式时无法正确打印汉字,所以在读取中文时会报错。有两种解决方法。一种是在python的编程工具Pycharm中设置默认编码pycharm下载地址:http://www.jetbrains.com/pycharm/选择社区版即可,免费。设置方法如下:入口A:工具栏-File-DefaultSettings-Editor-File…

    2022年8月27日
    6
  • imfinfo怎么用_matlablimit函数

    imfinfo怎么用_matlablimit函数imfinifo获取图像的相关信息!例如:>>Ipath=’F:\actionrecognition\code2\DB\JPEGImages\action0002.jpg’;>>info=imfinfo(Ipath)info=       Filename:’F:\actionrecognition\code2\DB\JPEGImages\action

    2022年10月5日
    3
  • Python进制转换和补零「建议收藏」

    Python进制转换和补零「建议收藏」1、补零1)字符串python中有一个zfill方法用来给字符串前面补0,非常有用n=”123″s=n.zfill(5)asserts==”00123″zfill()也可以给负数补0n=”-123″s=n.zfill(5)asserts==”-0123″2)数字补零通过格式化的方式来补0n=123s=”%05d”…

    2022年5月12日
    317
  • 扒站工具Teleport Pro教程

    扒站工具Teleport Pro教程1.下载软件http://www.jb51.net/softs/44134.html2.安装3.界面先点开帮助点注册(类似于激活成功教程要不全站扒不全)下面请看ppt,http://www.docin.com/p-633879246.html阿西吧,麻麻再也不用担心我的网站了转载于:https://www.cnblogs.com/MagicZhao123/p/64…

    2025年10月8日
    4
  • 安装vagrant&virtualBox

    安装vagrant&virtualBox

    2021年10月28日
    40

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号