日韩性视频-久久久蜜桃-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 解析流程的全部內容,希望文章能夠幫你解決所遇到的問題。

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

主站蜘蛛池模板: www.婷婷| 动漫美女舌吻 | 免费在线观看黄色网址 | 久久久久久久网 | 国产精品jizz在线观看软件 | 成人h动漫精品一区二区器材 | 久久尹人 | 久久性生活 | 中文一二区 | 中国美女黄色 | 人人爱国产 | 欧美精品黑人 | 99re久久精品国产 | 亚洲成人第一区 | 婷婷激情五月综合 | 精品欧美久久 | 久久久久久久久久久久久久久久久 | 欧美日韩性 | 综合久 | 国产亚洲精品成人 | 91黄色免费看 | 日本亚洲天堂 | 大尺度做爰呻吟62集 | 一对一色视频聊天a | 亚洲欧美在线一区二区 | 欧美一区二区三区激情视频 | 黑森林av导航 | 尤物视频网站在线观看 | 国产a级淫片 | 97国产精品人人爽人人做 | 四虎影院成人 | 欧美色综合网 | 在线观看中文字幕视频 | 亚洲国产麻豆 | 性欧美丰满熟妇xxxx性久久久 | 夜夜操夜夜操 | 免费av一级片 | 五月婷婷深深爱 | 日本女优一区 | 欧美成人一区二区 | 欧美二级片 | 中文字幕在线播放一区二区 | h视频亚洲| 91精品国产91久久久久久 | 国产精品夜夜夜爽张柏芝 | 久久一区二区精品 | 91视频一区 | 波多野结衣小视频 | 国产亚洲欧美视频 | 久久爱一区 | 国产福利视频一区二区 | 亚洲一区二区三区在线 | 久草综合在线视频 | 亲嘴扒胸摸屁股免费视频日本网站 | 黄色av网站免费在线观看 | 欧美精品日韩在线观看 | 色婷婷av久久久久久久 | 日韩精品视频在线播放 | 亚洲九九爱 | 玖玖在线精品 | 懂色av一区二区三区免费观看 | 一级黄色伦理片 | 亚洲视频色图 | 亚洲综合小说 | 中文字幕在线色 | 99精品欧美一区二区 | 最新日韩av在线 | 亚洲最大的成人网 | 成人网页在线观看 | 亚洲网址 | 四虎一区二区 | 国产中文视频 | 影音先锋啪啪 | 午夜在线视频观看 | 免费在线看a | 夜夜撸 | 精品一区在线 | av色网站 | 欧美专区视频 | 日日操网| 欧美成人一区二区视频 | 影音先锋蜜桃 | 欧美做爰xxxⅹ性欧美大片 | 丁香激情网 | 精品熟妇无码av免费久久 | 神马午夜888| 99re在线| 狠狠操综合网 | 日本久久一级片 | 和漂亮岳做爰3中文字幕 | 国产浪潮av | 蜜桃色av| 红桃视频亚洲 | 成人午夜免费观看 | 乱色精品无码一区二区国产盗 | 国产精品无码av在线有声小说 | 天天网综合 | 中文字幕在线视频精品 | 亚色av |