星空网 > 软件开发 > Java

Hibernate(三)__核心接口和类

Hibernate(三)__核心接口和类

 

该图显示了核心接口类以及配置文件的关系层次,越往下越偏向底层数据库

1. hibernate.cfg.

①该文件主要用于指定各个参数,是hibernate核心文件

②默认放在src目录下,也可以放在别的目录下。

③指定连接数据库的驱动、用户名、密码、url、连接池..

④指定对象关系映射文件的位置.

⑤也可使用hibernate.properties文件来替代该文件.(推荐使用 hibernate.cfg.

 

2.对象关系映射文件(*.hbm.

①该文件主要作用是建立表和类的映射关系,是不可或缺的重要文件.

②一般放在其映射的类同一个目录下,但不是必须的。

③命名方式一般是 类名.hbm.

 

3.Configuration 类

它的用处是:

①.读取hibernate.cfg.

Configuration()默认读取的是src 下的hibernate.cfg.

可指定配置文件Configuration(filename),更改放置的文件夹Configuration(com/xidian/config/hibernate.cfg.

②.管理对象关系映射文件 <mapping resource=””>

③.加载hibernate 的驱动,url ,用户..

④.管理hibernate配置信息

 

4.SessionFactory接口(会话工厂)

1.可以缓存sql语句和数据(称为session一级缓存)!!

2.是一个重量级的类,因此我们需要保证一个数据库,有一个SessionFactroy

Hibernate(三)__核心接口和类

3.这里我们讨论一下通过SessionFactory 获取 Session的两个方法 openSession() 一个 getCurrentSession();

①openSession() 是获取一个新的session

②getCurrentSession () 获取和当前线程绑定的session,换言之,在同一个线程中,我们获取的session是同一session,这样可以利于事务控制

虽然session不是线程安全的,但是通过这样的方式,每一个session都处于单线程中,避免session线程安全问题。

如果希望使用 getCurrentSession 需要配置 hibernate.cfg.中配置.

如果是本地事务(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在进行查询的时候是不需要事务提交的。

 Hibernate(三)__核心接口和类

深入探讨:

在 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(三)__核心接口和类

对于需要频繁查找的数据会放在以及缓存中,不是很频繁就会在二级缓存中。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。

Hibernate(三)__核心接口和类

线程局部模式中的变量、对象的生命周期是线程域的,直到线程结束才销毁。

—如何确定你的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();      }          }




原标题:Hibernate(三)__核心接口和类

关键词:Hibernate

*特别声明:以上内容来自于网络收集,著作权属原作者所有,如有侵权,请联系我们: admin#shaoqun.com (#换成@)。
相关文章
我的浏览记录
海外公司注册 | 跨境电商服务平台 | 深圳旅行社 | 东南亚物流