9、SSH整合续2

到目前,struts、hibernate、spring在总体上整合到一块了,但是struts组件还没有整合进来

14、让spring接管struts(就是接管Action控件)

14.1)在struts-config.xml文件中添加如下代码配置:

<!-- 配置代理请求处理 DelegatingRequestProcessor它的用处是将请求处理转到DelegatingRequestProcessor,从而在spring中查找action,从而由spring管理action -->
	<controller>
		<set-property property="processorClass" value="org.springframework.web.struts.DelegatingRequestProcessor" />
	</controller>

14.2)在applicationContext.xml文件中配置我们的action路径

<!-- 配置struts的组件:Action -->
<bean name="/login" class="com.cdtax.web.action.LoginAction">
</bean>
这里的name,其值要与struts-congfig.xml中的action的path值相同,class就是action的具体类名,因为这里已经有了class了,struts-config.xml中的action的type就可以去掉。

struts-config.xml文件:

<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE struts-config PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 1.3//EN" "http://struts.apache.org/dtds/struts-config_1_3.dtd">
<struts-config>
	<form-beans>
		<form-bean name="employeeForm" type="com.cdtax.web.forms.EmployeeForm" />
	</form-beans>
	<action-mappings>
		<action path="/login" parameter="flag" name="employeeForm">
			<forward name="ok" path="/WEB-INF/mainFrame.jsp" />
			<forward name="err" path="/WEB-INF/login.jsp" />
		</action>
	</action-mappings>
	
	<!-- 配置代理请求处理 DelegatingRequestProcessor它的用处是 -->
	<controller>
		<set-property property="processorClass" value="org.springframework.web.struts.DelegatingRequestProcessor" />
	</controller>
</struts-config>

14.3)这样我们就可以通过spring容器来获取action,并且可以同时配置action的一些属性

要通过spring的方式管理action,对action进行一些改造:

package com.cdtax.web.action;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.struts.action.ActionForm;
import org.apache.struts.action.ActionForward;
import org.apache.struts.action.ActionMapping;
import org.apache.struts.actions.DispatchAction;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.context.support.WebApplicationContextUtils;

import com.cdtax.domain.Employee;
import com.cdtax.service.interfaces.EmployeeServiceInter;
import com.cdtax.web.forms.EmployeeForm;

public class LoginAction extends DispatchAction
{
//	ApplicationContext ac = new ClassPathXmlApplicationContext("");
	
	//为了使用spring属性注入,定义变量及set方法
	private EmployeeServiceInter employeeServiceInter;
	
	public void setEmployeeServiceInter(EmployeeServiceInter employeeServiceInter)
	{
		System.out.println("setEmployeeServiceInter 方法被调用");
		this.employeeServiceInter = employeeServiceInter;
	}

	//响应登陆请求
	public ActionForward login(ActionMapping arg0, ActionForm arg1,
			HttpServletRequest arg2, HttpServletResponse arg3) throws Exception
	{
		System.out.println("------通过新的方式响应请求:spring管理-----");
		//通过下面语句,可以直接获取到spring容器实例,即我们前面讲的ApplicationContext
		//当通过spring来进行action的管理时,就不需要下面这行代码了,注释掉
//		WebApplicationContext ctx = 
//				WebApplicationContextUtils
//				.getWebApplicationContext(this.getServlet().getServletContext());
		
		
		//从spring容器中获取bean
		//当使用spring来进行action的管理时,这个实例可以通过spring的依赖注入来注入进来,定义一个变量employeeServiceInter,生成set方法
		//然后在spring配置文件中的bean中进行属性配置,如下:
		//		<bean name="/login" class="com.cdtax.web.action.LoginAction">
		//		<property name="employeeServiceInter" ref="employeeService" />
		//		</bean>
//		EmployeeServiceInter employeeServiceInter = (EmployeeServiceInter) ctx.getBean("employeeService");
	
		//取出表单,我们先打通练习,我们简单验证
		EmployeeForm employeeForm = (EmployeeForm)arg1;
		
		//构建一个Employee对象
		Employee e = new Employee();
		e.setId(Integer.parseInt(employeeForm.getId()));
		e.setPwd(employeeForm.getPwd());
		e = employeeServiceInter.checkEmployee(e);
		if(e != null)
		{
			//把雇员信息放入session,后面可以使用
			arg2.getSession().setAttribute("loginer", e);
			return arg0.findForward("ok");
		}
		else
		{
			return arg0.findForward("err");
		}
						
	}
	
