日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > c/c++ >内容正文

c/c++

spring mvc @RequestBody @ResponseBody 解析流程

發布時間:2025/3/19 c/c++ 28 豆豆
生活随笔 收集整理的這篇文章主要介紹了 spring mvc @RequestBody @ResponseBody 解析流程 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

一、此接口測試用例

@RequestMapping(value = "/save_user",produces = "application/xml;charset=utf-8")@ResponseBodypublic Map<String,Object> saveUser(@RequestBody UserParam userParam) throws Exception{log.info(" get user userParam:" + JSONUtil.toJsonStr(userParam));Map<String,Object> user = new HashMap<String,Object>();user.put("id",userParam.getAge() );user.put("name",userParam.getName() );return user; // return "liming";}

dispatchServlet.xml配置

<mvc:annotation-driven><mvc:message-converters><bean class="com.tpw.component.AkResponseConverter"></bean><bean class="org.springframework.http.converter.StringHttpMessageConverter"/><bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"/><bean class="org.springframework.http.converter.xml.MappingJackson2XmlHttpMessageConverter"></bean></mvc:message-converters><mvc:argument-resolvers><bean class="com.tpw.component.RequestPropertiesArgumentResolver"></bean></mvc:argument-resolvers></mvc:annotation-driven>

一、初始化加載解析器和轉換器

請求參數由HandlerMethodArgumentResolver解析,返回參數由HandlerMethodReturnValueHandler來解析。這些解析器和返回值轉換器存儲在RequestMappingHandlerAdapter類中。

?1.系統初始化request的參數解析器,這個存儲在

RequestMappingHandlerAdapter類的 private HandlerMethodArgumentResolverComposite argumentResolvers;

? ? 2 初始化在RequestMappingHandlerAdapter.afterPropertiesSet方法中

public void afterPropertiesSet() {// Do this first, it may add ResponseBody advice beansinitControllerAdviceCache();if (this.argumentResolvers == null) {List<HandlerMethodArgumentResolver> resolvers = getDefaultArgumentResolvers();this.argumentResolvers = new HandlerMethodArgumentResolverComposite().addResolvers(resolvers);}if (this.initBinderArgumentResolvers == null) {List<HandlerMethodArgumentResolver> resolvers = getDefaultInitBinderArgumentResolvers();this.initBinderArgumentResolvers = new HandlerMethodArgumentResolverComposite().addResolvers(resolvers);}if (this.returnValueHandlers == null) {List<HandlerMethodReturnValueHandler> handlers = getDefaultReturnValueHandlers();this.returnValueHandlers = new HandlerMethodReturnValueHandlerComposite().addHandlers(handlers);}}

?

3 然后會加載26個系統默認的參數解析器

private List<HandlerMethodArgumentResolver> getDefaultArgumentResolvers() {List<HandlerMethodArgumentResolver> resolvers = new ArrayList<>(30);// Annotation-based argument resolutionresolvers.add(new RequestParamMethodArgumentResolver(getBeanFactory(), false));resolvers.add(new RequestParamMapMethodArgumentResolver());resolvers.add(new PathVariableMethodArgumentResolver());resolvers.add(new PathVariableMapMethodArgumentResolver());resolvers.add(new MatrixVariableMethodArgumentResolver());resolvers.add(new MatrixVariableMapMethodArgumentResolver());resolvers.add(new ServletModelAttributeMethodProcessor(false));resolvers.add(new RequestResponseBodyMethodProcessor(getMessageConverters(), this.requestResponseBodyAdvice));resolvers.add(new RequestPartMethodArgumentResolver(getMessageConverters(), this.requestResponseBodyAdvice));resolvers.add(new RequestHeaderMethodArgumentResolver(getBeanFactory()));resolvers.add(new RequestHeaderMapMethodArgumentResolver());resolvers.add(new ServletCookieValueMethodArgumentResolver(getBeanFactory()));resolvers.add(new ExpressionValueMethodArgumentResolver(getBeanFactory()));resolvers.add(new SessionAttributeMethodArgumentResolver());resolvers.add(new RequestAttributeMethodArgumentResolver());// Type-based argument resolutionresolvers.add(new ServletRequestMethodArgumentResolver());resolvers.add(new ServletResponseMethodArgumentResolver());resolvers.add(new HttpEntityMethodProcessor(getMessageConverters(), this.requestResponseBodyAdvice));resolvers.add(new RedirectAttributesMethodArgumentResolver());resolvers.add(new ModelMethodProcessor());resolvers.add(new MapMethodProcessor());resolvers.add(new ErrorsMethodArgumentResolver());resolvers.add(new SessionStatusMethodArgumentResolver());resolvers.add(new UriComponentsBuilderMethodArgumentResolver());// Custom argumentsif (getCustomArgumentResolvers() != null) {resolvers.addAll(getCustomArgumentResolvers());}// Catch-allresolvers.add(new RequestParamMethodArgumentResolver(getBeanFactory(), true));resolvers.add(new ServletModelAttributeMethodProcessor(true));return resolvers;}

4 這里還會加載自定義參數解析器.自定義參數解析器在dispatch-servlet.xml配置

<mvc:annotation-driven><mvc:message-converters><bean class="org.springframework.http.converter.StringHttpMessageConverter"/><bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"/></mvc:message-converters><mvc:argument-resolvers><bean class="com.tpw.component.RequestPropertiesArgumentResolver"></bean></mvc:argument-resolvers> </mvc:annotation-driven>

即可。

5.默認的返回值轉換器為

6.@RequestBody和@ResponseBody都由RequestResponseBodyMethodProcessor這個類負責轉換。

@Overridepublic boolean supportsParameter(MethodParameter parameter) {return parameter.hasParameterAnnotation(RequestBody.class);}@Overridepublic boolean supportsReturnType(MethodParameter returnType) {return (AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(), ResponseBody.class) ||returnType.hasMethodAnnotation(ResponseBody.class));}

7 如果是springboot項目,則寫一個CONFIGUATION類實現WebMvcConfigurer接口即可。

org.springframework.boot.autoconfigure.web.servlet.WebMvcAutoConfiguration的內部類會自動創建RequestMappingHandlerAdapter @Configuration(proxyBeanMethods = false) public static class EnableWebMvcConfiguration extends DelegatingWebMvcConfiguration implements ResourceLoaderAware { @Configuration(proxyBeanMethods = false)public static class EnableWebMvcConfiguration extends DelegatingWebMvcConfiguration implements ResourceLoaderAware {@Bean@Overridepublic RequestMappingHandlerAdapter requestMappingHandlerAdapter(@Qualifier("mvcContentNegotiationManager") ContentNegotiationManager contentNegotiationManager,@Qualifier("mvcConversionService") FormattingConversionService conversionService,@Qualifier("mvcValidator") Validator validator) {RequestMappingHandlerAdapter adapter = super.requestMappingHandlerAdapter(contentNegotiationManager,conversionService, validator);adapter.setIgnoreDefaultModelOnRedirect(this.mvcProperties == null || this.mvcProperties.isIgnoreDefaultModelOnRedirect());return adapter;}

然后會調用WebMvcConfigurationSupport.requestMappingHandlerAdapter

public RequestMappingHandlerAdapter requestMappingHandlerAdapter(@Qualifier("mvcContentNegotiationManager") ContentNegotiationManager contentNegotiationManager,@Qualifier("mvcConversionService") FormattingConversionService conversionService,@Qualifier("mvcValidator") Validator validator) {RequestMappingHandlerAdapter adapter = createRequestMappingHandlerAdapter();adapter.setContentNegotiationManager(contentNegotiationManager);adapter.setMessageConverters(getMessageConverters());adapter.setWebBindingInitializer(getConfigurableWebBindingInitializer(conversionService, validator));adapter.setCustomArgumentResolvers(getArgumentResolvers());adapter.setCustomReturnValueHandlers(getReturnValueHandlers());if (jackson2Present) {adapter.setRequestBodyAdvice(Collections.singletonList(new JsonViewRequestBodyAdvice()));adapter.setResponseBodyAdvice(Collections.singletonList(new JsonViewResponseBodyAdvice()));}AsyncSupportConfigurer configurer = new AsyncSupportConfigurer();configureAsyncSupport(configurer);if (configurer.getTaskExecutor() != null) {adapter.setTaskExecutor(configurer.getTaskExecutor());}if (configurer.getTimeout() != null) {adapter.setAsyncRequestTimeout(configurer.getTimeout());}adapter.setCallableInterceptors(configurer.getCallableInterceptors());adapter.setDeferredResultInterceptors(configurer.getDeferredResultInterceptors());return adapter;}

最終會調用WebMvcConfigurerComposite.addArgumentResolvers,這個類的List<WebMvcConfigurer> delegates 即為當前容器所有實現了WebMvcConfigurer接口的類。

WebMvcConfigurerComposite@Overridepublic void addArgumentResolvers(List<HandlerMethodArgumentResolver> argumentResolvers) {for (WebMvcConfigurer delegate : this.delegates) {delegate.addArgumentResolvers(argumentResolvers);}}

二、@RequestBody請求解析流程

??

??1.DispatcherServlet.doDispatch 中間會調用當前請求接口方法中所有支持的參數解析器,進行相應的參數解析和轉換。

InvocableHandlerMethod.getMethodArgumentValues protected Object[] getMethodArgumentValues(NativeWebRequest request, @Nullable ModelAndViewContainer mavContainer,Object... providedArgs) throws Exception {MethodParameter[] parameters = getMethodParameters();if (ObjectUtils.isEmpty(parameters)) {return EMPTY_ARGS;}Object[] args = new Object[parameters.length];for (int i = 0; i < parameters.length; i++) {MethodParameter parameter = parameters[i];parameter.initParameterNameDiscovery(this.parameterNameDiscoverer);args[i] = findProvidedArgument(parameter, providedArgs);if (args[i] != null) {continue;}if (!this.resolvers.supportsParameter(parameter)) {throw new IllegalStateException(formatArgumentError(parameter, "No suitable resolver"));}try {args[i] = this.resolvers.resolveArgument(parameter, mavContainer, request, this.dataBinderFactory);}}return args;}

?這里面的this.resolvers的參數賦值來源于RequestMappingHandlerAdapter.invokeHandlerMethod,這個方法創建InvocableHandlerMethod,并設置解析器參數

@Nullableprotected ModelAndView invokeHandlerMethod(HttpServletRequest request,HttpServletResponse response, HandlerMethod handlerMethod) throws Exception {ServletWebRequest webRequest = new ServletWebRequest(request, response);try {WebDataBinderFactory binderFactory = getDataBinderFactory(handlerMethod);ModelFactory modelFactory = getModelFactory(handlerMethod, binderFactory);ServletInvocableHandlerMethod invocableMethod = createInvocableHandlerMethod(handlerMethod);if (this.argumentResolvers != null) {invocableMethod.setHandlerMethodArgumentResolvers(this.argumentResolvers);}if (this.returnValueHandlers != null) {invocableMethod.setHandlerMethodReturnValueHandlers(this.returnValueHandlers);}invocableMethod.setDataBinderFactory(binderFactory);invocableMethod.setParameterNameDiscoverer(this.parameterNameDiscoverer);ModelAndViewContainer mavContainer = new ModelAndViewContainer();mavContainer.addAllAttributes(RequestContextUtils.getInputFlashMap(request));modelFactory.initModel(webRequest, mavContainer, invocableMethod);mavContainer.setIgnoreDefaultModelOnRedirect(this.ignoreDefaultModelOnRedirect);AsyncWebRequest asyncWebRequest = WebAsyncUtils.createAsyncWebRequest(request, response);asyncWebRequest.setTimeout(this.asyncRequestTimeout);WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);asyncManager.setTaskExecutor(this.taskExecutor);asyncManager.setAsyncWebRequest(asyncWebRequest);asyncManager.registerCallableInterceptors(this.callableInterceptors);asyncManager.registerDeferredResultInterceptors(this.deferredResultInterceptors);invocableMethod.invokeAndHandle(webRequest, mavContainer);if (asyncManager.isConcurrentHandlingStarted()) {return null;}return getModelAndView(mavContainer, modelFactory, webRequest);}finally {webRequest.requestCompleted();}}

2.然后會調用RequestResponseBodyMethodProcessor.resolveArgument接口進行參數解析。

public Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer,NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception {parameter = parameter.nestedIfOptional();Object arg = readWithMessageConverters(webRequest, parameter, parameter.getNestedGenericParameterType());String name = Conventions.getVariableNameForParameter(parameter);if (binderFactory != null) {WebDataBinder binder = binderFactory.createBinder(webRequest, arg, name);if (arg != null) {validateIfApplicable(binder, parameter);if (binder.getBindingResult().hasErrors() && isBindExceptionRequired(binder, parameter)) {throw new MethodArgumentNotValidException(parameter, binder.getBindingResult());}}if (mavContainer != null) {mavContainer.addAttribute(BindingResult.MODEL_KEY_PREFIX + name, binder.getBindingResult());}}return adaptArgumentIfNecessary(arg, parameter);}

?3.readWithMessageConverters(webRequest, parameter, parameter.getNestedGenericParameterType()) 最終會調用

AbstractMessageConverterMethodArgumentResolver.readWithMessageConverters進行參數解析。

此函數內會使用當前容器中所有的messageConvert類BEAN測試是否支持可讀,如果可讀的話,則進行messageConvert的讀轉換接口。

protected <T> Object readWithMessageConverters(HttpInputMessage inputMessage, MethodParameter parameter,Type targetType) throws IOException, HttpMediaTypeNotSupportedException, HttpMessageNotReadableException {MediaType contentType;boolean noContentType = false;Object body = NO_VALUE;EmptyBodyCheckingHttpInputMessage message;try {message = new EmptyBodyCheckingHttpInputMessage(inputMessage);for (HttpMessageConverter<?> converter : this.messageConverters) {Class<HttpMessageConverter<?>> converterType = (Class<HttpMessageConverter<?>>) converter.getClass();GenericHttpMessageConverter<?> genericConverter =(converter instanceof GenericHttpMessageConverter ? (GenericHttpMessageConverter<?>) converter : null);if (genericConverter != null ? genericConverter.canRead(targetType, contextClass, contentType) :(targetClass != null && converter.canRead(targetClass, contentType))) {if (message.hasBody()) {HttpInputMessage msgToUse =getAdvice().beforeBodyRead(message, parameter, targetType, converterType);body = (genericConverter != null ? genericConverter.read(targetType, contextClass, msgToUse) :((HttpMessageConverter<T>) converter).read(targetClass, msgToUse));body = getAdvice().afterBodyRead(body, msgToUse, parameter, targetType, converterType);}else {body = getAdvice().handleEmptyBody(null, message, parameter, targetType, converterType);}break;}}}return body;}

這里的messageConvert實際上是MappingJackson2HttpMessageConverter,他支持JSON轉換

public MappingJackson2HttpMessageConverter(ObjectMapper objectMapper) {super(objectMapper, MediaType.APPLICATION_JSON, new MediaType("application", "*+json")); }

?4.resolveArgument方法中的validateIfApplicable(binder, parameter)會對帶有@validate注解的參數進行校驗。

5.解析完生成參數實體對象。

三、@Responsebody解析流程

1.接口方法執行完后,會調用HandlerMethodReturnValueHandlerComposite.

handleReturnValue進行轉換。這里尋找系統中所有returnHandler,查找是否支持可寫的HANDLER,進行處理。 public void handleReturnValue(@Nullable Object returnValue, MethodParameter returnType,ModelAndViewContainer mavContainer, NativeWebRequest webRequest) throws Exception {HandlerMethodReturnValueHandler handler = selectHandler(returnValue, returnType);if (handler == null) {throw new IllegalArgumentException("Unknown return value type: " + returnType.getParameterType().getName());}handler.handleReturnValue(returnValue, returnType, mavContainer, webRequest);}

找HANDLER的原則是看是否支持。

RequestResponseBodyMethodProcessor public boolean supportsReturnType(MethodParameter returnType) {return (AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(), ResponseBody.class) ||returnType.hasMethodAnnotation(ResponseBody.class));}

?2.最終會調用到AbstractMessageConverterMethodProcessor.writeWithMessageConverters進行結果轉換,這里又會尋找當前系統中所有的messageConvert類,找到支持寫的類,進行數據轉換。然后返回。

protected <T> void writeWithMessageConverters(@Nullable T value, MethodParameter returnType,ServletServerHttpRequest inputMessage, ServletServerHttpResponse outputMessage)throws IOException, HttpMediaTypeNotAcceptableException, HttpMessageNotWritableException {Object body;Class<?> valueType;Type targetType;if (selectedMediaType != null) {selectedMediaType = selectedMediaType.removeQualityValue();for (HttpMessageConverter<?> converter : this.messageConverters) {GenericHttpMessageConverter genericConverter = (converter instanceof GenericHttpMessageConverter ?(GenericHttpMessageConverter<?>) converter : null);if (genericConverter != null ?((GenericHttpMessageConverter) converter).canWrite(targetType, valueType, selectedMediaType) :converter.canWrite(valueType, selectedMediaType)) {body = getAdvice().beforeBodyWrite(body, returnType, selectedMediaType,(Class<? extends HttpMessageConverter<?>>) converter.getClass(),inputMessage, outputMessage);if (body != null) {Object theBody = body;LogFormatUtils.traceDebug(logger, traceOn ->"Writing [" + LogFormatUtils.formatValue(theBody, !traceOn) + "]");addContentDispositionHeader(inputMessage, outputMessage);if (genericConverter != null) {genericConverter.write(body, targetType, selectedMediaType, outputMessage);}else {((HttpMessageConverter) converter).write(body, selectedMediaType, outputMessage);}}else {if (logger.isDebugEnabled()) {logger.debug("Nothing to write: null body");}}return;}}}}

這里找到的是MappingJackson2XmlHttpMessageConverter

?最終通過objMapper進行對象轉換。我這里配置的返回類型為XML.

?最終將XML的數據流寫入resonse.

總結

以上是生活随笔為你收集整理的spring mvc @RequestBody @ResponseBody 解析流程的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

主站蜘蛛池模板: 香蕉视频成人在线观看 | 国产成人av网 | 亚洲欧美另类综合 | 日韩欧美成| 性视频欧美 | 穿扒开跪着折磨屁股视频 | 又黄又骚又爽 | 寂寞少妇让水电工爽hd | 欧美一区二区激情视频 | 午夜专区 | 99精品久久久久久 | 免费激情 | 水多多在线 | 色老头一区二区三区在线观看 | 成人黄色动漫在线观看 | 免费成人高清 | 国产精品老熟女视频一区二区 | 国产曰肥老太婆无遮挡 | 久久精品久久久 | 黄色理论片 | 国产古装艳史毛片hd | 精品免费在线视频 | 欧美九九视频 | 国产精品1234| 福利片在线观看 | 久草视频中文在线 | 国产av人人夜夜澡人人爽麻豆 | av最新天| 国产99久久久国产精品 | 欧美黄色大片免费看 | 亚洲天堂国产 | 国产精品区一区二 | www.999热| 夜夜操影院| 精品人妻av在线 | xxx在线播放 | 狠狠天天| 国产av一区精品 | 男女裸体影院高潮 | 国产精品无码粉嫩小泬 | 国产精品刺激 | 三级国产三级在线 | 女人张开双腿让男人捅 | 日韩成人精品一区二区 | 精品欧美一区二区三区在线观看 | 午夜日韩在线 | 99精品自拍 | 一级特黄色大片 | 欧美 亚洲 另类 偷偷 自拍 | 欧美精品第一页 | 色妞综合 | 99久久精品日本一区二区免费 | 一级性生活免费视频 | 综合网在线视频 | 日韩亚洲欧美一区二区三区 | 96福利视频 | 福利视频导航网 | 污视频在线免费 | 在线观看日本网站 | 色哟哟国产精品 | 成人黄色av网址 | 日韩精品一区二区亚洲av性色 | 欧美日韩免费高清 | 国产偷亚洲偷欧美偷精品 | 午夜影院h| 青青草原成人 | 欧美亚洲第一区 | 青青草一区二区三区 | 久久久久久av无码免费网站 | 欧美在线视频观看 | 美女隐私无遮挡网站 | 男女爱爱网站 | 欧美日韩性 | 影音先锋国产精品 | aaaa黄色 | av不卡一区二区三区 | 亚洲bb | 国产一区自拍视频 | 无人在线观看的免费高清视频 | 欧美打屁股 | 在线a视频 | 国产顶级毛片 | 超清av| 久久久噜噜噜www成人网 | 好看的黄色网址 | 成人免费观看cn | 国产自偷自拍视频 | 天天操天天射天天 | 伊人精品 | 免费一级欧美片在线播放 | 中文字幕天堂网 | 日本高清免费观看 | 天天拍天天干 | 国产主播av在线 | 欧美日韩视频在线观看一区 | 中文字幕在线观看91 | 超碰95在线 | 日韩欧美国产综合 | 九色porny自拍视频在线播放 |