首页 > 代码库 > struts2.3.16 整合spring4.0.5 和 hibernate4.3.0
struts2.3.16 整合spring4.0.5 和 hibernate4.3.0
1、由struts2 框架自身根据struts.xml 中 的映射实例化Action 对象
Action 类代码如下:
package com.hasonger.ssh.action; import java.util.Date; import com.hasonger.ssh.entity.User; import com.hasonger.ssh.service.UserService; import com.opensymphony.xwork2.ActionSupport; import com.opensymphony.xwork2.ModelDriven; public class UserAction extends ActionSupport implements ModelDriven<User> { private static final long serialVersionUID = 1L; private User user; private UserService userService; public UserAction(){ System.out.println("UserAction construstor........"); } public void setUserService(UserService userService) { this.userService = userService; } public String register() { user.setRegTime(new Date()); userService.save(user); return SUCCESS; } @Override public User getModel() { return user = new User(); } }
struts.xml 文件配置如下:
<package name="default" namespace="/" extends="struts-default"> <action name="user_register" class="com.hasonger.ssh.action.UserAction" method="register"> <result>/jsp/success.jsp</result> </action> </package>
在表单页面填写注册信息后,控制台输出如下:
信息: Starting ProtocolHandler ["ajp-apr-8009"] 七月 10, 2014 4:32:31 下午 org.apache.catalina.startup.Catalina start 信息: Server startup in 13551 ms UserAction construstor........ Hibernate: insert into ssh.user (USERNAME, PASSWORD, REG_TIME) values (?, ?, ?) UserAction construstor........ Hibernate: insert into ssh.user (USERNAME, PASSWORD, REG_TIME) values (?, ?, ?)
经过实际测试,并结合struts文档可以看出,在WEB 应用容器启动过程中Action 不会实例化,在客户发出action 请求后 由struts 框架实例化Action 对象,并且Action 默认是多实例的,即每发出一次action 请求,struts 都会产生一个新的Action 对象。
从UserAction 类的整个代码看,UserAction 类没有交给IoC容器管理,并且userService 属性也没有显式的装配,struts2文档中提到像这种情况,由struts 框架产生Acition 实例,并结合Spring IoC 容器自动装配依赖的属性。当然这需要给依赖的属性提供setter 方法才可以完成。
但是若把struts.xml 文件改成如下配置:
<package name="default" namespace="/" extends="struts-default"> <action name="user_register" class="userAction" method="register"> <result>/jsp/success.jsp</result> </action> </package>
再次发起action 请求时会产生错误:错误信息提示不能实例化userAction.
2、由Spring IoC 容器产生Action 实例,并负责管理Action 对象的生命周期:
Action 类代码如下:
package com.hasonger.ssh.action; import java.util.Date; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Controller; import com.hasonger.ssh.entity.User; import com.hasonger.ssh.service.UserService; import com.opensymphony.xwork2.ActionSupport; import com.opensymphony.xwork2.ModelDriven; @Controller public class UserAction extends ActionSupport implements ModelDriven<User> { private static final long serialVersionUID = 1L; private User user; private UserService userService; public UserAction(){ System.out.println("UserAction construstor........"); } @Autowired public void setUserService(UserService userService) { this.userService = userService; } public String register() { user.setRegTime(new Date()); userService.save(user); return SUCCESS; } @Override public User getModel() { return user = new User(); } }
struts.xml 配置如下:
<package name="default" namespace="/" extends="struts-default"> <action name="user_register" class="com.hasonger.ssh.action.UserAction" method="register"> <result>/jsp/success.jsp</result> </action> </package>
应用程序部署在WEB 容器后,启动WEB 容器控制台输出如下:
七月 10, 2014 5:16:23 下午 org.apache.catalina.core.ApplicationContext log 信息: No Spring WebApplicationInitializer types detected on classpath 七月 10, 2014 5:16:23 下午 org.apache.catalina.core.ApplicationContext log 信息: Initializing Spring root WebApplicationContext log4j:WARN No appenders could be found for logger (org.springframework.web.context.ContextLoader). log4j:WARN Please initialize the log4j system properly. log4j:WARN See http://logging.apache.org/log4j/1.2/faq.html#noconfig for more info. UserAction construstor........ 七月 10, 2014 5:16:28 下午 org.apache.coyote.AbstractProtocol start 信息: Starting ProtocolHandler ["http-apr-8080"] 七月 10, 2014 5:16:28 下午 org.apache.coyote.AbstractProtocol start 信息: Starting ProtocolHandler ["ajp-apr-8009"] 七月 10, 2014 5:16:28 下午 org.apache.catalina.startup.Catalina start 信息: Server startup in 14104 ms
从控制台输出的信息可以看出,Action 对象交给IoC 容器管理后,在服务器启动过程中IoC 容器初始化的同时也实例化了Action 对象。所以默认情况下,Action 的作用域是singleton 类型的。
多次发起action 请求继续测试:IoC 容器不会再次产生Action 类的实例了
若将UserAction 的作用域手动改为singleton 类型的,再次连续发起action 请求,控制台还是会显示上面的信息。
若将struts.xml 文件配置改为如下:即将<action> 节点的class 属性改为Action 类在IoC 容器中bean的name 属性值
<package name="default" namespace="/" extends="struts-default"> <action name="user_register" class="userAction" method="register"> <result>/jsp/success.jsp</result> </action> </package>
部署应用后启动服务器过程中,控制台的输出信息如下:
信息: Initializing Spring root WebApplicationContext log4j:WARN No appenders could be found for logger (org.springframework.web.context.ContextLoader). log4j:WARN Please initialize the log4j system properly. log4j:WARN See http://logging.apache.org/log4j/1.2/faq.html#noconfig for more info. UserAction construstor........ 七月 10, 2014 5:43:26 下午 org.apache.coyote.AbstractProtocol start 信息: Starting ProtocolHandler ["http-apr-8080"] 七月 10, 2014 5:43:26 下午 org.apache.coyote.AbstractProtocol start 信息: Starting ProtocolHandler ["ajp-apr-8009"] 七月 10, 2014 5:43:26 下午 org.apache.catalina.startup.Catalina start 信息: Server startup in 13848 ms
从控制台输出的信息可以看出: 在IoC 容器管理Action 对象后,<action> 节点的class 属性值不管是Action 的全类名还是bean 的name 属性值, IoC 容器都会实例化Action 对象。
当发起多个action 请求后控制台输出如下(<action> 节点的class 属性值为userAction):
信息: Initializing Spring root WebApplicationContext log4j:WARN No appenders could be found for logger (org.springframework.web.context.ContextLoader). log4j:WARN Please initialize the log4j system properly. log4j:WARN See http://logging.apache.org/log4j/1.2/faq.html#noconfig for more info. UserAction construstor........ 七月 10, 2014 5:43:26 下午 org.apache.coyote.AbstractProtocol start 信息: Starting ProtocolHandler ["http-apr-8080"] 七月 10, 2014 5:43:26 下午 org.apache.coyote.AbstractProtocol start 信息: Starting ProtocolHandler ["ajp-apr-8009"] 七月 10, 2014 5:43:26 下午 org.apache.catalina.startup.Catalina start 信息: Server startup in 13848 ms Hibernate: insert into ssh.user (USERNAME, PASSWORD, REG_TIME) values (?, ?, ?) Hibernate: insert into ssh.user (USERNAME, PASSWORD, REG_TIME) values (?, ?, ?) Hibernate: insert into ssh.user (USERNAME, PASSWORD, REG_TIME) values (?, ?, ?)
控制台的信息显示:Action 对象的作用域是singleton 类型的。
保持<action> 节点的class 属性值为userAction,继续测试,在UserAction 的作用域设置为prototype 类型后,启动服务器过程中没有Action 类 被实例化,而是每次发起action 请求时才会实例化Action 对象。代码如下:
package com.hasonger.ssh.action; import java.util.Date; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Scope; import org.springframework.stereotype.Controller; import com.hasonger.ssh.entity.User; import com.hasonger.ssh.service.UserService; import com.opensymphony.xwork2.ActionSupport; import com.opensymphony.xwork2.ModelDriven; @Controller @Scope("prototype") public class UserAction extends ActionSupport implements ModelDriven<User> { private static final long serialVersionUID = 1L; private User user; private UserService userService; public UserAction(){ System.out.println("UserAction construstor........"); } @Autowired public void setUserService(UserService userService) { this.userService = userService; } public String register() { user.setRegTime(new Date()); userService.save(user); return SUCCESS; } @Override public User getModel() { return user = new User(); } }
若将UserAction 的userService 属性setter 方法的@Autowired 注解去除,继续测试,发起action 请求一切正常,这说明Action 类 依赖属性不手动添加@Autowired 注解 也可以被注入。
通过实际测试实验总结如下:
struts2.3.16整合spring4.0.5、hibernate4.3.0 时,Action 类实例的产生和作用域情况:
1、由struts2 框架自身根据struts.xml 中 的映射实例化Action 对象:
①Action 类的依赖属性由struts2框架自动装配,不需要手动写@Autowired 注解;
②Action 类为多实例的;
③在struts.xml 配置中<action> 字节点的class 属性值只能是Action 类的全类名;
2、由Spring IoC 容器产生Action 实例,并负责管理Action 对象的生命周期:
①默认情况下Action 实例的scope 属性值为singleton;
②在WEB 环境下必须将Action 类scope 属性值设置为prototype 类型的;
③在struts.xml 配置中 <action> 字节点的class 属性值有两种选择:一种是Action 类的全类名,另一种是Action 类在 Spring IoC 容器中bean的 name 属性值;
④Action 类的依赖属性(如:业务层的对象)既可以由struts 框架自动的装配——不需要在依赖属性的setter 方法上加@Autowired 注解,也可以手工的加上@Autowired 注解,加上后不出错。
⑤上述所有自动装配依赖属性均需要提供相应的setter 方法,否则装配不上。