十一、如何实现全局的异常处理? 在spring MVC的配置文件中:
<!-- 总错误处理--> <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页面:
<%@ 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的测试类的父类
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了。
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 。
十四、转发与重定向 可以通过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注解
/** * 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; }
评论
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文件的问题,真的多谢楼主的总结,找了一天的问题,终于搞掂了。。。。 |
|
来自: BlazerOfIT > 《spring》