hibernate session 过滤器控制

来源:互联网 发布:linux的ps命令 编辑:程序博客网 时间:2024/04/30 01:06
 

hibernate session 过滤器控制

最近做一个公司项目的Demo,使用技术为jsp+dwr+hibernate,因为该Demo中所用的表多,有一对一,一对多和多对一的关系,所以利用hibernate的延迟加载(lazy="true")对性能的提高相当重要,但随之带来的session管理更为重要,弄不好常出现延迟加载异常,在这里我利用了Filter 来处理session的close的,相关代码如下:

 

1.HibernateSessionFactory.java(Eclipse自动生成,通过ThreadLocal 将session的非线程安全变成安全)

Java代码 复制代码 收藏代码
  1. import org.apache.log4j.Logger;   
  2. import org.hibernate.HibernateException;   
  3. import org.hibernate.Query;   
  4. import org.hibernate.Session;   
  5. import org.hibernate.cfg.Configuration;   
  6.   
  7.   
  8. public class HibernateSessionFactory {   
  9.   
  10.     private static final Logger log = Logger.getLogger(SessionFactory.class) ;   
  11.     private static final ThreadLocal<Session> threadLocal = new ThreadLocal<Session>();   
  12.     private static String CONFIG_FILE_LOCATION = "/hibernate.cfg.xml";   
  13.     private static Configuration configuration = new Configuration();   
  14.     private static org.hibernate.SessionFactory sessionFactory;   
  15.     private static String configFile = CONFIG_FILE_LOCATION;   
  16.   
  17.     static {   
  18.         try {   
  19.             configuration.configure(configFile);   
  20.             sessionFactory = configuration.buildSessionFactory();   
  21.         } catch (Exception e) {   
  22.                
  23.             log.error(" error creating sessionFactory") ;   
  24.             e.printStackTrace();   
  25.         }   
  26.     }   
  27.     public HibernateSessionFactory() {   
  28.     }   
  29.        
  30.   
  31.     public static Session getSession() throws HibernateException {   
  32.         log.info("getSession is run") ;   
  33.            
  34.         Session session = (Session) threadLocal.get();   
  35.         if (session == null || !session.isOpen()) {   
  36.             log.info("SessionFactory.getSession() session is null or close") ;   
  37.             if (sessionFactory == null) {   
  38.                 log.info("SessionFactory.getSession() rebuildSessionFactory() is running") ;   
  39.                 rebuildSessionFactory();   
  40.             }    
  41.             session = (sessionFactory != null) ? sessionFactory.openSession(): null;   
  42.             threadLocal.set(session);   
  43.         }   
  44.   
  45.         return session;   
  46.     }   
  47.   
  48.   
  49.     public static void rebuildSessionFactory() {   
  50.         log.info("rebuildSessionFactory is run") ;   
  51.         try {   
  52.             configuration.configure(configFile);   
  53.             sessionFactory = configuration.buildSessionFactory();   
  54.         } catch (Exception e) {   
  55.             log.error("erro creating sessionFactory") ;   
  56.             e.printStackTrace();   
  57.         }   
  58.     }   
  59.   
  60.   
  61.     public static void closeSession() throws HibernateException {   
  62.         log.info("closeSession is run") ;   
  63.            
  64.         Session session = (Session) threadLocal.get();   
  65.         threadLocal.set(null);   
  66.         if (session != null) {   
  67.             log.info("SessionFactory.closeSession() session is not null") ;   
  68.             if(!session.isOpen()){   
  69.                 System.out.println("该 session 已经关闭") ;   
  70.             }   
  71.             session.close();   
  72.         }   
  73.     }   
  74.   
  75.   
  76.     public static org.hibernate.SessionFactory getSessionFactory() {   
  77.         return sessionFactory;   
  78.     }   
  79.   
  80.   
  81.     public static void setConfigFile(String configFile) {   
  82.         SessionFactory.configFile = configFile;   
  83.         sessionFactory = null;   
  84.     }   
  85.   
  86.   
  87.     public static Configuration getConfiguration() {   
  88.         return configuration;   
  89.     }   
  90.        
  91.   
  92.   
  93. }  

 HibernateSessioniFactory主要负责session的开和关。

 

2.HibernateSessionFilter(要编写的过滤器,主要处理session的关闭)

