zoukankan      html  css  js  c++  java
  • Hibernate(三)__核心接口和类

     

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

    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();
                }
                
            }
  • 相关阅读:
    httpModules与httpHandlers之httpModules(转载)
    一个人的命运决定于晚上8点到10点之间【认真看完本篇文章,你的人生将会有所改变】
    关于错误Access Violation和too many consecutive exceptions 解决方法
    Delphi PChar与String互转
    Delphi+MySQL:TADOQuery使用插入中文乱码解决方法
    Delphi中的操作二进制文件的两个重要函数
    JavaWeb错误处理集锦
    Codeforces Round #286 (Div. 1) B. Mr. Kitayuta&#39;s Technology (强连通分量)
    从头认识java-16.4 nio的读与写(ByteBuffer的使用)
    php在数字前面补0得到固定长度数字的两种方法
  • 原文地址:https://www.cnblogs.com/xiangkejin/p/5993884.html
Copyright © 2011-2022 走看看