前言
所謂的SSH,就是表現層使用Struts2,業務邏輯層使用Spring,數據訪問層使用Hibernate,好了,廢話不多說,開打開打,哦,不,是開講開講。
前提,需要你熟悉Struts2,Spring,Hibernate的基本使用。
步驟,第一步,添加繁瑣的Jar包們,第二步:Spring整合Hibernate。第三步:Struts2整合Spring。
-------------------------------------------------不知道起啥名分割線--------------------------------------------------------------------------------
我在這四個文件中尋找jar包。
① 導入Struts jar包,struts-2.3.15.3\apps\struts2-blank\WEB-INF\lib下的jar包
上面落了一個Struts整和Spring的關鍵jar包:struts2-spring-plugin-2.3.15.3.jar,上面未列出。(共14個包)
②導入Spring jar包,一個是核心包:spring-framework-3.2.0.RELEASE\libs
另一個是依賴包:
其中,spring-orm.jar就是Spring整合Hibernate的關建包。(共13個包)
注意:若單獨引用Spring,還需引入一個依賴包:com.springsource.org.apache.commons.logging-1.1.1.jar,但是由於前面Struts2引入了commons-logging-1.1.3.jar包,所以不需要再次引用。其實,它們是一樣的代碼,只是Spring將其改了名字,自己可以查看代碼就會發現了。
③導入Hibernate Jar包
ehcache.jar包是Hibernate的二級緩存包,其依賴backport-util-concurrent.jar,和commons-logging.jar包。(共10個包)
Hibernate中也有一個javassist-3.12.0.GA.jar包,Struts2中已經引用了javassist-3.11.0.GA.jar,所有我替換成了12版本。上面未列出。
④當然,最後還有一個jar包,那就是mysql-connector-java-5.0.8-bin.jar。
到此為止,共添加了38個包,包不再改動。
----------------------------------------情景設置分割線--------------------------------------------------------------------------------------------------
業務:用戶添加
實體類:User(兩個字段:name和age)
------------------------------[Spring+Hibernate]分割線-----------------------------------------------------------------------------------------------
Spring開始整合Hibernate,眾所周知,這兩個框架都有自己的配置文件,分別是:Spring的applicationContext.xml(名字任意),Hibernate的配置文件與映射文件hibernate.cfg.xml與【實體類】.hbm.xml文件。整合的結果就是Hibernate的配置文件消失(當然它的配置信息跑到了Spring的配置文件中去了)。
程序代碼:
//DAL層的用戶接口 public interface UserDao { boolean insert(User user); } //DAL層的用戶接口的具體實現 public class UserDaoImpl extends HibernateDaoSupport implements UserDao { @Override public boolean insert(User user) { HibernateTemplate s=this.getHibernateTemplate(); s.save(user); return true; } }
applicationContext.xml文件配置:(User.hbm.xml不多說,沒區別)
<!-- 使用C3P0得到數據源--> <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"> <property name="driverClass" value="com.mysql.jdbc.Driver"></property> <property name="jdbcUrl" value="jdbc:mysql://localhost:3306/test"></property> <property name="user" value="root"></property> <property name="password" value="123456"></property> </bean> <!-- 使用FactoryBean創建Spring得到 SessionFactory --> <bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean"> <!-- 把hibernate.cfg.xml文件中的配置信息注入進來,從而可以刪掉配置文件 --> <property name="dataSource" ref="dataSource"></property> <property name="hibernateProperties"> <props> <prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop> <prop key="javax.persistence.validation.mode">none</prop> <prop key="hibernate.show_sql">true</prop> <prop key="hibernate.format_sql">true</prop> <prop key="hibernate.hbm2ddl.auto">update</prop> <prop key="hibernate.current_session_context_class">thread</prop> </props> </property> <property name="mappingLocations" value="classpath:king/domain/User.hbm.xml"></property> </bean> <!-- 目的:創建DAO --> <bean id="userDao" class="king.dao.impl.UserDaoImpl"> <property name="sessionFactory" ref="sessionFactory"></property> </bean> <!-- Spring的事務管理之Hibernate,需要注入SessionFactory --> <bean id="txManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager"> <property name="sessionFactory" ref="sessionFactory"></property> </bean> <!-- 通知 --> <tx:advice id="adviceId" transaction-manager="txManager"> <tx:attributes> <tx:method name="*" propagation="REQUIRED" isolation="DEFAULT"/> </tx:attributes> </tx:advice> <aop:config> <aop:advisor advice-ref="adviceId" pointcut="execution(* king.dao.impl.*.*(..))"/> </aop:config>
測試代碼:
@Test public void test1(){ ClassPathXmlApplicationContext app=new ClassPathXmlApplicationContext("applicationContext.xml"); UserDao user=app.getBean(UserDao.class); User u=new User(); u.setAge(1); u.setName("King"); user.insert(u); app.close(); }
這樣就實現了Spring與Hibernate的整合,去掉了hibernate.cfg.xml文件,由原來直接操作SessionFactory,Session到現在我們操作Spring提供的類來操作。
解讀:
之前我們使用Spring的JdbcTemplate時,需要DAL層對象繼承JdbcDaoSupport,而我們整合了Hibernate,就需要DAO對象繼承HibernateDaoSupport。而我們繼承這個類是為了得到HibernateTemplate對象,來操作數據庫,HibernateTemplate使用起來和Session類似,其底層就是采用Session。而數據源等等各種配置肯定在我們得到HibernateTemplate對象之前啦,所以這時肯定就需要注入【原來寫在hibernate.cfg.xml中的配置信息】。說到HibernateDaoSupport的注入,要麼注入SessionFactory,要麼注入HibernateTemplate。這裡我們注入SessionFactory。可以在上面的配置文件中看出。
說一下applicationContext.xml文件,我們要得到UserDaoImpl對象,需要往裡面注入SessionFactory。我們要得到SessionFactory,需要往裡面注入DataSource與各種配置信息。這裡的數據源(連接池)我們采用C3P0。
說一下UserDaoImpl對象的生成,我們得到的肯定是代理對象,因為下面已經加上了<aop:config></aop:config>標簽,所以是全自動方式的代理對象,是基於接口的JDK代理對象方式生成,當然也可以設置為CGlib生成代理對象,需要添加額外配置<proxy-target-class="true">
這裡的代理對象的通知就是事務。通過txManager來對目標方法的前後添加【開啟事務】【提交事務/回滾事務等等操作】,然後返回代理對象。這時,我們所有的操作都是在事務之中,只要發生異常,事務就會回滾。這裡是聲明式事務。
說一下我們加的幾個包的作用。
Spring-orm.jar 它就是Spring整合Hibernate的關鍵,它依賴spring-jdbc.jar包,因為它會用到其中的LobHandler類。
Spring-web.jar 它是讓Tomcat容器加載Spring配置文件的關鍵。因為我們需要容器為我們加載配置文件,創建代理對象。
slf4j-api.jar SLF4J(Simple logging Facade for Java)不是一個真正的日志實現,而是一個抽象層,它允許你在後台使用任意一個日志類庫。如果是在編寫供內外部都可以使用的API或者通用類庫,那麼你真不會希望使用你類庫的客戶端必須使用你選擇的日志類庫。所以我們若要使用log4j,則還需引入slf4j-log4j.jar包。
----------------------------------------[Struts2+Spring]整合分割線-------------------------------------------------------------------
struts.xml:
<struts> <constant name="struts.devMode" value="true"/> <package name="User" namespace="/User" extends="struts-default"> <!-- action的實例由struts來創建 --> <action name="AddUser" class="king.action.UserAction" method="addUser"> <result name="success" type="redirect">/success.html</result> </action> </package> </struts>
web.xml:
<!-- 這兩處配置是Spring為了整合Web,這是因為原來手動加載.xml文件,現在需要Tomcat啟動加載.xml文件, 所以需要Spring整合Web,讓Tomcat加載配置文件,來實現注入,生成代理等等各種功能。spring-web.jar包在起作用。 --> <!-- 很多Spring的jar包依賴commons-logging.jar包 Commons-logging.jar的目的是為“所有的Java日志實現”提供一個統一的接口。 --> <!-- 配置spring配置文件的位置 --> <context-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:applicationContext.xml</param-value> </context-param> <!-- 如果只配置spring的監聽,則會加載默認的位置:src/applicationContext.xml 作用:啟動Web容器時,自動裝配ApplicationContext的配置信息。 --> <listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener>
<!-- struts的過濾器 --> <filter> <filter-name>struts2</filter-name> <filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class> </filter> <filter-mapping> <filter-name>struts2</filter-name> <url-pattern>/*</url-pattern> </filter-mapping>
UserAction類:
/** * ActionSupport出自xwork-core.jar * * 疑問:userDao是誰注入的? * 答案:struts2-spring-plugin.jar中的struts-plugin.xml,設置默認的對象工廠為Spring,由Spring進行注入。 * * @author King * */ public class UserAction extends ActionSupport implements ModelDriven<User> { private static final long serialVersionUID = 1885943501464870512L; private User user=new User(); @Override public User getModel() { return user; } private UserDao userDao; public void setUserDao(UserDao userDao) { this.userDao = userDao; } public String addUser(){ boolean isOk=userDao.insert(user); if(isOk){ return "success"; }else{ return "fail"; } } }
這樣我們就實現了Struts2+Spring的整合,關鍵之處在於web.xml的配置和struts2-spring-plugin.jar中的struts-plugin.xml,兩個配置文件。
解讀:
Struts2有一系列的配置文件按照一定順序加載 1 :default.propertities 2:struts-default.xml 3:struts-plugin.xml 4 :struts.xml。
而我們引入了struts2-spring-plugin.jar中,其中有struts-plugin.xml,裡面設置了對象工廠由Spring來創建。所以明明這裡的UserAction是Struts2創建的,但是UserDao對象是由Spring注入的,注意:<bean id="userDao"></bean>與代碼中 private UserDao userDao; 名字要相同,這樣Spring才能注入,否則會出現空指針異常。
之前我們測試類中是手動加載applicationContext.xml文件,得到其中的bean,而整合了Struts2,我們發現沒有這個過程了,而Spring照樣正常工作,這是為啥呢?
這就是上面我們配置監聽類<listener-class>ContextLoaderListener</listener-class>的作用:加載applicationContext.xml文件,所以我們沒有手動加載配置文件而Spring正常工作的原因。你可以看,這個類所屬的包就是spring-web.jar,所以這個包也是整合它們的關鍵。
好了,到現在為止,SSH三大框架整合完畢。