Java代码 复制代码 收藏代码
  1. import java.io.IOException;   
  2.   
  3.   
  4. import javax.servlet.Filter;   
  5. import javax.servlet.FilterChain;   
  6. import javax.servlet.FilterConfig;   
  7. import javax.servlet.ServletException;   
  8. import javax.servlet.ServletRequest;   
  9. import javax.servlet.ServletResponse;   
  10. import org.apache.commons.logging.Log;   
  11. import org.apache.commons.logging.LogFactory;   
  12. import com.ad.session.HibernateSessionFactory;   
  13.   
  14. public class HibernateSessionFilter implements Filter {     
  15.     private static final Log log = LogFactory.getLog(HibernateSessionFilter.class);     
  16.        
  17.       
  18.       @Override  
  19.     public void init(FilterConfig filterConfig) throws ServletException {   
  20.         System.out.println("HibernateSessionFilter is end") ;   
  21.     }    
  22.      
  23.    @Override  
  24.     public void doFilter(ServletRequest arg0, ServletResponse arg1,     
  25.             FilterChain chain) throws IOException, ServletException {     
  26.         log.debug("HibernateSessionFilter start");     
  27.            
  28.         try{     
  29.             //request 之前要处理的代码   
  30.             chain.doFilter(arg0, arg1);        
  31.            //response之后要要处理的代码       
  32.               
  33.         }catch (Exception e) {     
  34.             e.printStackTrace();                
  35.         }   finally{             
  36.             HibernateSessionFactory.closeSession();               
  37.         }     
  38.      
  39.     }   
  40.      
  41.         
  42.    @Override  
  43.      public void destroy() {        
  44.        System.out.println("HibernateSessionFilter is start") ;   
  45.      }     
  46.      
  47. }  

  该过滤器只处理session的关闭,似乎有点浪费了,在这也可以加上字符集的处理。

  Filter运行原理(个人认为,有误请多指教):首先建立一个过滤类,需继承Filter接口,必须实现三个方法:

  init:服务器启动时运行该方法,运行期间不执行;

 doFilter:运行期间执行,其作用是request之前要做那些工作,response之后(即服务器将内容全部发回View层后)

               要做那些工作。每次访问服务器都会执行doFilter方法

 destroy:服务器终止时运行。

 

3. 在web.xml中的配置如下:

Xml代码 复制代码 收藏代码
  1.        <filter>  
  2.     <filter-name>hibernateSession</filter-name>  
  3.     <filter-class>  
  4.         com.c35.ad.filter.HibernateSessionFilter   
  5.     </filter-class>  
  6. </filter>  
  7. <filter-mapping>  
  8.     <filter-name>hibernateSession</filter-name>  
  9.     <url-pattern>/*</url-pattern><!-- 过滤所有发回服务器的请求-->  
  10. </filter-mapping>  

 

  4.dao(demo中的UserDao)

 

Java代码 复制代码 收藏代码
  1. public class UserDao  implements BaseDao {   
  2.   
  3.     private static final Logger log = Logger.getLogger(UserDao.class);   
  4.        
  5.     private  Session session  ;   
  6.       
  7.     public UserDao(){   
  8.        log.info("UserDao construct is running") ;       
  9.            session = SessionFactory.getSession() ;   
  10.     }   
  11.        
  12.   
  13.     @Override  
  14.     public boolean save(Object o) {   
  15.         log.info("UserDao save is running ");   
  16.         boolean flag = false ;     
  17.         Transaction tran = null;       
  18.         try {          
  19.             tran = session.beginTransaction();   
  20.             session.save(o);   
  21.             tran.commit();   
  22.             flag = true ;   
  23.         } catch (Exception e) {   
  24.             log.error("error save user");   
  25.             flag = false ;   
  26.             if (tran != null) {   
  27.                 try {   
  28.                     tran.rollback();   
  29.                 } catch (HibernateException e1) {   
  30.                     log.error("save Transaction rollback is error "+e1.getMessage()) ;   
  31.                 }   
  32.             }   
  33.             e.printStackTrace();   
  34.         }finally{   
  35.             //session.close() ;//这里就不需要在关闭session直接交给filter来处理  
  36.         }   
  37.         return flag ;   
  38.     }   
  39. }  

    以上是本人的学习总结,有误之处请多多指教,希望对刚学hibernate的同志们有所帮助。

 

原创粉丝点击