该图显示了核心接口类以及配置文件的关系层次,越往下越偏向底层数据库。
1. hibernate.cfg.xml文件
①该文件主要用于指定各个参数,是hibernate核心文件
②默认放在src目录下,也可以放在别的目录下。
③指定连接数据库的驱动、用户名、密码、url、连接池..
④指定对象关系映射文件的位置.
⑤也可使用hibernate.properties文件来替代该文件.(推荐使用 hibernate.cfg.xml)。
2.对象关系映射文件(*.hbm.xml)
①该文件主要作用是建立表和类的映射关系,是不可或缺的重要文件.
②一般放在其映射的类同一个目录下,但不是必须的。
③命名方式一般是 类名.hbm.xml,但不是必须的。
3.Configuration 类
它的用处是:
①.读取hibernate.cfg.xml
Configuration()默认读取的是src 下的hibernate.cfg.xml
可指定配置文件Configuration(filename),更改放置的文件夹Configuration(com/xidian/config/hibernate.cfg.xml)
②.管理对象关系映射文件 <mapping resource=””>
③.加载hibernate 的驱动,url ,用户..
④.管理hibernate配置信息
4.SessionFactory接口(会话工厂)
1.可以缓存sql语句和数据(称为session一级缓存)!!
2.是一个重量级的类,因此我们需要保证一个数据库,有一个SessionFactroy
3.这里我们讨论一下通过SessionFactory 获取 Session的两个方法 openSession() 一个 getCurrentSession();
①openSession() 是获取一个新的session
②getCurrentSession () 获取和当前线程绑定的session,换言之,在同一个线程中,我们获取的session是同一session,这样可以利于事务控制
虽然session不是线程安全的,但是通过这样的方式,每一个session都处于单线程中,避免session线程安全问题。
如果希望使用 getCurrentSession 需要配置 hibernate.cfg.xml中配置.
如果是本地事务(jdbc事务) 只操作一个数据库
<property name="current_session_context_class">thread</property>
如果使用的是全局事务(jta事务) 跨数据库
<property name="current_session_context_class">jta</property>
③如何选择
原则:
(1)如果需要在同一线程中,保证使用同一个Session则,使用getCurrentSession()
(2)如果在一个线程中,需要使用不同的Session,则使用opentSession()
④.通过 getCurrentSession() 获取的session在事务提交或者rollback后,会自动关闭,通过openSession()获取的session则必须手动关闭。
⑤.如果是通过getCurrentSession() 获取 sesssion ,所有操作都必须使用事务提交,包括进行查询.而getOpenSession在进行查询的时候是不需要事务提交的。
深入探讨:
在 SessionFactory启动的时候,Hibernate 会根据配置创建相应的 CurrentSessionContext,在getCurrentSession()被调用的时候,实际被执行的方法是
CurrentSessionContext.currentSession()。在 currentSession()执行时,如果当前Session为空,currentSession会调用SessionFactory的openSession。
问题:
如果在前一个service中开启事务但不提交,在下一个service中开启事务并提交事务,(都必须开启事务)那么下一个事务提交的时候会将
前一个事务一并提交。如果在后面一个事务中有了事务回滚,则一并回滚。换句话说者两个service中的事务可看成一个事务。
问题是我用getCurrentSession和getOpenSession获取session在这种情况下产生的效果是一样的。
用getOpenSession在不同的service中获取到的session应该是不一样的,那么事务也应该是不一致的,
那么我后面service提交的应该是自己的事务,而不影响前一个的事务,可是还是一并提交了。
如何理解?希望有大神指点填坑。
5.sesison接口
它的主要功能和作用是:
①Session一个实例代表与数据库的一次操作(当然一次操作可以是crud组合)
②Session实例通过SessionFactory获取,用完需要关闭。
③Session是线程不同步的(不安全),因此要保证在同一线程中使用,可以用getCurrentSession()。
④Session可以看做是持久化管理器,它是与持久化操作相关的接口
session中有几个重要的方法:save、delete、update、get/load
get()和load()区别
①如果查询不到数据,get 会返回 null,但是不会报错, load 如果查询不到数据,则报错ObjectNotFoundException。
②使用get 去查询数据,(先到一级/二级)会立即向db发出查询请求(select ...), 如果你使用的是 load查询数据,(先到一级、二级))即使查询到对象,
返回的是一个代理对象,如果后面没有使用查询结果,它不会真的向数据库发select ,当程序员使用查询结果的时候才真的发出select ,
这个现象我们称为懒加载(lazy)。
③通过修改配置文件,我们可以取消懒加载
<class name="Employee" lazy="false" table="employee">
如何选择使用哪个: 如果你确定DB中有这个对象就用load(),不确定就用get()(这样效率高)
对于需要频繁查找的数据会放在以及缓存中,不是很频繁就会在二级缓存中。hibernate的二级缓存是需要配置的,以及缓存是必须使用的。
ThreadLocal模式 (线程局部变量模式) 管理Session
我们对获取session的工具类(原来的代码在hibernate(一)中)升级,让它可以直接返回 全新的session和线程相关的session
public class HibernateUtil {
//使用线程局部模式
private static final ThreadLocal <Session> threadLocal =new ThreadLocal<Session>();
private static final SessionFactory sessionFactory;
private HibernateUtil(){};
static {
sessionFactory = new Configuration().configure().buildSessionFactory();
}
//获取全新的sesession
public static Session getOpenSession(){
return sessionFactory.openSession();
}
//获取和线程关联的session
public static Session getCurrentSession() throws HibernateException {
Session s = threadLocal.get();
//判断是否得到
if(s == null) {
s = sessionFactory.openSession();
//把session对象放置到threadLocal中,这样就进行了绑定。
threadLocal.set(s);
}
return s;
}
public static void closeSession() throws HibernateException {
Session s = (Session) threadLocal.get();
if(s != null)
{ s.close();}
threadLocal.set(null);
}
}
升级获取session的方式后,不需要再配置<property name="current_session_context_class">thread</property>而可以直接使用currentSession。
线程局部模式中的变量、对象的生命周期是线程域的,直到线程结束才销毁。
—如何确定你的session有没有及时关闭?
netstat –an [oracle 1521 mysql 3306 sql server 1433] 查看端口是否被监听
6.Transaction事务
数据库事务是指由一个或多个SQL语句组成的工作单元,这个工作单元中的SQL语句相互依赖,如果有一个SQL语句执行失败,就必须撤销整个工作单元。
简单说要么全部成功,要么全部失败。
在并发环境中,多个事务同时访问相同的数据资源时,可能会造成各种并发问题,可通过设定数据库的事务隔离级别来避免。
在hibernate中一个session可以开启多个事务,当一个事务结束或者撤销后,session就会关闭。
7.query接口
通过query接口我们可以完成更加复杂的查询任务.
举例: 通过用户来查询数据.
Session session=HibernateUtil.getCurrentSession();
Transaction ts=null;
try {
ts=session.beginTransaction();
//获取query引用[注意:这里 Employee不是表.而是domain类名]
//[where 后面的条件可以是类的属性名,也可以是表的字段,按照hibernate规定我们还是应该使用类的属性名.]
Query query=session.createQuery("from Employee where namehsp='shunping'");
//通过list方法获取结果,这个list会自动的将封装成对应的domain对象
//所以我们jdbc进行二次封装的工作没有.
List<Employee> list=query.list();
for(Employee e: list){
System.out.println(e.getAaaid()+" "+e.getHiredate());
}
ts.commit();
} catch (Exception e) {
if(ts!=null){
ts.rollback();
}
throw new RuntimeException(e.getMessage());
}finally{
//关闭session
if(session!=null&&session.isOpen()){
session.close();
}
}
8.criteria 接口
criteria接口的简单使用:
Session session=HibernateUtil.getCurrentSession();
Transaction ts=null;
try {
ts=session.beginTransaction();
Criteria cri=session.createCriteria(Employee.class).
setMaxResults(2).addOrder(Order.desc("id") );
List<Employee> list=cri.list();
for(Employee e: list){
System.out.println(e.getAaaid());
}
ts.commit();
} catch (Exception e) {
if(ts!=null){
ts.rollback();
}
throw new RuntimeException(e.getMessage());
}finally{
//关闭session
if(session!=null&&session.isOpen()){
session.close();
}
}