分享

spring MVC3深入研究(2)

 BlazerOfIT 2012-03-29

十一、如何实现全局的异常处理?

在spring MVC的配置文件中:

Xml代码  
<!-- 总错误处理-->  <bean id="exceptionResolver" class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">      <property name="defaultErrorView">            <value>/error/error</value>      </property>      <property name="defaultStatusCode">            <value>500</value>      </property>     <property name="warnLogCategory">            <value>org.springframework.web.servlet.handler.SimpleMappingExceptionResolver</value>      </property>     </bean>   

 

这里主要的类是SimpleMappingExceptionResolver类,和他的父类AbstractHandlerExceptionResolver类。

具体可以配置哪些属性,我是通过查看源码知道的。

你也可以实现HandlerExceptionResolver接口,写一个自己的异常处理程序。spring的扩展性是很好的。

 

 

通过SimpleMappingExceptionResolver我们可以将不同的异常映射到不同的jsp页面(通过exceptionMappings属性的配置)。

 

同时我们也可以为所有的异常指定一个默认的异常提示页面(通过defaultErrorView属性的配置),如果所抛出的异常在exceptionMappings中没有对应的映射,则Spring将用此默认配置显示异常信息。

注意这里配置的异常显示界面均仅包括主文件名,至于文件路径和后缀已经在viewResolver中指定。如/error/error表示/error/error.jsp

 

 

显示错误的jsp页面:

Html代码  
<%@ page language="java" contentType="text/html; charset=GBK"      pageEncoding="GBK"%>  <%@ page import="java.lang.Exception"%>  <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www./TR/html4/loose.dtd">  <html>  <head>  <meta http-equiv="Content-Type" content="text/html; charset=GBK">  <title>错误页面</title>  </head>  <body>  <h1>出错了</h1>  <%  Exception e = (Exception)request.getAttribute("exception");  out.print(e.getMessage());  %>  </body>  </html>  

其中一句:request.getAttribute("exception"),key是exception,也是在SimpleMappingExceptionResolver类默认指定的,是可能通过配置文件修改这个值的,大家可以去看源码。

 

 参考文章:

http://www./wuxufeng8080/articles/191150.html

http://fangjunai.blog.163.com/blog/static/1124970520108102013839/

 

 

 

十二、如何把全局异常记录到日志中?

在前的配置中,其中有一个属性warnLogCategory,值是 “SimpleMappingExceptionResolver类的全限定名”。我是在SimpleMappingExceptionResolver 类父类AbstractHandlerExceptionResolver类中找到这个属性的。查看源码后得知:如果warnLogCategory不为空,spring就会使用apache的org.apache.commons.logging.Log日志工具,记录这个异常,级别是warn。

值:“org.springframework.web.servlet.handler.SimpleMappingExceptionResolver”,是“SimpleMappingExceptionResolver类的全限定名”。这个值不是随便写的。 因为我在log4j的配置文件中还要加入 log4j.logger.org.springframework.web.servlet.handler.SimpleMappingExceptionResolver=WARN,保证这个级别是warn的日志一定会被记录,即使log4j的根日志级别是ERROR。

 

 

 

 

 十三、如何给spring3 MVC中的Action做JUnit单元测试?

 使用了spring3 MVC后,给action做单元测试也很方便,我以前从来不给action写单元测试的,再在不同了,方便了,所以一定要写。

 

 JUnitActionBase类是所有JUnit的测试类的父类

