开发者社区> 游客tqcqp5pa4cdy4> 正文
阿里云
为了无法计算的价值
打开APP
阿里云APP内打开

发送http请求(2):RestTemplate发送http请求

简介: 发送http请求(2):RestTemplate发送http请求
+关注继续查看

image.png


关联阅读:


发送http请求(1):发送http请求的几种方式


相关组件


RestTemplate 是Spring提供的用于访问Rest服务的客户端工具,它提供了多种便捷访问远程Http服务的方法,能够大大提高客户端的编写效率。 RestTemplate 相比于他们应该算是一个整合框架,一个可以统一各种请求发送方式的框架。

先分析下其相关组件

继承体系

RestTemplate 继承了InterceptingHttpAccessor 实现了RestOperations

public class RestTemplate extends InterceptingHttpAccessor implements RestOperations {
}
RestOperations

定义rest的各种操作, 使用了大量的重载方法。例如,我们常用的

getForObject
postForObject
HttpAccessor

HttpAccessor 提供了ClientHttpRequestFactory属性,用于创建ClientHttpRequest。默认初始化SimpleClientHttpRequestFactory工厂类,

public abstract class HttpAccessor {
private ClientHttpRequestFactory requestFactory = new SimpleClientHttpRequestFactory();
...
}

发送请求的几种方式上节中我们说过SimpleClientHttpRequestFactory 是针对JDK原生HttpURLConnection工厂类。也就是说RestTemplate 默认是使用HttpURLConnection来发送http请求。

不过我们可以通过set方法设置其他工厂类。换用其他发送http请求的方式。

public void setRequestFactory(ClientHttpRequestFactory requestFactory) {
        super.setRequestFactory(requestFactory);
        this.interceptingRequestFactory = null;
}
InterceptingHttpAccessor

InterceptingHttpAccessor 继承了HttpAccessor。 从其名称,我们也可以看出这是一个具有拦截器功能的HttpAccessor

public abstract class InterceptingHttpAccessor extends HttpAccessor {
    private List<ClientHttpRequestInterceptor> interceptors = new ArrayList<ClientHttpRequestInterceptor>();
    public void setInterceptors(List<ClientHttpRequestInterceptor> interceptors) {
        this.interceptors = interceptors;
    }
    public List<ClientHttpRequestInterceptor> getInterceptors() {
        return interceptors;
    }
    @Override
    public ClientHttpRequestFactory getRequestFactory() {
        ClientHttpRequestFactory delegate = super.getRequestFactory();
        if (!CollectionUtils.isEmpty(getInterceptors())) {
            return new InterceptingClientHttpRequestFactory(delegate, getInterceptors());
        }
        else {
            return delegate;
        }
    }
}

getRequestFactory()方法,通过判断是否具有拦截器,决定是创建具有拦截功能的InterceptingClientHttpRequestFactory ,还是创建默认的SimpleClientHttpRequestFactory

InterceptingClientHttpRequestFactory 创建的ClientHttpRequest 是InterceptingClientHttpRequest

小结:

  • HttpAccessor: 提供对普通Http客户端 ClientHttpRequest的支持
  • InterceptingHttpAccessor: 提供对具有拦截功能的ClientHttpRequest的支持


属性体系

RestTemplate 属性也有几个关键组件,我们应该了解。

RequestCallback

Callback interface for code that operates on a ClientHttpRequest. Allows to manipulate the request headers, and write to the request body. 用于操作请求头和body,在请求发出前执行。

ResponseExtractor

解析HTTP响应的数据,从Response中提取数据,通过它来从ClientHttpResponse提取出指定内容(比如请求头、请求Body体等)

UriTemplateHandler

用来处理:Path Param 与 Query Param

ResponseErrorHandler

错误响应处理器


关联组件

InterceptingRequestExecution

当我们给Resttenplate设置拦截器时,默认会创建一个InterceptingClientHttpRequest客户端。 当我们调用InterceptingClientHttpRequest.execute()发送请求时,最终会调用InterceptingClientHttpRequest.executeInternal()方法

@Override
    protected final ClientHttpResponse executeInternal(HttpHeaders headers, byte[] bufferedOutput) throws IOException {
    //根据拦截器创建一个拦截器链。
        InterceptingRequestExecution requestExecution = new InterceptingRequestExecution();
        return requestExecution.execute(this, bufferedOutput);
    }

此时创建一个拦截器执行链,并把当前(this=InterceptingClientHttpRequest) 作为参数传入到执行链中。

ClientHttpRequestInterceptor

我们看看ClientHttpRequestInterceptor 是如何执行的。 InterceptingRequestExecution#execute() 会首先执行拦截器。

