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


相关文章
|
18天前
|
数据采集
Haskell爬虫:连接管理与HTTP请求性能
Haskell爬虫:连接管理与HTTP请求性能
|
25天前
|
JSON 安全 前端开发
类型安全的 Go HTTP 请求
类型安全的 Go HTTP 请求
|
1天前
|
监控 网络协议 应用服务中间件
【Tomcat源码分析】从零开始理解 HTTP 请求处理 (第一篇)
本文详细解析了Tomcat架构中复杂的`Connector`组件。作为客户端与服务器间沟通的桥梁,`Connector`负责接收请求、封装为`Request`和`Response`对象,并传递给`Container`处理。文章通过四个关键问题逐步剖析了`Connector`的工作原理,并深入探讨了其构造方法、`init()`与`start()`方法。通过分析`ProtocolHandler`、`Endpoint`等核心组件,揭示了`Connector`初始化及启动的全过程。本文适合希望深入了解Tomcat内部机制的读者。欢迎关注并点赞,持续更新中。如有问题,可搜索【码上遇见你】交流。
【Tomcat源码分析】从零开始理解 HTTP 请求处理 (第一篇)
|
24天前
|
数据采集 JSON API
异步方法与HTTP请求:.NET中提高响应速度的实用技巧
本文探讨了在.NET环境下,如何通过异步方法和HTTP请求提高Web爬虫的响应速度和数据抓取效率。介绍了使用HttpClient结合async和await关键字实现异步HTTP请求,避免阻塞主线程,并通过设置代理IP、user-agent和cookie来优化爬虫性能。提供了代码示例,演示了如何集成这些技术以绕过目标网站的反爬机制,实现高效的数据抓取。最后,通过实例展示了如何应用这些技术获取API的JSON数据,强调了这些方法在提升爬虫性能和可靠性方面的重要性。
异步方法与HTTP请求:.NET中提高响应速度的实用技巧
|
10天前
|
JSON JavaScript 前端开发
Haskell中的数据交换:通过http-conduit发送JSON请求
Haskell中的数据交换:通过http-conduit发送JSON请求
|
12天前
|
JSON API 开发者
Python网络编程新纪元:urllib与requests库,让你的HTTP请求无所不能
【9月更文挑战第9天】随着互联网的发展,网络编程成为现代软件开发的关键部分。Python凭借简洁、易读及强大的特性,在该领域展现出独特魅力。本文介绍了Python标准库中的`urllib`和第三方库`requests`在处理HTTP请求方面的优势。`urllib`虽API底层但功能全面,适用于深入控制HTTP请求;而`requests`则以简洁的API和人性化设计著称,使HTTP请求变得简单高效。两者互补共存,共同推动Python网络编程进入全新纪元,无论初学者还是资深开发者都能从中受益。
31 7
|
10天前
|
开发者
HTTP状态码是由网页服务器返回的三位数字响应代码,用于表示请求的处理结果和状态
HTTP状态码是由网页服务器返回的三位数字响应代码,用于表示请求的处理结果和状态
17 1
|
21天前
|
缓存 网络协议 安全
揭秘浏览器背后的神秘之旅:一网打尽HTTP请求流程,让你网络冲浪更顺畅!
【8月更文挑战第31天】当在浏览器中输入网址并按下回车键时,一系列复杂的HTTP请求流程随即启动。此流程始于DNS解析,将域名转化为IP地址;接着是与服务器的TCP三次握手建立连接。连接建立后,浏览器发送HTTP请求,其中包含请求方法、资源及版本等信息。服务器接收请求并处理后返回HTTP响应,包括状态码、描述及页面内容。浏览器解析响应,若状态码为200则渲染页面,否则显示错误页。整个流程还包括缓存处理和HTTPS加密等步骤,以提升效率和保障安全。理解该流程有助于更高效地利用网络资源。通过抓包工具如Wireshark,我们能更直观地观察和学习这一过程。
34 4
|
20天前
|
JSON 监控 API
http 请求系列
XMLHttpRequest(XHR)是一种用于在客户端和服务器之间进行异步HTTP请求的API,广泛应用于动态更新网页内容,无需重新加载整个页面。本文提供了多个官方学习资源,包括MDN Web Docs、WhatWG和W3C的规范文档,涵盖属性、方法、事件及示例代码。XHR的主要应用场景包括动态内容更新、异步表单提交、局部数据刷新等,具有广泛的支持和灵活性,但也存在处理异步请求的复杂性等问题。最佳实践包括使用异步请求、处理请求状态变化、设置请求头、处理错误和超时等。这些资源和实践将帮助你更好地理解和使用XHR。
20 1
|
1月前
|
JSON API 数据格式
Python网络编程:HTTP请求(requests模块)
在现代编程中,HTTP请求几乎无处不在。无论是数据抓取、API调用还是与远程服务器进行交互,HTTP请求都是不可或缺的一部分。在Python中,requests模块被广泛认为是发送HTTP请求的最简便和强大的工具之一。本文将详细介绍requests模块的功能,并通过一个综合示例展示其应用。