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  字段

(3)HQL实战

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

             group by c.name 

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

        (C)使用自连接,求出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();
		}
	}
}



        
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Hibernate 中,可以使用 HQL 或者 Criteria API 来进行内外连接查询。 1. 内连接查询 内连接查询可以使用 HQL 或者 Criteria API 来实现。以下是 HQL 的示例: ```java String hql = "select c.name, o.orderNumber from Customer c inner join c.orders o where c.name like '%John%'"; Query query = session.createQuery(hql); List<Object[]> results = query.list(); for(Object[] row : results) { String customerName = (String) row[0]; String orderNumber = (String) row[1]; // do something with the result } ``` 以上 HQL 查询使用了 `inner join` 关键字。在这里,我们查询了所有名字包含 "John" 的客户及其所有订单号。 以下是 Criteria API 的示例: ```java CriteriaBuilder builder = session.getCriteriaBuilder(); CriteriaQuery<Object[]> query = builder.createQuery(Object[].class); Root<Customer> customer = query.from(Customer.class); Join<Customer, Order> order = customer.join("orders", JoinType.INNER); query.multiselect(customer.get("name"), order.get("orderNumber")); query.where(builder.like(customer.get("name"), "%John%")); List<Object[]> results = session.createQuery(query).getResultList(); for(Object[] row : results) { String customerName = (String) row[0]; String orderNumber = (String) row[1]; // do something with the result } ``` 以上 Criteria API 查询使用了 `join` 方法,并指定了 `JoinType.INNER` 来进行内连接查询。在这里,我们查询了所有名字包含 "John" 的客户及其所有订单号。 2. 外连接查询 外连接查询同样可以使用 HQL 或者 Criteria API 来实现。以下是 HQL 的示例: ```java String hql = "select c.name, o.orderNumber from Customer c left join c.orders o"; Query query = session.createQuery(hql); List<Object[]> results = query.list(); for(Object[] row : results) { String customerName = (String) row[0]; String orderNumber = (String) row[1]; // do something with the result } ``` 以上 HQL 查询使用了 `left join` 关键字。在这里,我们查询了所有客户及其所有订单号,如果客户没有订单则订单号为 null。 以下是 Criteria API 的示例: ```java CriteriaBuilder builder = session.getCriteriaBuilder(); CriteriaQuery<Object[]> query = builder.createQuery(Object[].class); Root<Customer> customer = query.from(Customer.class); Join<Customer, Order> order = customer.join("orders", JoinType.LEFT); query.multiselect(customer.get("name"), order.get("orderNumber")); List<Object[]> results = session.createQuery(query).getResultList(); for(Object[] row : results) { String customerName = (String) row[0]; String orderNumber = (String) row[1]; // do something with the result } ``` 以上 Criteria API 查询同样使用了 `join` 方法,并指定了 `JoinType.LEFT` 来进行左外连接查询。在这里,我们查询了所有客户及其所有订单号,如果客户没有订单则订单号为 null。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值