首页 > 其他分享 >Could not extract response: no suitable `HttpMessageConverter` found for response type [class wechat

Could not extract response: no suitable `HttpMessageConverter` found for response type [class wechat

时间:2023-08-04 23:33:15浏览次数:44  
标签:HttpMessageConverter contentType MediaType canRead type response

1. 问题复现

话不多说,先贴出问题代码:这里的GetUserInfoByAccessToken是我自定义的一个实体类。

GetUserInfoByAccessToken getUserInfoByAccessTokenString = restTemplate.getForObject(userInfoByAccessCodeURL, GetUserInfoByAccessToken.class);

异常信息:Could not extract response: no suitable HttpMessageConverter found for response type [class wechat.wxRes.GetUserInfoByAccessToken] and content type [text/plain],很明显这段异常的意思是在指定返回类型为GetUserInfoByAccessToken,并且服务端响应报文的content-type为text/plain的情况下找不到一个合适的HttpMessageConverter 来处理这种情况

2. 处理方法

这里举例两种处理请求

1.首先StringHttpMessageConverter这个处理器是可以处理content-type为text/plain的响应报文的。但阅读源码知道必须放回类型是String才可以使用它,所有我们需要改写下代码,将放回类型改为String。需要的时候可以利用JSON工具类将其转为你需要的类型。

GetUserInfoByAccessToken getUserInfoByAccessTokenString = restTemplate.getForObject(userInfoByAccessCodeURL, String.class);

需要注意的是使用StringHttpMessageConverter容易出现中文乱码的情况,因为它默认支持的字符集是ISO-8859-1,这种时候可以参考以下代码更改StringHttpMessageConverter的默认字符集,我这里将其改为utf-8了。

RestTemplate customRestTemplate = new RestTemplate();
List<HttpMessageConverter<?>> list = customRestTemplate.getMessageConverters();
for (HttpMessageConverter<?> httpMessageConverter : list) {
    if(httpMessageConverter instanceof StringHttpMessageConverter) {
       ((StringHttpMessageConverter) httpMessageConverter).setDefaultCharset(Charset.forName("utf-8"));
                break;
      }
 }

2.往restTemplate的转换器里再加一个支持JSON转换的转换器,比如MappingJackson2HttpMessageConverter

RestTemplate customRestTemplate = new RestTemplate();
MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter = new MappingJackson2HttpMessageConverter();
mappingJackson2HttpMessageConverter.setSupportedMediaTypes(Arrays.asList(MediaType.TEXT_HTML,MediaType.TEXT_PLAIN));
restTemplate.getMessageConverters().add(mappingJackson2HttpMessageConverter);
GetUserInfoByAccessToken getUserInfoByAccessTokenString = restTemplate.getForObject(userInfoByAccessCodeURL, GetUserInfoByAccessToken.class);

3. 源码分析问题

3.1 关键代码extractData方法

extractData方法将接口请求拿到的响应报文拿来给HttpMessageConverter解析,这里会找到合适的解析器来解析响应报文,解析成我们指定的返回类型的数据,如果找不到或者处理出现异常就会抛出异常。

代码清单1-1 org.springframework.web.client.HttpMessageConverterExtractor#extractData
// 这里的入参是请求之后的响应体
public T extractData(ClientHttpResponse response) throws IOException {
    //创建一个名为responseWrapper的MessageBodyClientHttpResponseWrapper,用于包装响应对象response,方便操作响应数据。
	MessageBodyClientHttpResponseWrapper responseWrapper = new MessageBodyClientHttpResponseWrapper(response);
    // 检查响应是否有消息体,并且消息体不为空。如果不满足条件,则返回null。
	if (!responseWrapper.hasMessageBody() || responseWrapper.hasEmptyMessageBody()) {
		return null;
	}
    // 获取响应内容类型contentType。
	MediaType contentType = getContentType(responseWrapper);

	try {
        // 遍历已注册的HttpMessageConverter列表。
		for (HttpMessageConverter<?> messageConverter : this.messageConverters) {
            // 对于实现了GenericHttpMessageConverter接口的转换器,检查是否可以读取responseType对应的类型,并且内容类型匹配。
			if (messageConverter instanceof GenericHttpMessageConverter) {
				GenericHttpMessageConverter<?> genericMessageConverter =
						(GenericHttpMessageConverter<?>) messageConverter;
				if (genericMessageConverter.canRead(this.responseType, null, contentType)) {
					if (logger.isDebugEnabled()) {
						ResolvableType resolvableType = ResolvableType.forType(this.responseType);
						logger.debug("Reading to [" + resolvableType + "]");
					}
					return (T) genericMessageConverter.read(this.responseType, null, responseWrapper);
				}
			}
            // 如果没有找到合适的GenericHttpMessageConverter,则检查是否指定了responseClass。
			if (this.responseClass != null) {
                // 如果指定了responseClass,则检查是否有转换器可以读取该类型,并且内容类型匹配。见相关代码`canRead`方法中的代码清单1-2
				if (messageConverter.canRead(this.responseClass, contentType)) {
					if (logger.isDebugEnabled()) {
						String className = this.responseClass.getName();
						logger.debug("Reading to [" + className + "] as \"" + contentType + "\"");
					}
                    // 如果匹配成功,使用该转换器读取响应数据,并返回结果。
					return (T) messageConverter.read((Class) this.responseClass, responseWrapper);
				}
			}
		}
	}
	catch (IOException | HttpMessageNotReadableException ex) {
		throw new RestClientException("Error while extracting response for type [" +
				this.responseType + "] and content type [" + contentType + "]", ex);
	}

	throw new UnknownContentTypeException(this.responseType, contentType,
			response.getRawStatusCode(), response.getStatusText(), response.getHeaders(),
			getResponseBody(response));
}

3.2 相关代码messageConverter.canRead(this.responseClass, contentType)方法

canRead(java.lang.Class, org.springframework.http.MediaType)方法判断当前的HttpMessageConverter是否可以读取响应报文ContentType为服务端指定的数据,并且内容和你指定的返回值类型匹配。

代码清单1-2 org.springframework.http.converter.AbstractHttpMessageConverter#canRead(java.lang.Class, org.springframework.http.MediaType)
// 判断`HttpMessageConverter`转换器是否可以读取该ContentType的数据,并且内容和你指定的返回值类型匹配
public boolean canRead(Class<?> clazz, @Nullable MediaType mediaType) {
    // supports判断HttpMessageConverter转换器是否支持你指定的返回类型,参考代码清单1-3。canRead
	return supports(clazz) && canRead(mediaType);
}

这是StringHttpMessageConverter的supports方法,可以看出他可以处理返回类型为String的数据。

代码清单1-3 org.springframework.http.converter.StringHttpMessageConverter#supports
public boolean supports(Class<?> clazz) {
	return String.class == clazz;
}

上面代码supports方法返回true会调用canRead(org.springframework.http.MediaType)方法,这段代码主要就是判断当前的HttpMessageConverter 是否可以处理content-type为服务端指定类型的响应报文,比如content-type为text/plain。

代码清单1-4 org.springframework.http.converter.AbstractHttpMessageConverter#canRead(org.springframework.http.MediaType)
protected boolean canRead(@Nullable MediaType mediaType) {
	if (mediaType == null) {
		return true;
	}
	for (MediaType supportedMediaType : getSupportedMediaTypes()) {
		if (supportedMediaType.includes(mediaType)) {
			return true;
		}
	}
	return false;
}

4.关键点截图

以下是我在调试中截取的一些图片。

这里可以看到响应体的contentType为text/plain,接下来就要找可以处理这种响应类型的HttpMessageConverter

这里可以看到已注册的HttpMessageConverter列表里面有九个元素,并且通过他们的supportedMediaTypes属性看到他们可以处理的contentType

首先判断HttpMessageConverter是否可以读取我们指定的返回类,这里我指定的是我自定义的一个返回类GetUserInfoByAccessToken.class

在这里是在判断当前HttpMessageConverter是否可以处理当前响content-type为text/plain的响应报文。

标签:HttpMessageConverter,contentType,MediaType,canRead,type,response
From: https://www.cnblogs.com/yuyiming/p/17607286.html

相关文章

  • [Typescript] Partial autocompletion (string & {})
    constpresetSizes={xs:"0.5rem",sm:"1rem",};typeSize=keyoftypeofpresetSizes;//typeLooseSize=Size|string;//theresultwillbestringtypeLooseSize=Size|(string&{});//workingexportconstIcon=(pr......
  • 使用 Spring 3 MVC HttpMessageConverter 功能构建 RESTful web 服务(转)
    Spring,构建Java™平台和EnterpriseEdition(JavaEE)应用程序的著名框架,现在在其模型-视图-控制器(Model-View-Controller,MVC)层支持具象状态传输(REST)。RESTfulweb服务根据客户端请求生成多个具象(representations)很重要。在本篇文章中,学习使用HttpMessageConverter 生成......
  • Spring源码分析(五) MappingJackson2HttpMessageConverter
    大家用过springmvc的肯定都用过@RequestBody和@ResponseBody注解吧,你了解这个的原理吗?这篇文章我们就来说下它是怎么实现json转换的。首先来看一个类RequestResponseBodyMethodProcessor,这个类继承了AbstractMessageConverterMethodProcessor,我们来看看这个类的构造方法protec......
  • delegate open and send for XMLHttpRequest by rewrite the prototype
     varsendProxied=window.XMLHttpRequest.prototype.send;window.XMLHttpRequest.prototype.send=function(){varobject={};letdata=arguments[0]if(data&&data.forEach){data.forEach((value,key)=>obj......
  • Typecho建站:腾讯云轻量应用服务器搭建博客网站教程
    腾讯云轻量应用服务器自带Typecho应用模板镜像,腾讯云提供的Typecho模板镜像是基于CentOS7.664位操作系统,并已预置Nginx、PHP、MariaDB软件程序,使用Typecho应用模板可以快速搭建博客、企业官网、电商及论坛等各类网站。腾讯云服务器网分享使用腾讯云轻量应用服务器Typecho应用模板......
  • Typescript使用基础篇
    前言对于typescript而言如果想要能在项目中熟练的应用,个人认为需要详细去了解以下几点内容:类型(基本类型,联合类型,类型断言,泛型),模块和命名空间,接口和类基础篇开发环境编译包全局安装npminstall-gtypescript编译ts文件命令tsc+ts文件监听文件变化并编译tsc-w+ts文件基础类型ty......
  • Golang反射type和kind有什么区
    一、前言Go语言中的反射是由reflect包提供支持的,它定义了两个重要的类型Type和Value。任意值在反射中都可以理解为由reflect.Type和reflect.Value两部分组成,并且reflect包提供了reflect.TypeOf和reflect.ValueOf两个函数来获取任意对象的Value和Type。在Go语......
  • 2.auto、decltype和decltype(auto)的用法
    2.auto、decltype和decltype(auto)的用法1.auto编程时常常需要把表达式的值赋给变量,这就要求声明变量时清楚的知道表达式的类型。然而有些情况是声明的变量的类型我们并不知道,比如在模板编程时。为了解决这个问题,C++11引入了auto类型说明符,用它来让编译器替我们去分析表达式所属......
  • 关于vue中同时使用v-if和nth-of-type时的bug
    问题引出需求:例如我想要在某一个ul元素中使用v-if条件时渲染2个li元素,同时第一个和第二个的样式不同,这里我使用了nth-of-type选择器去设置样式,但是当v-if条件改变时,li元素的样式没有改变,看了下开发者工具,元素对应的选择器没有改变,即使此时仅剩第二个li元素,它仍旧被nth-of-ty......
  • 论文解读(APCA)《Adaptive prototype and consistency alignment for semi-supervised d
    [Wechat:Y466551|付费咨询,非诚勿扰]论文信息论文标题:Adaptiveprototypeandconsistencyalignmentforsemi-superviseddomainadaptation论文作者:JihongOuyang、ZhengjieZhang、QingyiMeng论文来源:2023aRxiv论文地址:download 论文代码:download视屏讲解:click1介绍......