	//响应注销请求
	public ActionForward logout(ActionMapping arg0, ActionForm arg1,
			HttpServletRequest arg2, HttpServletResponse arg3) throws Exception
	{
		// TODO Auto-generated method stub
		return super.execute(arg0, arg1, arg2, arg3);
	}
	
}
当通过spring来进行action的管理时,就不需要下面这行代码来手动获取applicationContext实例了,注释掉
WebApplicationContext ctx = WebApplicationContextUtils.getWebApplicationContext(this.getServlet().getServletContext());

当使用spring来进行action的管理时,employeeServiceInter这个实例可以通过spring的依赖注入来注入进来,定义一个变量employeeServiceInter,生成set方法,/然后在spring配置文件中的bean中进行属性配置,如下:
<bean name="/login" class="com.cdtax.web.action.LoginAction">
<property name="employeeServiceInter" ref="employeeService" />
</bean>
这一行就注释掉// EmployeeServiceInter employeeServiceInter = (EmployeeServiceInter) ctx.getBean("employeeService");

我们在employeeServiceInter的set方法中添加打印语句,看一下它的调用时机。

然后再看看applicationContext.xml的配置情况:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
		xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
		xmlns:context="http://www.springframework.org/schema/context"
		xmlns:tx="http://www.springframework.org/schema/tx"
		xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
				http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd
				http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">

<!-- 配置一个testService对象,测试spring集成是否成功用 -->
<bean id="testService"  class="com.cdtax.test.TestService">
	<property name="name" value="小明"></property>
</bean>

<!-- 配置数据源 -->
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
	<property name="driverClassName" value="com.mysql.jdbc.Driver" />
	<property name="url" value="jdbc:mysql://localhost:3306/hibernate" />
	<property name="username" value="root" />
	<property name="password" value="root" />
	<!-- 连接池启动时的初始值 -->
	<property name="initialSize" value="3" />
	<!-- 连接池的最大值 -->
	<property name="maxActive" value="500" />
	<!-- 最大空闲值.当经过一个高峰时间后,连接池可以慢慢将已经用不到的连接慢慢释放一部分,一直减少到maxIdle为止 -->
	<property name="maxIdle" value="2" />
	<!-- 最小空闲值.当空闲的连接数少于阀值时,连接池就会预申请去一些连接,以免洪峰来时来不及申请 -->
	<property name="minIdle" value="1" />
</bean>

<!-- 配置会话工厂() -->
<bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
	<!-- 设置数据源 -->
	<property name="dataSource" ref="dataSource" /><!-- 应该想到类中有setDataSource()方法 -->
	<!-- 接管了haibernate的对象映射文件 -->
	<property name="mappingResources"><!-- 应该想到类中有setMappingResources()方法 -->
		<list>
			<value>com/cdtax/domain/Employee.hbm.xml</value>
		</list>
	</property>
	
	<property name="hibernateProperties">
		<value>
			hibernate.dialect=org.hibernate.dialect.MySQLDialect
			hibernate.hbm2ddl.auto=update
			hibernate.show_sql=true
			hibernate.format_sql=true
			<!-- 配置hibernate二级缓存 -->
			hibernate.cache.use_second_level_cache=true
        	hibernate.cache.provider_class=org.hibernate.cache.EhCacheProvider
        	hibernate.generate_statistics=true
		</value>
	</property>
</bean>

<!-- 配置EmployeeService对象 -->
<bean id="employeeService" class="com.cdtax.service.impl.EmployeeService">
	<property name="sessionFactory" ref="sessionFactory"></property>
</bean>

<!-- 配置struts的组件:Action -->
<bean name="/login" class="com.cdtax.web.action.LoginAction">
	<property name="employeeServiceInter" ref="employeeService" />
</bean>

<!-- 配置事务管理器,统一管理sessionFactory的事务 -->
<bean id="txManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
	<property name="sessionFactory" ref="sessionFactory" />
</bean>

<!-- 启用事务注解 -->
<tx:annotation-driven transaction-manager="txManager"/>

</beans>
注意看配置struts的组件:action的配置项

这些完成后,进行重新部署,然后启动tomcat,这时请注意观察,虽然我们还没有进行任何登陆操作(如打开登陆页面等),也就是没有向web服务器发送任何请求,但是:setEmployeeServiceInter 方法被调用这一句已经被打印出来了,说明在初始化时,action就被创建了,并且其属性employeeServiceInter也被注入了。这就是spring容器初始化的工作。

至此,struts与spring整合完成,struts的action由spring管理

14.4)通过使用spring来接管我们的Action还有一个好处,可以解决action是单例的问题。对于struts框架,action都是单例的,也就是说,在整个web生命周期中,同一个action只存在一个实例,存在线程安全与处理效率方面的问题,通过在applicationContext.xml文件中配置属性