Java代码  
package test;  import javax.servlet.http.HttpServletRequest;  import javax.servlet.http.HttpServletResponse;  import org.junit.BeforeClass;  import org.springframework.mock.web.MockServletContext;  import org.springframework.web.context.WebApplicationContext;  import org.springframework.web.context.support.XmlWebApplicationContext;  import org.springframework.web.servlet.HandlerAdapter;  import org.springframework.web.servlet.HandlerExecutionChain;  import org.springframework.web.servlet.HandlerMapping;  import org.springframework.web.servlet.ModelAndView;  import org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter;  import org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping;  /**  * 说明: JUnit测试action时使用的基类  * @author  赵磊 * @version 创建时间:2011-2-2 下午10:27:03   */   public class JUnitActionBase {      private static HandlerMapping handlerMapping;      private static HandlerAdapter handlerAdapter;      /**      * 读取spring3 MVC配置文件      */      @BeforeClass   public static void setUp() {          if (handlerMapping == null) {              String[] configs = { "file:src/springConfig/springMVC.xml" };              XmlWebApplicationContext context = new XmlWebApplicationContext();              context.setConfigLocations(configs);              MockServletContext msc = new MockServletContext();              context.setServletContext(msc);         context.refresh();              msc.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, context);              handlerMapping = (HandlerMapping) context                      .getBean(DefaultAnnotationHandlerMapping.class);              handlerAdapter = (HandlerAdapter) context.getBean(context.getBeanNamesForType(AnnotationMethodHandlerAdapter.class)[0]);             }      }        /**      * 执行request对象请求的action      *       * @param request      * @param response      * @return      * @throws Exception      */      public ModelAndView excuteAction(HttpServletRequest request, HttpServletResponse response)   throws Exception {          HandlerExecutionChain chain = handlerMapping.getHandler(request);          final ModelAndView model = handlerAdapter.handle(request, response,                  chain.getHandler());          return model;      }  }  

 

 

 

 

 

这是个JUnit测试类,我们可以new Request对象,来参与测试,太方便了。给request指定访问的URL,就可以请求目标Action了。

 

