一直不能坐下來(lái)好好學(xué)習一下, 最近研究了spring framework, 一點(diǎn)感受:
1. IOC
傳統方法:如果動(dòng)態(tài)設置一個(gè)對象屬性,可以借助Java的Reflection機制完成,invoke()激活返回調用
Class cls = Class.forName("com.eking.User");
Method mtd = cls.getMethod("setName",new Class[]{String.class});
Object obj = (Object)cls.newInstance();
mtd.invoke(obj,new Object[]{"Erica"});
return obj;
在spring中, 面向接口編程,動(dòng)態(tài)代理機制:BeanWrapper, BeanFactory,ApplicationContext提供了管理javabean的包裝器,所有的一切都在容器中配置,dependency injection.
也可以不提供接口,CGLib與Dynamic Proxy的代理機制基本類(lèi)似,只是其動(dòng)態(tài)生成的代理對象并非某接口的實(shí)現,而是針對目標類(lèi)擴展的子類(lèi)。換句話(huà)說(shuō),Dynamic Proxy返回的動(dòng)態(tài)代理類(lèi),是目標類(lèi)所實(shí)現的接口的另一個(gè)實(shí)版本,它實(shí)現了對目標類(lèi)的代理(如同UserDAOProxy與UserDAOImp的關(guān)系)。而CGLib返回的動(dòng)態(tài)代理類(lèi),則是目標代理類(lèi)的一個(gè)子類(lèi)(代理類(lèi)擴展了UserDAOImp類(lèi))。
2. AOP
各種通知類(lèi)型有MethodInterceptor (來(lái)自AOP聯(lián)盟的攔截器API)和定義在org.springframework.aop包中的 通知接口。所有通知必須實(shí)現org.aopalliance.aop.Advice標簽接口。 取出就可使用的通知有 MethodInterceptor、 ThrowsAdvice、 BeforeAdvice和 AfterReturningAdvice。
也可以自己寫(xiě)個(gè)攔截器,在實(shí)現自己的方法之前出發(fā)某個(gè)動(dòng)作,執行一些處理。
3. WebApplicationContext
1) web.xml 中通過(guò)聲明監聽(tīng)器接口 或servlet類(lèi)加載
通過(guò):
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener-->
或:
<servlet>
<servlet-name>SpringContextServlet</servlet-name>
<servlet-class>org.springframework.web.context.ContextLoaderServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
Web 容器會(huì )自動(dòng)加載 /WEB-INF/applicationContext.xml 初始化 ApplicationContex t實(shí)例;
也可以通過(guò)
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>/WEB-INF/applicationContext-*.xml</param-value>
</context-param>
使 Web 容器加載指定名稱(chēng)路徑的 Spring 配置文件。
我個(gè)人認為Listerner要比Servlet更好一些,因為Listerner監聽(tīng)應用的啟動(dòng)和結束,而Servlet得啟動(dòng)要稍微延遲一些,如果在這時(shí)要做一些業(yè)務(wù)的操作,啟動(dòng)的前后順序是有影響的。
那么在ContextLoaderListener和ContextLoaderServlet中到底做了什么呢?
以ContextLoaderListener為例,我們可以看到
public void contextInitialized(ServletContextEvent event) {
this.contextLoader = createContextLoader();
this.contextLoader.initWebApplicationContext(event.getServletContext());
}
protected ContextLoader createContextLoader() {
return new ContextLoader();
}
ContextLoader是一個(gè)工具類(lèi),用來(lái)初始化WebApplicationContext,其主要方法就是initWebApplicationContext,我們繼續追蹤initWebApplicationContext這個(gè)方法(具體代碼我不貼出,大家可以看Spring中的源碼),我們發(fā)現,原來(lái)ContextLoader是把WebApplicationContext(XmlWebApplicationContext是默認實(shí)現類(lèi))放在了ServletContext中,ServletContext也是一個(gè)“容器”,也是一個(gè)類(lèi)似Map的結構,而WebApplicationContext在ServletContext中的KEY就是WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE,我們如果要使用WebApplicationContext則需要從ServletContext取出,Spring提供了一WebApplicationContextUtils類(lèi),可以方便的取出WebApplicationContext,只要把ServletContext傳入就可以了。
2) struts-config.xml 中通過(guò)插件加載
通過(guò)
<plug-in className="org.springframework.web.struts.ContextLoaderPlugIn">
<set-property property="contextConfigLocation"
value="/WEB-INF/applicationContext.xml,
/WEB-INF/action-servlet.xml"/>
</plug-in>
來(lái)加載 Spring 配置文件。
在struts-config.xml中Action的配置變成類(lèi)似下面的樣子
<action attribute="aForm" name="aForm" path="/aAction" scope="request" type="org.springframework.web.struts.DelegatingActionProxy">
<forward name="forward" path="forward.jsp" />
</action>
別急,我們還是來(lái)看一下ContextLoaderPlugIn的源碼(源碼不再貼出),我們可以發(fā)現,原來(lái)ContextLoaderPlugIn仍然是把WebApplicationContext放在ServletContext中,只是這個(gè)KEY不太一樣了,這個(gè)KEY值為ContextLoaderPlugIn.SERVLET_CONTEXT_PREFIX+ModuleConfig.getPrefix()(具體請查看源代碼),這下好了,我們知道了WebApplicationContext放在哪里,只要我們在Web應用中能夠取到ServletContext也就能取到WebApplicationContext了
4.Struts+spring+hibernate spring live中myusers,一個(gè)3層架構的web 程序,通過(guò)一個(gè)Action 來(lái)調用業(yè)務(wù)代理,再通過(guò)它來(lái)回調 DAO類(lèi)。下面的流程圖表示了MyUsers是如何工作的。數字表明了流程的先后順序,從web層(UserAction)到中間層(UserManager),再到數據層(UserDAO),然后返回。 Spring是AOP, UserManager和UserDAO都是接口. 1) web層(UserAction) :調用中間層的接口方法,將UserManager作為屬性注入 2) 中間層(UserManager):將UserDAO作為屬性注入,其實(shí)現主要是調用數據層接口的一些方法; 它處于事務(wù)控制中 3) 數據層(UserDAO):實(shí)現類(lèi)繼承HibernateDaoSupport類(lèi),在該類(lèi)中可以調用getHibernateTemplate()的一些方法執行具體的數據操作。 spring配置部分文件: <bean id="userDAO" class="com.eking.entity.UserDAOImp"> <property name="sessionFactory"> <ref local="sessionFactory" /> </property> </bean> <bean id="userManagerTarget" class="com.eking.service.UserManagerImp"> <property name="userDAO"><ref local="userDAO"/></property> </bean> <bean id="userManager" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"> <property name="transactionManager"> <ref local="transactionManager" /> </property> <property name="target"> <ref local="userManagerTarget" /> </property> <property name="transactionAttributes"> <props> <prop key="insert*">PROPAGATION_REQUIRED</prop> <prop key="get*">PROPAGATION_REQUIRED</prop> <prop key="is*">PROPAGATION_REQUIRED</prop> </props> </property> </bean> <bean name="/login" class="com.eking.struts.action.LoginAction" singleton="false"> <property name="userManager"> <ref local="userManager" /> </property> </bean>
spring live中myusers,一個(gè)3層架構的web 程序,通過(guò)一個(gè)Action 來(lái)調用業(yè)務(wù)代理,再通過(guò)它來(lái)回調 DAO類(lèi)。下面的流程圖表示了MyUsers是如何工作的。數字表明了流程的先后順序,從web層(UserAction)到中間層(UserManager),再到數據層(UserDAO),然后返回。
Spring是AOP, UserManager和UserDAO都是接口.
1) web層(UserAction) :調用中間層的接口方法,將UserManager作為屬性注入
2) 中間層(UserManager):將UserDAO作為屬性注入,其實(shí)現主要是調用數據層接口的一些方法; 它處于事務(wù)控制中
3) 數據層(UserDAO):實(shí)現類(lèi)繼承HibernateDaoSupport類(lèi),在該類(lèi)中可以調用getHibernateTemplate()的一些方法執行具體的數據操作。
spring配置部分文件:
<bean id="userDAO" class="com.eking.entity.UserDAOImp">
<property name="sessionFactory">
<ref local="sessionFactory" />
</property>
</bean>
<bean id="userManagerTarget" class="com.eking.service.UserManagerImp">
<property name="userDAO"><ref local="userDAO"/></property>
</bean>
<bean id="userManager" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
<property name="transactionManager">
<ref local="transactionManager" />
</property>
<property name="target">
<ref local="userManagerTarget" />
</property>
<property name="transactionAttributes">
<props>
<prop key="insert*">PROPAGATION_REQUIRED</prop>
<prop key="get*">PROPAGATION_REQUIRED</prop>
<prop key="is*">PROPAGATION_REQUIRED</prop>
</props>
</property>
</bean>
<bean name="/login" class="com.eking.struts.action.LoginAction" singleton="false">
<property name="userManager">
<ref local="userManager" />
</property>
</bean>
聯(lián)系客服