<bean scope="singleton/prototype/request/session/globalsession" />,可以定义bean是单例的(singleton)还是每次请求都生成一个新的实例(如prototype)等。

举例测试,修改LoginAction:

package com.cdtax.web.action;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.struts.action.ActionForm;
import org.apache.struts.action.ActionForward;
import org.apache.struts.action.ActionMapping;
import org.apache.struts.actions.DispatchAction;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.context.support.WebApplicationContextUtils;

import com.cdtax.domain.Employee;
import com.cdtax.service.interfaces.EmployeeServiceInter;
import com.cdtax.web.forms.EmployeeForm;

public class LoginAction extends DispatchAction
{
//	ApplicationContext ac = new ClassPathXmlApplicationContext("");
	
	//为了使用spring属性注入,定义变量及set方法
	private EmployeeServiceInter employeeServiceInter;
	
	//测试单例与否的变量
	private int a = 0;
	
	public void setEmployeeServiceInter(EmployeeServiceInter employeeServiceInter)
	{
		System.out.println("setEmployeeServiceInter 方法被调用");
		this.employeeServiceInter = employeeServiceInter;
	}

	//响应登陆请求
	public ActionForward login(ActionMapping arg0, ActionForm arg1,
			HttpServletRequest arg2, HttpServletResponse arg3) throws Exception
	{
		System.out.println("------通过新的方式响应请求:spring管理-----");
		
		//如果每次请求打印a都相同,证明不是单例,如果每次都是递增,说明是单例
		System.out.println("a = " + (++a));
		//通过下面语句,可以直接获取到spring容器实例,即我们前面讲的ApplicationContext
		//当通过spring来进行action的管理时,就不需要下面这行代码了,注释掉
//		WebApplicationContext ctx = 
//				WebApplicationContextUtils
//				.getWebApplicationContext(this.getServlet().getServletContext());
		
		
		//从spring容器中获取bean
		//当使用spring来进行action的管理时,这个实例可以通过spring的依赖注入来注入进来,定义一个变量employeeServiceInter,生成set方法
		//然后在spring配置文件中的bean中进行属性配置,如下:
		//		<bean name="/login" class="com.cdtax.web.action.LoginAction">
		//		<property name="employeeServiceInter" ref="employeeService" />
		//		</bean>
//		EmployeeServiceInter employeeServiceInter = (EmployeeServiceInter) ctx.getBean("employeeService");
	
		//取出表单,我们先打通练习,我们简单验证
		EmployeeForm employeeForm = (EmployeeForm)arg1;
		
		//构建一个Employee对象
		Employee e = new Employee();
		e.setId(Integer.parseInt(employeeForm.getId()));
		e.setPwd(employeeForm.getPwd());
		e = employeeServiceInter.checkEmployee(e);
		if(e != null)
		{
			//把雇员信息放入session,后面可以使用
			arg2.getSession().setAttribute("loginer", e);
			return arg0.findForward("ok");
		}
		else
		{
			return arg0.findForward("err");
		}
						
	}
	
	//响应注销请求
	public ActionForward logout(ActionMapping arg0, ActionForm arg1,
			HttpServletRequest arg2, HttpServletResponse arg3) throws Exception
	{
		// TODO Auto-generated method stub
		return super.execute(arg0, arg1, arg2, arg3);
	}
	
}
spring的配置文件applicationContext.xml不变,部署运行,第一次请求结果

------通过新的方式响应请求:spring管理-----
a = 1

第二次请求:

------通过新的方式响应请求:spring管理-----
a = 2

第三次请求

------通过新的方式响应请求:spring管理-----
a = 3

说明是同一个action在为我们服务,是单例的

修改applicationContext.xml中的action的bean配置:

<!-- 配置struts的组件:Action -->
<bean name="/login" scope="prototype" class="com.cdtax.web.action.LoginAction">
	<property name="employeeServiceInter" ref="employeeService" />
</bean>

再次部署运行,这次不管请求多少次,结果都是

------通过新的方式响应请求:spring管理-----
a = 1

说明每次为我们服务的action都是新生成的实例。

spring管理struts的action的流程图:

9、SSH整合续2_第1张图片

15、解决中文乱码问题

思路1:自己配置过滤器

步骤:(1)开发一个Filter(过滤器,过滤器属于web层的,因为他的本质也是一个servlet)

package com.cdtax.web.filter;

import java.io.IOException;
import java.io.PrintWriter;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

public class MyFilter extends HttpServlet implements Filter
{

	public void doFilter(ServletRequest request, ServletResponse response,
			FilterChain chain) throws IOException, ServletException
	{
		request.setCharacterEncoding("utf-8");
		chain.doFilter(request, response);
	}