Java代码  
package test.com.app.user;  import org.junit.Assert;  import org.junit.Test;  import org.springframework.mock.web.MockHttpServletRequest;  import org.springframework.mock.web.MockHttpServletResponse;  import org.springframework.web.servlet.ModelAndView;    import test.JUnitActionBase;    /**  * 说明: 测试OrderAction的例子  * @author  赵磊  * @version 创建时间:2011-2-2 下午10:26:55   */     public class TestOrderAction extends JUnitActionBase {      @Test      public void testAdd() throws Exception {      MockHttpServletRequest request = new MockHttpServletRequest();          MockHttpServletResponse response = new MockHttpServletResponse();          request.setRequestURI("/order/add");          request.addParameter("id""1002");          request.addParameter("date""2010-12-30");          request.setMethod("POST");          // 执行URI对应的action          final ModelAndView mav = this.excuteAction(request, response);          // Assert logic          Assert.assertEquals("order/add", mav.getViewName());          String msg=(String)request.getAttribute("msg");          System.out.println(msg);      }  }  

 需要说明一下 :由于当前最想版本的Spring(Test) 3.0.5还不支持@ContextConfiguration的注解式context file注入,所以还需要写个setUp处理下,否则类似于Tiles的加载过程会有错误,因为没有ServletContext。3.1的版本应该有更好的解决方案,参见:https://jira./browse/SPR-5243

参考 :http://www./topic/828513

 

 

 

 

 十四、转发与重定向

可以通过redirect/forward:url方式转到另一个Action进行连续的处理。

可以通过redirect:url 防止表单重复提交

写法如下:

return "forward:/order/add";

return "redirect:/index.jsp";

 

 

 

 

 十五、处理ajax请求

 

1、引入下面两个jar包,我用的是1.7.2,好像1.4.2版本以上都可以,下载地址:http://wiki./JacksonDownload

jackson-core-asl-1.7.2.jar 

jackson-mapper-asl-1.7.2.jar

 

2、spring的配置文件中要有这一行,才能使用到spring内置支持的json转换。如果你手工把POJO转成json就可以不须要使用spring内置支持的json转换。

<mvc:annotation-driven />

 

3、使用@ResponseBody注解

Java代码  
/**  * ajax测试 * http://127.0.0.1/mvc/order/ajax  */    @RequestMapping("/ajax")  @ResponseBody  public Object ajax(HttpServletRequest request){      List<String> list=new ArrayList<String>();      list.add("电视");  nbsp;       list.add("洗衣机");      list.add("冰箱");      list.add("电脑");      list.add("汽车");      list.add("空调");      list.add("自行车");      list.add("饮水机");      list.add("热水器");      return list;  }  

 

 

 


19:22评论 / 浏览 (11 / 3839)分类:企业架构相关推荐
评论
11 楼 Jackie_GP 前天  
弄一个demo提供下载就好了!
10 楼 王星游 2011-09-01  
java.lang.NullPointerException
at org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter$ServletHandlerMethodResolver.useTypeLevelMapping(AnnotationMethodHandlerAdapter.java:666)
at org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter$ServletHandlerMethodResolver.resolveHandlerMethod(AnnotationMethodHandlerAdapter.java:574)
at org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter.invokeHandlerMethod(AnnotationMethodHandlerAdapter.java:431)
at org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter.handle(AnnotationMethodHandlerAdapter.java:424)
at com.watchzerg.account.controller.test.LoginControllerTest.testLogin(LoginControllerTest.java:63)
--------------------------------------------
我的Action单元测试居然在这一行报错:ModelAndView model=handlerAdapter.handle(request,response,handler);
debug发现request,response,handler三个参数都不为空,看trace是spring内部调用的时候空指针了,不太清楚我少设了什么参数:
request.setRequestURI("/login.do");
request.setServletPath("/login.do");
request.setAttribute("method", "loginCheck");
request.setAttribute("username", "watchzerg");
request.setAttribute("password", "123456");
request.setMethod("POST");
或者我少配置了什么东西……
--------------------------------------------
顺便请教下楼主,这种对web层的测试方法看起来有些hack的意思,请求应该不会经过编码的过滤器等……那么spring官方没有提供web层的测试方法吗?还是我们只有用selenium这样的工具在browser端测试?
我是初学者,见谅。
9 楼 elf8848 2011-08-25  
作者补充--错误修正:

测试OrderAction的例子  中的第22行的
request.setRequestURI("/order/add");
改为
request.setServletPath("/order/add");
 
8 楼 xhs10195 2011-08-23  
我刚遇到springMVC拦截静态文件的引用,非常感谢楼主的分享。赞一个!
7 楼 bosh 2011-08-19  
写得很有水平啊,赞一个
6 楼 elf8848 2011-04-05  
获取Spring context的两种方式:

1、servlet方式加载时:
【web.xml】
<servlet>
<servlet-name>springMVC</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath*:/springMVC.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>

spring容器放在ServletContext中的key是org.springframework.web.servlet.FrameworkServlet.CONTEXT.springMVC
注意后面的springMVC,是你的servlet-name配置的值,注意适时修改。
ServletContext sc=略
WebApplicationContext attr = (WebApplicationContext)sc.getAttribute("org.springframework.web.servlet.FrameworkServlet.CONTEXT.springMVC");



2、listener方式加载时:
【web.xml】
<context-param>
  <param-name>contextConfigLocation</param-name>
  <param-value>/WEB-INF/applicationContext</param-value>
</context-param>

<listener>
  <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
【jsp/servlet】
可已这样取得
ServletContext context = getServletContext();
WebApplicationContext applicationContext  = WebApplicationContextUtils .getWebApplicationContext(context);

5 楼 elf8848 2011-04-03  
http://zywang./blog/983801
Spring3.0中的异常处理配置方法

放在这里备份一下
4 楼 zywang 2011-03-24  
非常详细啊,很实用。
顺便请教个问题,Spring3拦截所有请求时,比如REST结构的那种,如果希望下载一个文件怎么处理,虽然直接用response手工编码控制输出文件可以实现下载,但是这这种做法不支持多线程下载,不知楼主有什么好的建议没有
3 楼 elf8848 2011-03-23  
作者补充说明:
(由于je的编辑器有bug,原文再保存格式就乱了,所以写在这里)

本例子中使用了一个spring配置文件叫“springMVC.xml”, 大家在应用时,关于数据源,事务等等相关的配置也放到这同一个文件中。

就不要再多配置一个applicationContext.xml文件了,
web.xml文件中也不要再配置org.springframework.web.context.ContextLoaderListener的listener了。
不然spring会扫描两次。

出现事务不好使的现象。
spring3 MVC,使用注解事务(@Transactional),不回滚,不起作用。就是因为,如果写两个配置文件, 先执行springMVC.xml(xxx-servlet.xml),后执行applicationContext.xml。
先执行springMVC.xml(xxx-servlet.xml)时,由于数据源、事务都配置在applicationContext.xml中,还没有执行, 所以导致了上面的问题。

2 楼 lijingzhi 2011-03-23  
赞一个,很到位了,还望继续
1 楼 cowwen 2011-03-15  
刚刚遇到了被spring拦截了js文件的问题,真的多谢楼主的总结,找了一天的问题,终于搞掂了。。。。

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约