InterceptingRequestExecution类
@Override
public ClientHttpResponse execute(HttpRequest request, byte[] body)  {
    if (this.iterator.hasNext()) {
        ClientHttpRequestInterceptor nextInterceptor = this.iterator.next();
        return nextInterceptor.intercept(request, body, this);
    }       
    ....    
}

拦截器执行链中,获取一个拦截器,执行其intercept方法。在每个intercept方法中都调用execution.execute() 从而形成链式调用。


请求过程


讲完了组件,下面讲讲执行过程 以getForObject为例

@Override
    public <T> T getForObject(String url, Class<T> responseType, Object... urlVariables) throws RestClientException {
        RequestCallback requestCallback = acceptHeaderRequestCallback(responseType);
        HttpMessageConverterExtractor<T> responseExtractor =
                new HttpMessageConverterExtractor<T>(responseType, getMessageConverters(), logger);
        return execute(url, HttpMethod.GET, requestCallback, responseExtractor, urlVariables);
    }
    public <T> T execute(String url, HttpMethod method, RequestCallback requestCallback,
            ResponseExtractor<T> responseExtractor, Object... urlVariables) throws RestClientException {
        URI expanded = getUriTemplateHandler().expand(url, urlVariables);
        return doExecute(expanded, method, requestCallback, responseExtractor);
    }

调用doExecute方法

protected <T> T doExecute(URI url, HttpMethod method, RequestCallback requestCallback,
            ResponseExtractor<T> responseExtractor) throws RestClientException {
        ClientHttpResponse response = null;
        try {
        //创建一个http请求客户端,根据
            ClientHttpRequest request = createRequest(url, method);
            if (requestCallback != null) {
            //如果requestCallback不为null。则处理request
                requestCallback.doWithRequest(request);
            }
            //执行请求
            response = request.execute();
            //处理请求,包含错误响应的处理
            handleResponse(url, method, response);
            //提前指定数据
            if (responseExtractor != null) {
                return responseExtractor.extractData(response);
            }
            else {
                return null;
            }
        }
        ... 
        finally {
            if (response != null) {
                response.close();
            }
        }
    }
  1. 首先获取到RequestCallback ,ResponseExtractor 工具。并调用UriTemplateHandler处理url中的参数问题
  2. 从HttpAccessor获取一个ClientHttpRequest,如果配置了ClientHttpRequestInterceptor,则通过InterceptingClientHttpRequestFactory创建具有拦截器功能的InterceptingClientHttpRequest客户端。如果没有使用普通的ClientHttpRequestFactory创建一个普通客户端,默认使用的是SimpleClientHttpRequestFactory 创建HttpURLConnection 的客户端
  3. 得到ClientHttpRequest后,如果RequestCallback不为null。则执行RequestCallback
  4. 执行ClientHttpRequest.execute(). 如果此时有ClientHttpRequestInterceptor,则先执行ClientHttpRequestInterceptor,再发送请求。
  5. ResponseErrorHandler 处理可能的错误响应
  6. 如果存在responseExtractor响应提取器, 则调用responseExtractor 提取response数据。
  7. 数据返回


总结


Resttemplate 应该说是对发送请求的方式的一种抽象,他不生产请求,只做请求的整合工。这种高级整合带来的是更加便捷的,更加丰富的发送请求方式。


版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
前端构造 HTTP 请求的四种方法
前端构造 HTTP 请求的四种方法
0 0
商库圈_Http_请求方式 | 学习笔记
快速学习商库圈_Http_请求方式。
0 0
1、Http协议(概述、请求方法、状态码)
1、Http协议(概述、请求方法、状态码)
0 0
Http实战之编码、分块传输、范围请求
Http实战之编码、分块传输、范围请求
0 0
使用tcpdump查看HTTP请求响应 详细信息 数据
使用tcpdump查看HTTP请求响应 详细信息 数据
0 0
Nginx中安装免费SSL证书开启Https请求(下)
在部分场景中,我们必须使用 https 请求,因为 https 请求更为安全,常用于支付方面的请求调用
0 0
Nginx中安装免费SSL证书开启Https请求(上)
在部分场景中,我们必须使用 https 请求,因为 https 请求更为安全,常用于支付方面的请求调用
0 0
http详解笔记学习9-socket编程怎么发送get请求-1
http详解笔记学习9-socket编程怎么发送get请求-1
0 0
http详解4-http协议和工作流程4形成post请求
http详解4-http协议和工作流程4形成post请求
0 0
学习笔记jira项目24用fetch抽象http请求
学习笔记jira项目24用fetch抽象http请求
0 0
文章
问答
文章排行榜
最热
最新
相关电子书
更多
阿里巴巴HTTP 2.0实践及无线通信协议的演进之路
立即下载
数据智能人机交互实践
立即下载
低代码开发师(初级)实战教程
立即下载