	public void init(FilterConfig filterConfig) throws ServletException
	{
		// TODO Auto-generated method stub
		
	}
	

}

        (2)在web.xml中配置过滤器

<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.5" 
	xmlns="http://java.sun.com/xml/ns/javaee" 
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
	xsi:schemaLocation="http://java.sun.com/xml/ns/javaee 
	http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
 
  <!-- 配置struts -->
  <servlet>
  	<servlet-name>struts</servlet-name>
  	<servlet-class>org.apache.struts.action.ActionServlet</servlet-class>
  	<init-param>
  		<param-name>config</param-name>
  		<param-value>/WEB-INF/struts-config.xml</param-value>
  	</init-param>
  	<load-on-startup>0</load-on-startup>
  </servlet>
  <servlet>
    <description>This is the description of my J2EE component</description>
    <display-name>This is the display name of my J2EE component</display-name>
    <servlet-name>MyFilter</servlet-name>
    <servlet-class>com.cdtax.web.filter.MyFilter</servlet-class>
  </servlet>

  <servlet-mapping>
  	<servlet-name>struts</servlet-name>
  	<url-pattern>*.do</url-pattern>
  </servlet-mapping>
  <servlet-mapping>
    <servlet-name>MyFilter</servlet-name>
    <url-pattern>/MyFilter</url-pattern>
  </servlet-mapping>
  
<!-- 指定spring的配置文件,默认从web根目录寻找配置文件,我们可以通过spring提供的classpath:前缀指定从类路径下寻找 -->

<context-param>
   <param-name>contextConfigLocation</param-name>
   <param-value>classpath:applicationContext.xml</param-value>
</context-param> 
<!-- 对Spring容器进行实例化 -->

<listener>
      <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>

<!-- 配置自己写的过滤器解决乱码问题 -->
<filter>
	<filter-name>MyFilter</filter-name>
	<filter-class>com.cdtax.web.filter.MyFilter</filter-class>
</filter>
<filter-mapping>
	<filter-name>MyFilter</filter-name>
	<url-pattern>/*</url-pattern>
</filter-mapping>
 </web-app>
思路2:使用spring框架提供的处理中文乱码的过滤器

<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.5" 
	xmlns="http://java.sun.com/xml/ns/javaee" 
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
	xsi:schemaLocation="http://java.sun.com/xml/ns/javaee 
	http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
 
  <!-- 配置struts -->
  <servlet>
  	<servlet-name>struts</servlet-name>
  	<servlet-class>org.apache.struts.action.ActionServlet</servlet-class>
  	<init-param>
  		<param-name>config</param-name>
  		<param-value>/WEB-INF/struts-config.xml</param-value>
  	</init-param>
  	<load-on-startup>0</load-on-startup>
  </servlet>
  <servlet>
    <description>This is the description of my J2EE component</description>
    <display-name>This is the display name of my J2EE component</display-name>
    <servlet-name>MyFilter</servlet-name>
    <servlet-class>com.cdtax.web.filter.MyFilter</servlet-class>
  </servlet>

  <servlet-mapping>
  	<servlet-name>struts</servlet-name>
  	<url-pattern>*.do</url-pattern>
  </servlet-mapping>
  <servlet-mapping>
    <servlet-name>MyFilter</servlet-name>
    <url-pattern>/MyFilter</url-pattern>
  </servlet-mapping>
  
<!-- 指定spring的配置文件,默认从web根目录寻找配置文件,我们可以通过spring提供的classpath:前缀指定从类路径下寻找 -->

<context-param>
   <param-name>contextConfigLocation</param-name>
   <param-value>classpath:applicationContext.xml</param-value>
</context-param> 
<!-- 对Spring容器进行实例化 -->

<listener>
      <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>

<!-- 配置自己写的过滤器解决乱码问题 -->
<!-- 
<filter>
	<filter-name>MyFilter</filter-name>
	<filter-class>com.cdtax.web.filter.MyFilter</filter-class>
</filter>
<filter-mapping>
	<filter-name>MyFilter</filter-name>
	<url-pattern>/*</url-pattern>
</filter-mapping>
 -->

<!-- 使用spring框架提供的过滤器 解决乱码问题-->
<filter>
	<filter-name>encoding</filter-name>
	<filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
	<init-param>
		<param-name>encoding</param-name>
		<param-value>UTF-8</param-value>
	</init-param>
</filter>
<filter-mapping>
	<filter-name>encoding</filter-name>
	<url-pattern>/*</url-pattern>
</filter-mapping>

 </web-app>

org.springframework.web.filter.CharacterEncodingFilter的作用与我们自己写的filter作用是一样的。



你可能感兴趣的:(9、SSH整合续2)