上一篇文章讲述了最简单的mvc:annotation-driven,此次就要说说@ResponseBody注解,很明显这个注解就是将方法的返回值做为reponse的body部分。咱们进一步分析下这个过程涉及到的内容,首先就是方法返回的类型,能够是字节数组、字符串、对象引用等,将这些返回类型以什么样的内容格式(即response的content-type类型,同时还要考虑到客户端是否接受这个类型)存进response的body中返回给客户端是一个问题,对于这个过程的处理都是靠许许多多的HttpMessageConverter转换器来完成的,这即是本章要讲的内容。
经常使用的content-type类型有:text/html、text/plain、text/xml、application/json、application/x-www-form-urlencoded、image/png等,不一样的类型,对body中的数据的解析也是不同的。
咱们的@ResponseBody能够指定content-type,打开ResponseBody注释,咱们能够看到这两个属性consumes和produces,它们就是用来指定request的content-type和response的content-type的。均可以接收一个或者多个,用法注释中已经给出了说明。
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- String[] consumes() default {};
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- String[] produces() default {};
当request的content-type不在consumes指定的范围内,则这个request就不会匹配到这个方法。produces 同时指定了方法的返回值将以什么样的content-type写入response的body中。若是这个属性进行了配置下文在获取服务器端指定的content-type就是所配置的值,不然则会获取默认的全部content-type
当咱们对@ResponseBody什么都没有配置时,SpringMVC便启用默认的策略帮咱们自动寻找一种最佳的方式将方法的返回值写入response的body中。
接下来,咱们就须要探究SpringMVC是如何处理这一过程的。先说说个人方式,就是调试,当方法执行完返回后,看DispatcherServlet的doDispatch方法的代码:
-
- HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
-
-
- String method = request.getMethod();
- boolean isGet = "GET".equals(method);
- if (isGet || "HEAD".equals(method)) {
- long lastModified = ha.getLastModified(request, mappedHandler.getHandler());
- if (logger.isDebugEnabled()) {
- logger.debug("Last-Modified value for [" + getRequestUri(request) + "] is: " + lastModified);
- }
- if (new ServletWebRequest(request, response).checkNotModified(lastModified) && isGet) {
- return;
- }
- }
-
- if (!mappedHandler.applyPreHandle(processedRequest, response)) {
- return;
- }
-
- try {
-
-
- mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
- }
mv = ha.handle(processedRequest, response, mappedHandler.getHandler());这里是适配器进行调度咱们的handler地方,因为咱们使用的是注解,因此对应的适配器是RequestMappingHandlerAdapter,经过一步步的函数调用,最终找到咱们的关注重点到RequestMappingHandlerAdapter的方法invokeHandleMethod,具体实现逻辑:
- private ModelAndView invokeHandleMethod(HttpServletRequest request,
- HttpServletResponse response, HandlerMethod handlerMethod) throws Exception {
-
- ServletWebRequest webRequest = new ServletWebRequest(request, response);
-
- WebDataBinderFactory binderFactory = getDataBinderFactory(handlerMethod);
- ModelFactory modelFactory = getModelFactory(handlerMethod, binderFactory);
- ServletInvocableHandlerMethod requestMappingMethod = createRequestMappingMethod(handlerMethod, binderFactory);
-
- ModelAndViewContainer mavContainer = new ModelAndViewContainer();
- mavContainer.addAllAttributes(RequestContextUtils.getInputFlashMap(request));
- modelFactory.initModel(webRequest, mavContainer, requestMappingMethod);
- mavContainer.setIgnoreDefaultModelOnRedirect(this.ignoreDefaultModelOnRedirect);
-
- AsyncWebRequest asyncWebRequest = WebAsyncUtils.createAsyncWebRequest(request, response);
- asyncWebRequest.setTimeout(this.asyncRequestTimeout);
-
- final WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);
- asyncManager.setTaskExecutor(this.taskExecutor);
- asyncManager.setAsyncWebRequest(asyncWebRequest);
- asyncManager.registerCallableInterceptors(this.callableInterceptors);
- asyncManager.registerDeferredResultInterceptors(this.deferredResultInterceptors);
-
- if (asyncManager.hasConcurrentResult()) {
- Object result = asyncManager.getConcurrentResult();
- mavContainer = (ModelAndViewContainer) asyncManager.getConcurrentResultContext()[0];
- asyncManager.clearConcurrentResult();
-
- if (logger.isDebugEnabled()) {
- logger.debug("Found concurrent result value [" + result + "]");
- }
- requestMappingMethod = requestMappingMethod.wrapConcurrentResult(result);
- }
-
-
- requestMappingMethod.invokeAndHandle(webRequest, mavContainer);
-
- if (asyncManager.isConcurrentHandlingStarted()) {
- return null;
- }
-
- return getModelAndView(mavContainer, modelFactory, webRequest);
- }
继续深刻看下这个requestMappingMethod.invokeAndHandle方法:
-
-
-
-
-
-
-
-
- public final void invokeAndHandle(ServletWebRequest webRequest,
- ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception {
-
- Object returnValue = invokeForRequest(webRequest, mavContainer, providedArgs);
-
- setResponseStatus(webRequest);
-
- if (returnValue == null) {
- if (isRequestNotModified(webRequest) || hasResponseStatus() || mavContainer.isRequestHandled()) {
- mavContainer.setRequestHandled(true);
- return;
- }
- }
- else if (StringUtils.hasText(this.responseReason)) {
- mavContainer.setRequestHandled(true);
- return;
- }
-
- mavContainer.setRequestHandled(false);
-
- try {
-
- this.returnValueHandlers.handleReturnValue(returnValue, getReturnValueType(returnValue), mavContainer, webRequest);
- }
- catch (Exception ex) {
- if (logger.isTraceEnabled()) {
- logger.trace(getReturnValueHandlingErrorMessage("Error handling return value", returnValue), ex);
- }
- throw ex;
- }
- }
mavContainer是ModelAndViewContainer类型,主要存储着model信息和view信息,它的一个属性requestHandled为true表示response直接处理不须要view的解决方案(便是须要返回一个视图的)。这里的mavContainer.setRequestHandled(false)只是初始时默认采用view的解决方案。
继续看this.returnValueHandlers.handleReturnValue具体内容:
-
-
-
-
- @Override
- public void handleReturnValue(
- Object returnValue, MethodParameter returnType,
- ModelAndViewContainer mavContainer, NativeWebRequest webRequest)
- throws Exception {
-
- HandlerMethodReturnValueHandler handler = getReturnValueHandler(returnType);
- Assert.notNull(handler, "Unknown return value type [" + returnType.getParameterType().getName() + "]");
- handler.handleReturnValue(returnValue, returnType, mavContainer, webRequest);
- }
继续看下它是如何找到合适的HandlerMethodReturnValueHandler的
-
-
-
- private HandlerMethodReturnValueHandler getReturnValueHandler(MethodParameter returnType) {
- for (HandlerMethodReturnValueHandler returnValueHandler : returnValueHandlers) {
- if (logger.isTraceEnabled()) {
- logger.trace("Testing if return value handler [" + returnValueHandler + "] supports [" +
- returnType.getGenericParameterType() + "]");
- }
- if (returnValueHandler.supportsReturnType(returnType)) {
- return returnValueHandler;
- }
- }
- return null;
- }
遍历全部的已注册的HandlerMethodReturnValueHandler,而后调用他们的supportsReturnType方法来判断他们各自是否支持这个返回值类型,经过调试发现会有13个HandlerMethodReturnValueHandler,以后再说这些数据是在何时哪一个地方注册的。列举下经常使用的:
ModelAndViewMethodReturnValueHandler:支持返回值是ModelAndView类型的
ModelMethodProcessor:支持返回值是Model的
ViewMethodReturnValueHandler:支持返回值是View
HttpEntityMethodProcessor:支持返回值是HttpEntity
RequestResponseBodyMethodProcess:支持类上或者方法上含有@ResponseBody注解的
ViewNameMethodReturnValueHandler:支持返回类型是void或者String
因此咱们想扩展的话,就能够自定实现一个HandlerMethodReturnValueHandler,而后在初始化时注册进去(这个过程后面再说)。言归正转,对于本工程RequestResponseBodyMethodProcess是支持的,因此它将被做为HandlerMethodReturnValueHandler返回,继续执行上面的handler.handleReturnValue(returnValue, returnType, mavContainer, webRequest)方法,咱们来看下RequestResponseBodyMethodProcess具体的处理过程:
- @Override
- public void handleReturnValue(Object returnValue, MethodParameter returnType,
- ModelAndViewContainer mavContainer, NativeWebRequest webRequest)
- throws IOException, HttpMediaTypeNotAcceptableException {
-
- mavContainer.setRequestHandled(true);
- if (returnValue != null) {
- writeWithMessageConverters(returnValue, returnType, webRequest);
- }
- }
方法writeWithMessageConverters的具体内容为:
- protected <T> void writeWithMessageConverters(T returnValue, MethodParameter returnType,
- ServletServerHttpRequest inputMessage, ServletServerHttpResponse outputMessage)
- throws IOException, HttpMediaTypeNotAcceptableException {
-
- Class<?> returnValueClass = returnValue.getClass();
- HttpServletRequest servletRequest = inputMessage.getServletRequest();
-
- List<MediaType> requestedMediaTypes = getAcceptableMediaTypes(servletRequest);
-
-
- List<MediaType> producibleMediaTypes = getProducibleMediaTypes(servletRequest, returnValueClass);
-
- Set<MediaType> compatibleMediaTypes = new LinkedHashSet<MediaType>();
- for (MediaType requestedType : requestedMediaTypes) {
- for (MediaType producibleType : producibleMediaTypes) {
- if (requestedType.isCompatibleWith(producibleType)) {
- compatibleMediaTypes.add(getMostSpecificMediaType(requestedType, producibleType));
- }
- }
- }
- if (compatibleMediaTypes.isEmpty()) {
- throw new HttpMediaTypeNotAcceptableException(producibleMediaTypes);
- }
-
- List<MediaType> mediaTypes = new ArrayList<MediaType>(compatibleMediaTypes);
- MediaType.sortBySpecificityAndQuality(mediaTypes);
-
- MediaType selectedMediaType = null;
- for (MediaType mediaType : mediaTypes) {
- if (mediaType.isConcrete()) {
- selectedMediaType = mediaType;
- break;
- }
- else if (mediaType.equals(MediaType.ALL) || mediaType.equals(MEDIA_TYPE_APPLICATION)) {
- selectedMediaType = MediaType.APPLICATION_OCTET_STREAM;
- break;
- }
- }
-
- if (selectedMediaType != null) {
- selectedMediaType = selectedMediaType.removeQualityValue();
- for (HttpMessageConverter<?> messageConverter :
-
-
- this.messageConverters) {
- if (messageConverter.canWrite(returnValueClass, selectedMediaType)) {
- ((HttpMessageConverter<T>) messageConverter).write(returnValue, selectedMediaType, outputMessage);
- if (logger.isDebugEnabled()) {
- logger.debug("Written [" + returnValue + "] as \"" + selectedMediaType + "\" using [" +
- messageConverter + "]");
- }
- return;
- }
- }
- }
- throw new HttpMediaTypeNotAcceptableException(this.allSupportedMediaTypes);
- }
获取客户端的content-type,只需解析Accept头字段便可,获取服务器端指定的content-type则分两种状况,第一种状况为:你在@RequestMapping中指定了produces的content-type类型(会将这一信息存进request的属性中,属性名为HandlerMapping接口名+'.producibleMediaTypes')若是没指定,则第二种状况:获取全部的已注册的messageConverter,获取它们全部的支持的content-type类型,而且过滤掉那些不支持returnValueClass的类型。而后在这两组List<MediaType> requestedMediaTypes和producibleMediaTypes中进行比较匹配(这里的比较规则也挺多的,涉及到q值,有兴趣大家能够总结下),选出一个最合适的content-type,至此有了返回值类型returnValueClass和要写进reponseBody的content-type类型,而后就是要找到一个支持这二者的HttpMessageConverter,已注册的HttpMessageConverter以下:
ByteArrayHttpMessageConverter:支持返回值类型为byte[],content-type为application/octet-stream,*/*
StringHttpMessageConverter:支持的返回值类型为String,content-type为 text/plain;charset=ISO-8859-1,*/*
ResourceHttpMessageConverter:支持的返回值类型为Resource,content-type为 */*
SourceHttpMessageConverter:支持的返回值类型为DomSource,SAXSource,Source,StreamSource,content-type为application/xml,text/xml,application/*+xml
MappingJacksonHttpMessageConverter:判断返回值可否被格式化成json,content-type为 application/json,application/*+json
AllEncompassingFormHttpMessageConverter:支持的返回值类型为MultiValueMap,content-type为application/x-www-form-urlencoded,multipart/form-data
对于咱们的工程来讲,返回类型为String,选出来的最合适的content-type是text/html,而且StringHttpMessageConverter的*/*是兼容任意类型的,因此StringHttpMessageConverter会被选中,而后将返回值以text/html形式写进response的body中。
顺便说下对于content-length是这样获取的:
首先从指定的content-type(本工程即text/html)中获取字符集,若能获取到则使用该字符集,若获取不到则使用默认的字符集,对于本工程来讲,text/html不像application/json;charset=utf-8那样含有字符集,因此将会使用StringHttpMessageConverter默认的字符集
- public static final Charset DEFAULT_CHARSET = Charset.forName("ISO-8859-1");
-
- private final Charset defaultCharset;
-
- private final List<Charset> availableCharsets;
-
- private boolean writeAcceptCharset = true;
-
-
-
-
-
-
- public StringHttpMessageConverter() {
- this(DEFAULT_CHARSET);
- }
-
-
-
-
-
- public StringHttpMessageConverter(Charset defaultCharset) {
- super(new MediaType("text", "plain", defaultCharset), MediaType.ALL);
- this.defaultCharset = defaultCharset;
- this.availableCharsets = new ArrayList<Charset>(Charset.availableCharsets().values());
- }
StringHttpMessageConverter有两个构造函数。当你没有给它指定字符集时,使用默认的ISO-8859-1,这即是形成乱码的一个缘由,因为咱们常用utf-8,因此能够在构造它时指定一下字符集。继续content-length的计算,有了字符集就好办了,计算方法为str.getBytes(字符集).length即是content-length的值,代码以下。
- if (headers.getContentLength() == -1) {
- Long contentLength = getContentLength(t, headers.getContentType());
- if (contentLength != null) {
- headers.setContentLength(contentLength);
- }
- }
和
- @Override
- protected Long getContentLength(String s, MediaType contentType) {
- Charset charset = getContentTypeCharset(contentType);
- try {
- return (long) s.getBytes(charset.name()).length;
- }
- catch (UnsupportedEncodingException ex) {
-
- throw new IllegalStateException(ex);
- }
- }
继续说StringHttpMessageConverter的写入过程,
- @Override
- protected void writeInternal(String s, HttpOutputMessage outputMessage) throws IOException {
- if (this.writeAcceptCharset) {
- outputMessage.getHeaders().setAcceptCharset(getAcceptedCharsets());
- }
- Charset charset = getContentTypeCharset(outputMessage.getHeaders().getContentType());
-
- StreamUtils.copy(s, charset, outputMessage.getBody());
- }
上面的charset就是ISO-8859-1,也就是将返回的字符串以ISO-8859-1编码写进response的body中。至此就完成了,而后就出现了ä¸å½这种乱码。
下一篇文章将会详细说说乱码,再下一篇文章还要继续本节遗留的不少问题,第一个HandlerMethodReturnValueHandler的来源及使用以及咱们来自定义一个HandlerMethodReturnValueHandler,第二个问题:其余HttpMessageConverter的使用以及自定义HttpMessageConverter。