Hibernate入门4.核心技能 20131128
代码下载 链接: http://pan.baidu.com/s/1Ccuup 密码: vqlv
前言:
前面学习了Hibernate3的基本知识,并且应用在简单的实践中。这些知识Hibernate的基本知识,只能说是会使用简单的hibernate了。这一章学习的是Hibernate的核心技能。主要知识点:
Hibernate中持久化类的关联关系、Hibernate的批量处理技术、Query接口的核心方法和使用、HQL语言的查询、Criteria接口的核心方法以及查询技巧、Restrictions的使用方法、DetachedCriteria离线查询的技巧、Hibernate事务处理
1.Hibernate中的关联关系
关联关系指的是类之间的引用关系,是实体对象之间普遍存在的一种关系。使用Hibernate框架可以完整的表述这种关联关系,如果映射的十分恰当,Hibernate的关联映射将在很大程度上简化对于持久层数据的访问。
1.1一对多的关联关系
单项的N-1的关系,只能够从N访问到1的一端,在类与类的各种关联关系中,单项N-1关联和关系数据库中的外检参照关系最为相似。比如Customer和Order,使用的是N-1的单项关联,即从Order到Customer的关联关系。(其实这里已经买下一个地雷了,我们使用Order表示订单,在数据库中,对应的表的名字可以是order,当然是不可以的,因为order是关键字。在数据库的创建表的过程中,我习惯使用小写字母,这里就会出现非常隐蔽的错误,我们不断的调试程序,却始终找不到错误。)
Customer.java
package com.yang.hib.model;
public class Customer {
private int id;
private String userName;
private String password;
private String realName;
private String address;
private String mobile;
private String email;
public Customer(){}
public Customer(String userName, String password, String realName,
String address, String mobile, String email) {}
//getter & setter
}
Order.java:
package com.yang.hib.model;
import java.util.Date;
public class Order {
private int id;
private String orderNo;
private Date date;
private double total;
private Customer customer;
public Order(){}
public Order(String orderNo, Date date, double total, Customer customer) {}
//getter & setter
public void setCustomer(Customer customer){this.customer = customer;}
public Customer getCustomer(){return this.customer;}
}
Customer.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="com.yang.hib.model">
<class name="Customer" table="customer">
<!—参考之前的代码-->
</class>
</hibernate-mapping>
Order.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="com.yang.hib.model">
<class name="Order" table="myorder">
<!—注意这里的数据表不可以是order,关键字一般情况下不能够用作表名-->
<id name="id" column="id">
<generator class="native"/>
</id>
<property name="orderNo" column="orderno" type="string"/>
<property name="date" column="orderdate" type="timestamp"/>
<property name="total" column="total" type="double"/>
<many-to-one name="customer" column="customer_id" class="Customer" cascade="save-update"/>
<!—这一句是最为核心的配置信息,实现多对一的级联关系-->
</class>
</hibernate-mapping>
HibernateUtil.java
package com.yang.hib.util;
public class HibernateUtil {
private static String HIBERNATE_CONFIG_FILE = "hibernate.cfg.xml";
private static final ThreadLocal<Session> threadLocal = new ThreadLocal<Session>();
private static Configuration configuration = new Configuration();
private static SessionFactory sessionFactory ;
/*静态代码段,在该类加载的时候,执行这一段代码,这里使用的是单例模式,同时为了在多线程中使用session,使用了线程的局部变量,之前在Java进阶中的关于线程同步的知识中,最后一种方式便是使用的这一种线程局部变量将他保护起来的方式*/
static{
try{
configuration.configure(ClassLoader.getSystemResource(HIBERNATE_CONFIG_FILE));
sessionFactory = configuration.buildSessionFactory();
}catch(Exception e){
System.err.println("%%%%%Error Creating session factory %%%%%%%%");
e.printStackTrace();
}
}
//禁止外部创建该类型的对象
private HibernateUtil(){}
public static Session getSession() throws HibernateException{
Session session = threadLocal.get();
if(session == null || !session.isOpen()){
if(sessionFactory == null){
sessionFactory = configuration.buildSessionFactory();
}
session = (sessionFactory != null)?sessionFactory.openSession():null;
threadLocal.set(session);
}
return session;
}
public static void reBuildSessionFactory(){
try{
configuration.configure(HIBERNATE_CONFIG_FILE);
sessionFactory = configuration.buildSessionFactory();
}catch(Exception e){
System.err.println("%%%%Error Creating SessionFactory %%%%%");
e.printStackTrace();
}
}
public static void closeSession() throws HibernateException{
Session session = threadLocal.get();
threadLocal.set(null);
if(session != null){
session.close();
}
}
public static SessionFactory getSessionFactory(){
return sessionFactory;
}
public static void setHibernateConfigFile(String hibernateConfigFile){
HIBERNATE_CONFIG_FILE = hibernateConfigFile;
}
public static Configuration getConfiguration(){
return configuration;
}
}
TeatMain.java
package com.yang.main;
public class TestMain {
public static void addCustomerAndOrder(){
Customer customer = new Customer("yang","12345","杨腾飞","广州","15800027127","1076906529@qq.com");
System.out.println("add customer to database");
addCustomer(customer);
Order order1 = new Order("A1",new Date(),42.8,customer);
System.out.println("add order to database");
addOrder(order1);
order1 = new Order("A2",new Date(),35.9,customer);
System.out.println("add antother order to database");
addOrder(order1);
}
public static void addCustomer(Customer customer){
Session session = HibernateUtil.getSession();
Transaction trans = session.beginTransaction();
session.save(customer);
trans.commit();
HibernateUtil.closeSession();
}
public static void addOrder(Order order){
Session session = HibernateUtil.getSession();
Transaction trans = session.beginTransaction();
session.save(order);
trans.commit();
HibernateUtil.closeSession();
}
public static void main(String[] args) {
// TODO Auto-generated method stub
addCustomerAndOrder();
}
}
我们可以根据订单查询出数据库中的对应的顾客的信息,修改TestMain.java函数:
public static void findOrder(int id){
Session session = HibernateUtil.getSession();
Order order = (Order) session.get(Order.class, id);
Customer customer = order.getCustomer();
System.out.println( "Id:"+order.getId()+
"\ntotal:" + order.getTotal()+
"\norderdate:" + order.getDate().toLocaleString() +
"\nrealName:"+customer.getRealName()+
"\nmobile:" + customer.getMobile());
HibernateUtil.closeSession();
}
public static void main(String[] args) {
// TODO Auto-generated method stub
//addCustomerAndOrder();
findOrder(1);
}
这个时候,我们发现在控制台中是执行了两个SQL语句,对于两个表的查询,我们可以使用连接的形式使用一条SQL语句便可以完成这些查询:
在hibernate-mapper配置信息中,对于<many-to-one>元素中可以使用指定的属性来让hibernate使用连接的方式:
<many-to-one name=”customer” fetch=”join” column=”customer_id” class=”Customer”/>实现。对于many-to-one元素的fetch属性也可以使用outer-join属性替代,默认情况之下使用的是fetch=”select”。
1.2单项的1-N的关系
Customer与Order就是一对多的关系,我们可以在Order中添加orders集合属性
Customer.java
private Set<Order> orders=new HashSet<Order>();
public void setOrders(Set<Order> orders){ this.orders = orders;}
public Set<Order> getOrders(){return this.orders;}
配置文件中添加如下配置信息:
Customer.hbm.xml
<set name="orders">
<key column="customer_id"/><!—这里是对应的另一个表中的字段-->
<one-to-many class="Order"/>
</set>
在Hibernate中通过比较两个持久化对象的标识符属性ID来判断两者是否是相等的,这需要在实体类中对equals()和hashCode()方法重写。
重写hashCode()
public int hashCode(){
final int prime = 31;
int result = 1;
result = result*prime + ((id==null)?0:id.hashCode());
return result;
}
也就是将id+prime作为其对象的hashCode
同时对已比较两个对象是否相等,如果两者的引用是同一个对象的话,那么无论怎样都是相等的,否则如果对象是null的话,那么两者肯定不相等,因为如果两个都是null的话,那么在前面的判断中两者引用都是null则相等。继而继续判断:将obj转换成为目标类型的对象,这个时候原始的对象的id如果是null,也就是数据库中没有该数据,另一个对象也是没有持久化的,内容虽然不相同,但是两者的都没有id,则判断两者是相等的。
参考代码:
Customer c1 = new Customer("yang","12345","杨腾飞","广州","15800027127","1076906529@qq.com");
Customer c2 = new Customer("yang","134545","杨腾飞","广州","15800027127","1076906529@qq.com");
System.out.println(c1.equals(c2));
正常情况之下,两者是不想等的,但是在Hibernate重写equals函数之后,两者就是相等的。很诡异的吧,其实我现在也不是很理解。但是这是在Hibernate中比较数据库中的两个对象,就只会根据id来判断。
public boolean equals(Object obj){
if(this == obj){ return true; }
if(obj == null){ return false;}
if(getClass() != obj.getClass()){ return false;}
Customer other = (Customer) obj;
if(id == null){
if(other.id != null){return false;}
}else if( ! id.equals(other.id)){
return false;
}
return true;
}
下面查询执行用户的订单:
public static void findOrderByCustomer(Integer id){
Session session = HibernateUtil.getSession();
Customer customer = (Customer) session.get(Customer.class, id);
Set<Order> orders = customer.getOrders();
for(Order order : orders){
System.out.println( "id:"+ order.getId()+
",\torderNo:"+order.getOrderNo()+
",\ttotal:" + order.getTotal()+
",\tdate:"+order.getDate().toLocaleString());
}
}
一般使用的技术是双向的1-N级联,也就是在两个里面都配置级联关系,形成双向的级联。
YangTengfei
2013.11.28