Spring MVC数据绑定的扩展

开发 后端
Spring MVC经过三个版本,功能已经改进和完善了很多。尤其是2.5以来采用的Annotation的参数绑定,极大的方便了开发,3.0对其进行更进一步的完善。

Spring MVC经过三个版本,功能已经改进和完善了很多。尤其是2.5以来采用的Annotation的参数绑定,极大的方便了开发,3.0对其进行更进一步的完善。对于一些特殊的前台框架,传到后台的不是普通的request中的参数,而是request流中的xml格式,这时就不能采用SpringMVC自带的参数绑定方法。这时候考虑是否能扩展一下。

SpringMVC默认使用的是AnnotationMethodHandlerAdapter.java,可以修改这个类来实现扩展。关键位置在如下方法中:

  1. otected ModelAndView invokeHandlerMethod(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { 
  2.  
  3.      ServletHandlerMethodResolver methodResolver= getMethodResolver(handler); 
  4.  
  5.      Method handlerMethod = methodResolver.resolveHandlerMethod(request); 
  6.  
  7.      ServletHandlerMethodInvoker methodInvoker=new ServletHandlerMethodInvoker(methodResolver); 
  8.  
  9.      ServletWebRequest webRequest = new ServletWebRequest(request, response); 
  10.  
  11.      ExtendedModelMap implicitModel = new BindingAwareModelMap(); 
  12.  
  13.      Object result=methodInvoker.invokeHandlerMethod(handlerMethod,handler,webRequest, implicitModel); 
  14.  
  15.      ModelAndView mav=methodInvoker.getModelAndView(handlerMethod,handler.getClass(),result, implicitModel, webRequest);    methodInvoker.updateModelAttributes(handler,(mav!=null?mav.getModel():null),implicitModel,webRequest); 
  16.  
  17.      return mav; 
  18.     } 

蓝色位置是关键点,ServletHandlerMethodInvoker.java是内部类,继承自HandlerMethodInvoker.java,invokeHandlerMethod方法需要扩展,继续跟踪这个方法,发现是HandlerMethodInvoker.java这个类的方法,这个方法中的关键方法是resolveHandlerArguments(),关键部分如下:

  1. if (RequestParam.class.isInstance(paramAnn)) { 
  2. RequestParam requestParam = (RequestParam) paramAnn; 
  3. paramName = requestParam.value(); 
  4. required = requestParam.required(); 
  5. defaultValue = parseDefaultValueAttribute(requestParam.defaultValue()); 
  6. annotationsFound++; 
  7. else if (RequestHeader.class.isInstance(paramAnn)) { 
  8. RequestHeader requestHeader = (RequestHeader) paramAnn; 
  9. headerName = requestHeader.value(); 
  10. required = requestHeader.required(); 
  11. defaultValue = parseDefaultValueAttribute(requestHeader.defaultValue()); 
  12. annotationsFound++; 

到此扩展的话需要添加自己的类型,如RequestParamExt,添加在后面,模仿如下:

  1. else if (RequestParamExt.class.isInstance(paramAnn)) { 
  2. RequestParamExtrequestParam = (RequestParamExt) paramAnn; 
  3. paramName = requestParam.value(); 
  4. defaultValue = parseDefaultValueAttribute(requestParam.defaultValue()); 
  5. miType = requestParam.type(); 
  6. annotationsFound++; 

  1. else if (paramName != null) { 
  2. args[i] = resolveRequestParam(paramName, required, defaultValue, methodParam, webRequest, handler); 

这个方法上面添加扩展逻辑:

  1. if(!RequestParamExt.TYPE_NONE.equals(miType)){ 
  2. if(null == platformRequest){ 
  3. HttpServletRequest request = webRequest.getNativeRequest(HttpServletRequest.class); 
  4. platformRequest = new PlatformRequest((HttpServletRequest)request, "utf-8"); 
  5. platformRequest.receiveData(); 
  6. if(RequestParamExt.TYPE_PLATFORMREQUEST.equals(miType)){ 
  7.     args[i] = platformRequest; 
  8. else if(RequestParamExt.TYPE_STR.equals(miType)){ 
  9. args[i] = resolveRequestStrParamExt(platformRequest, methodParam); 
  10. }else
  11. args[i] = resolveRequestParamExt(miType,platformRequest,paramName, defaultValue, methodParam, webRequest, handler); 

两个resolveRequest*Ext方法如下:

  1. protected Object resolveRequestStrParamExt(PlatformRequest platformRequest, MethodParameter methodParam){ 
  2. VariableList inVl = platformRequest.getVariableList(); 
  3. String paraName = methodParam.getParameterName(); 
  4. return inVl.getValueAsObject(paraName); 
  5. protected Object resolveRequestParamExt(String miType,PlatformRequest platformRequest, String paramName,  
  6. String defaultValue,MethodParameter methodParam,NativeWebRequest webRequest, Object handler)throws Exception{ 
  7. if(StringUtils.isBlank(paramName)){ 
  8. paramName = defaultValue; 
  9. Class<?> paramType = methodParam.getParameterType(); 
  10. DatasetList inDl = platformRequest.getDatasetList(); 
  11. VariableList inVl = platformRequest.getVariableList(); 
  12. if(RequestParamExt.TYPE_DS.equals(miType)){//绑定的关键过程 
  13. Dataset ds = inDl.getDataset(paramName); 
  14. Object vo = paramType.newInstance(); 
  15. MiPDataBinder dataBinder = new MiPDataBinder(vo, false); 
  16.     dataBinder.bind(inVl); 
  17.     return dataBinder.getTarget(); 

同时还需要一个annotation的定义:示例如下:

  1. package com.company.springext.web.bind.annotation; 
  2. import java.lang.annotation.Documented; 
  3. import java.lang.annotation.ElementType; 
  4. import java.lang.annotation.Retention; 
  5. import java.lang.annotation.RetentionPolicy; 
  6. import java.lang.annotation.Target; 
  7. @Target(ElementType.PARAMETER) 
  8. @Retention(RetentionPolicy.RUNTIME) 
  9. @Documented 
  10. public @interface RequestParamExt { 
  11.     public static final String TYPE_NONE            = "none"
  12.     public static final String TYPE_DS              = "ds"
  13.     public static final String TYPE_VL              = "vl"
  14.     public static final String TYPE_STR             = "string";    
  15. String type() default TYPE_NONE;  
  16. String value() default "";     
  17. String defaultValue() default "ds"

***是修改Spring配置:

  1. <bean class="com.company.springext.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapterExt">     
  2. </bean>   

到此就实现了自定义格式的数据绑定。

对于特定格式的输出,如果需要自定义的话,同样需要修改AnnotationMethodHandlerAdapterExt.java这个类,关键位置在getModelAndView()方法。在如下位置:

  1. else if (AnnotationUtils.findAnnotation(handlerMethod, ResponseBody.class) != null) { 
  2.             handleResponseBody(returnValue, webRequest); 
  3.             return null
  4.         } 

添加自己的扩展方法:

  1. else if (AnnotationUtils.findAnnotation(handlerMethod, ResponseBodyExt.class) != null) { 
  2.                 ResponseBodyExt bodyMi = AnnotationUtils.findAnnotation(handlerMethod, ResponseBodyExt.class); 
  3.                 handleResponseBodyExt(returnValue, webRequest, bodyMi); 
  4.                 return null
  5.             } 

定义handleResponseBodyExt方法:

  1. private void handleResponseBodyExt(Object returnValue, ServletWebRequest webRequest, ResponseBodyMI bodyMi) throws Exception { 
  2.            HttpServletResponse servletResponse = (HttpServletResponse) webRequest.getNativeResponse(); 
  3.            writeWithExtConverters(returnValue, servletResponse, bodyMi); 
  4.        } 

writeWithExtConverters()方法如下:

  1. private void writeWithExtConverters(Object returnValue, HttpServletResponse response, ResponseBodyMI bodyMi) throws Exception {             
  2.     convertToXML(...);     
  3.    }; 

使用方式如下:

  1. @RequestMapping(value="/getContractList"
  2. @ResponseBodyExt(isCheck=true, resultType="sql", sqlColumns="ID,TUREID"
  3.  
  4. public Page<Contract> getContractList(@RequestParamExt(value = "ds_search", type = "ds") Contract cp) throws Exception { 
  5. Page<Contract> page = method1(); 
  6. return page; 

原文链接:http://www.blogjava.net/ghostzhang/archive/2011/09/12/358486.html

【编辑推荐】

  1. Spring事务配置的五种方式
  2. Spring声明性事务常见问题分析
  3. 深入浅出Java三大框架SSH与MVC的设计模式
  4. Spring MVC拦截器实现分析
  5. 用Spring让Java Mail支持简化邮件发送
责任编辑:林师授 来源: GhostZhang的博客
相关推荐

2021-11-02 19:14:58

Spring数据

2024-02-01 08:28:28

2012-06-17 20:19:29

2021-09-01 10:37:25

鸿蒙HarmonyOS应用

2009-06-19 11:43:59

Spring MVC框

2022-05-30 09:32:07

Spring容器

2023-09-04 11:52:53

SpringMVC性能

2009-06-19 11:28:45

2011-08-29 09:48:30

springMVC

2009-06-24 16:01:28

Spring MVC

2021-10-30 18:56:12

Spring工作框架

2023-11-02 18:01:24

SpringMVC配置

2009-06-11 10:37:58

netbeans spMVC基础

2009-06-22 11:54:28

Spring MVCSpringframe

2021-09-01 14:36:14

鸿蒙HarmonyOS应用

2023-09-28 08:49:41

springBean

2021-12-30 23:37:51

SpringMVC RequestResponse

2011-05-24 09:22:44

Spring3异常处理

2023-07-10 08:00:13

架构Rest返回值

2011-12-08 09:01:59

JavaSpring
点赞
收藏

51CTO技术栈公众号