在上一篇,我们讨论了如何配置Tomcat 5.5的Context.xml文件,它能够建立Tomcat 5.5中的只读JNDI。通过这个JNDI,可以为你的Tomcat servlet提供数据源—无论在你的servlet中使用的是Hibernate还是原始JDBC。而且在上一篇中,相应的示例servlet使用了原始JDBC存取我们的MySQL数据库。
在本篇中,相应的示例servlet则使用Hibernate(不再是原始JDBC)存取MySQL数据库,但是将继续使用相同的Context.xml文件(没有作任何改变)。因此,在本文中,Hibernate仍将利用与原始JDBC(已在上篇中使用)相同的JNDI数据源。
广州网站建设,网站建设,广州网页设计,广州网站设计
一、Web.xml
让我们先从分析web.xml入手(我们假定,你已经理解web.xml的作用—用于标识和配置任何Java web应用程序组件,并且你已经理解了这个文件的格式):
|
|
上面描述了web.xml的所有内容(适用于本系列中的三篇)。它包含了三个servlet:
◆Retrieval—在第一篇中利用原始JDBC的servlet。 ◆RetrieveViaHibernate—本篇中的servlet,它查询与第一篇中相同的MySQL数据库,但现在使用的是Hibernate。 ◆InsertViaHibernate—第三篇中的servlet,它在一个事务(从Hibernate中获取)边界上执行数据库插入操作。
另外,在靠近web.xml的顶部是一个xml listener元素—在本例中,它用于定义一个ServletContextListener。该监听器是一个类,当web应用程序已经准备好进行请求处理(例如在Tomcat服务器启动时)时,这个类被实例化。具体地说,在处理任何请求之前,应用程序将调用这个类的contextInitialized()方法来进行必要的初始化。例如,在第一次调用这个web.xml中的任何一个servlet之前,在我们的应用程序中会首先激活contextInitialized()方法。此后,这个方法将不被再次激活,除非应用程序重新运行或Tomcat重新启动。
二、ServletContextListener
在我们的例子中,我们想让ServletContextListener加载我们的HibernateUtil类,以便再由该类负责实例化一个静态Hibernate会话工厂。结果是,在我们的任何一个servlet收到Http请求之前,该Hibernate会话工厂已经准备好了。下面,让我们来分析一下我们的监听器(tomcatJndi.HibernateAppListener)的相应代码:
广州网站建设,网站建设,广州网页设计,广州网站设计
|
package tomcatJndi; import javax.servlet.ServletContextListener; import javax.servlet.ServletContextEvent; import org.apache.log4j.Logger; import org.hibernate.Session; public class HibernateAppListener implements ServletContextListener { /*应用程序启动事件*/ public void contextInitialized(ServletContextEvent ce) { Logger loggerConnect = Logger.getLogger("Connect"); try { loggerConnect.debug("In HibernateAppListener.contextInitialized"); Class.forName("tomcatJndi.HibernateUtil").newInstance(); loggerConnect.debug("In HibernateAppListener, Class.forName for tomcatJndi.HibernateUtil successful"); } catch (Exception e) { loggerConnect.debug("In HibernateAppListener, Class.forName for tomcatJndi.HibernateUtil throws Exception"); } } /*应用程序退出事件*/ public void contextDestroyed(ServletContextEvent ce) {} } |
注意上面调用的Class.forName(...).newInstance()方法,它试图确保指定的类(tomcatJndi.HibernateUtil)被加载,这样才能调用HibernateUtil中的静态初始化器。接下来,让我们分析一下HibernateUtil。
三、HibernateUtil
|
package tomcatJndi; import org.hibernate.*; import org.hibernate.cfg.*; import org.apache.log4j.Logger; public class HibernateUtil { private static final SessionFactory sessionFactory; static { try { Logger loggerRoot = Logger.getRootLogger(); Logger loggerConnect = Logger.getLogger("Connect"); loggerRoot.debug("In HibernateUtil try-clause"); loggerRoot.error("In HibernateUtil try-clause"); loggerConnect.debug("In HibernateUtil try-clause via loggerConnect DEBUG*****"); //从hibernate.cfg.xml中创建SessionFactory sessionFactory = new Configuration().configure().buildSessionFactory(); } catch (Throwable ex) { System.err.println("Initial SessionFactory creation failed." + ex); throw new ExceptionInInitializerError(ex); } } public static SessionFactory getSessionFactory() { return sessionFactory; } } |
注意,在此,静态SessionFactory(一个singleton)被buildSessionFactory()所实例化。接口org.hibernate.SessionFactory描述了一个Hibernate工厂—我们的代码能够从这个工厂中获得org.hibernate.session对象。在此,这个Hibernate会话(Session)是我们的应用程序中的重点;它负责实现与Hibernate的交互,并且要求它(Hibernate)完成任何功能。



