一、需要的JAR文件为:Spring和Struts2框架本身需要的JAR文件以及他们所依赖的JAR文件,比如commons-logging.jar等等,另外还需要Struts2发布包中的struts2-spring-plugin-x.xx.jar。
二、整合过程:
(1)在web.xml中增加WebApplicationContext的相应配置,
以下两种配置方式本质是一样的。
1.Servlet 2.3及以上版本可以使 用监听器,相应配置如下:
<context-param> <param-name>contextConfigLocation</param-name> <param-value>/WEB-INF/classes/applicationContext.xml</param-value> </context-param> <listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener>
如果spring配置文件被命名为applicationContext.xml,并且放在WEB-INF目录下,则不需要配置<context-param>,因为ContextLoaderListener默认在WEB-INF目录下寻找名为applicationContext.xml的文件。
若要放于WEB-INF/classes/目录下,或若存在多个Spring配置文件,则需要上面的配置,且在<param-value>中依次列出,之间以逗号隔开。
2.Servlet 2.3以下版本由于不支持<listener>,需要配置<servlet>,格式如下:
<context-param> <param-name>contextConfigLocation</param-name> <param-value>/WEB-INF/classes/applicationContext.xml</param-value> </context-param> <servlet> <servlet-name>contextLoaderServlet</servlet-name> <servlet-class>org.springframework.web.context.ContextLoaderServlet</servlet-class> <load-on-startup>1</load-on-startup> </servlet>
由于该Servlet配置只是为了在容器启动时能启动ContextLoaderServlet使其工作,而不需要引用该Servlet,所以不需要配置<servlet-mapping>。
(2)在web.xml中完成加载WebApplicationContext之后,接下来就可以做到Spring和Struts2的整合了。
整合有两种方法,分别叙述如下:
1.第一种实现方法:
1)将Struts的业务逻辑控制器类配置在Spring的配置文件中,业务逻辑控制器中引用的业务类一并注入。注意,必须将业务逻辑控制器类配置为scope=”prototype”!
示例如下:
<bean id=”LoginAction” class=”yaso.struts.action.LoginAction”>
<property name=”loginDao” ref=”LoginDao”/>
</bean>
2) 在struts.xml或者等效的Struts2配置文件中配置Action时,指定<action>的class属性为Spring配置文件中相应bean的id或者name值。示例如下:
<action name=”LoginAction” class=”LoginAction”>
<result name=”success”>/index.jsp</result>
</action>
2.第二种实现方法:
1)业务类在Spring配置文件中配置,业务逻辑控制器类不需要配置,Struts2的Action像没有整合Spring之前一样配置,<action>的class属性指定业务逻辑控制器类的全限定名。
2)业务逻辑控制器类中引用的业务类不需要自己去初始化,Struts2的Spring插件会使用bean的自动装配将业务类注入进来,其实业务逻辑控制器也不是Struts2创建的,而是Struts2的Spring插件创建的。默认情况下,插件使用by name的方式装配,可以通过增加Struts2常量来修改匹配方式:设置方式为:struts.objectFactory.spring.autoWire = typeName,可选的装配参数如下:
a) name:等价于Spring配置中的autowire=”byName”,这是缺值。
b) type:等价于Spring配置中的autowire=”byType”。
c)auto:等价于Spring配置中的autowire=”autodetect”。
d)constructor:等价于Spring配置中的autowire=” constructor”。
四、如果原先在Struts2中使用了多个object factory,则需要通过Struts2常量显式指定object factory,方式如下:struts.objectFactory = spring;如果没有使用多个object factory,这一步可以省略。
五、可以通过设增加Struts2常量来指定是否使用Spring自身的类缓存机制。可以设定的值为true或false,默认为true。设置方式为:struts.objectFactory.spring.useClassCache = false。
六、至此,完成了两种方式的整合。比较这两种整合方式,其本质是一样的。不同之处在于,使用第二种自动装配的方式时,由于没有在Spring中配置业务逻辑控制器,所以需要对其配置一些AOP之类的内容时就很难实现了。
Spring Autowire自动装配
在应用中,我们常常使用<ref>标签为JavaBean注入它依赖的对象。但是对于一个大型的系统,这个操作将会耗费我们大量的资源,我们不得不花费大量的时间和精力用于创建和维护系统中的<ref>标签。
Spring为我们提供了一个自动装配的机制,在定义Bean时,<bean>标签有一个autowire属性,我们可以通过指定它来让容器为受管 JavaBean自动注入依赖对象。
<bean>的autowire属性有如下六个取值,他们的说明如下:
1、 No:即不启用自动装配。Autowire默认的值。
2、 byName:通过属性的名字的方式查找JavaBean依赖的对象并为其注入。比如说类Computer有个属性printer,指定其 autowire属性为byName后,Spring IoC容器会在配置文件中查找id/name属性为printer的bean,然后使用Seter方法为其注入。
3、 byType:通过属性的类型查找JavaBean依赖的对象并为其注入。比如类Computer有个属性printer,类型为Printer,那么,指定其autowire属性为byType后,Spring IoC容器会查找Class属性为Printer的bean,使用Seter方法为其注入。
4、 constructor:通byType一样,也是通过类型查找依赖对象。与byType的区别在于它不是使用Seter方法注入,而是使用构造子注入。
5、 autodetect:在byType和constructor之间自动的选择注入方式。
6、 default:由上级标签<beans>的default-autowire属性确定。
注意:在配置bean时,<bean>标签中Autowire属性的优先级比其上级标签高,即是说,如果在上级标签中定义default- autowire属性为byName,而在<bean>中定义为byType时,Spring IoC容器会优先使用<bean>标签的配置。
下面通过一个例子来说明如何在应用中使用自动装配(工程代码见例程3.2)。新建一个java工程,为其添加上Spring开发能力后,创建一个ioc.test包,再分别创建电脑类(Computer)、主机类(Host)和显示器类(Dispaly),为电脑类添加Host类型的属性host和Display类型的属性display,再添加一个run方法,让电脑可以“运行”一起来。属性代码如下:
Computer.java
package ioc.test;
public class Computer {
private Host host;
private Display display;
//电脑运行方法
public void run(){
System.out.println("你好,我是电脑,正在运行!");
System.out.print(" "+host.run()+",");
System.out.println(display.run());
}
//Geter和Seter方法,省略
}
Host.java
package ioc.test;
public class Host {
public String run(){
return "我是主机,正在运行!";
}
}
<span style="font-family:SimSun;"><span style="font-size:14px;">Display.java</span></span><pre name="code" class="java"><span style="font-family:SimSun;font-size:14px;">package ioc.test;
public class Display {
public String run(){
return "我是显示器,正在运行!";
}
}
下面便是修改Spring的配置文件,让IoC容器为我们的“电脑”自动装配“主机”和“显示器了”。分别配置两个bean,host和display。再配置一个名computer1的bean,autowire属性设为byName,同理配置computer1和computer3,autowire属性分别设为byType和default,最后设置<beans>标签的default-autowire属性为autodetect。至此,配置工作已经完成,可以看到,我们并没有显式的给computer bean它注入依赖对象host和display。配置代码如下:
<?xml version="1.0" encoding="UTF-8"?>
<beans default->
<bean id="computer1" class="ioc.test.Computer" ></bean>
<bean id="computer2" class="ioc.test.Computer" ></bean>
<bean id="computer3" class="ioc.test.Computer" ></bean>
<bean id="host" class="ioc.test.Host"></bean>
<bean id="display" class="ioc.test.Display"></bean>
</beans>
现在可以建立一个测试类来测试一下Spring时候真的为我们自动装配好了我们需要的bean。
package ioc.test;
//import省略
public class TestMain {
public static void main(String[] args) {
ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
//byName
Computer computer1 = (Computer)ac.getBean("computer1");
System.out.println("autowire="byName":");
computer1.run();
//byType
Computer computer2 = (Computer)ac.getBean("computer2");
System.out.println("autowire="byType":");
computer2.run();
//default
Computer computer3 = (Computer)ac.getBean("computer3");
System.out.println("autowire="default":");
computer3.run();
}
}
原文参考: http://blog.sina.com.cn/s/blog_511364b10100plaw.html