SpringBoot RestTemplate 的使用

一、简介

RestTemplate 在JDK HttpURLConnection、Apache HttpComponents、OkHttp等基础上,封装了更高级别的API,默认依赖JDK HttpURLConnection,连接方式默认长连接。

二、使用

2.1、引入依赖

<dependency>
  <groupId>org.springframework</groupId>
  <artifactId>spring-web</artifactId>
</dependency>

2.2、创建RestTemplate 

2.2.1、配置参数 

http:
  maxTotal: 100         #最大连接数
  defaultMaxPerRoute: 20  #并发数
  connectTimeout: 1000   #创建连接的最长时间
  connectionRequestTimeout: 500  #从连接池中获取到连接的最长时间
  socketTimeout: 10000 #数据传输的最长时间
  staleConnectionCheckEnabled: true  #提交请求前测试连接是否可用
  validateAfterInactivity: 3000000   #可用空闲连接过期时间,重用空闲连接时会先检查是否空闲时间超过这个时间,如果超过,释放socket重新建立

2.2.2、用 HttpClient 为底层客户端  建 Bean 注入 IOC 

import org.apache.http.Header;
import org.apache.http.client.HttpClient;
import org.apache.http.client.config.RequestConfig;
import org.apache.http.config.Registry;
import org.apache.http.config.RegistryBuilder;
import org.apache.http.conn.socket.ConnectionSocketFactory;
import org.apache.http.conn.socket.PlainConnectionSocketFactory;
import org.apache.http.conn.ssl.SSLConnectionSocketFactory;
import org.apache.http.impl.client.DefaultConnectionKeepAliveStrategy;
import org.apache.http.impl.client.DefaultHttpRequestRetryHandler;
import org.apache.http.impl.client.HttpClientBuilder;
import org.apache.http.impl.conn.PoolingHttpClientConnectionManager;
import org.apache.http.message.BasicHeader;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.client.ClientHttpRequestFactory;
import org.springframework.http.client.HttpComponentsClientHttpRequestFactory;
import org.springframework.http.client.OkHttp3ClientHttpRequestFactory;
import org.springframework.web.client.RestTemplate;

import java.util.ArrayList;
import java.util.List;

@Configuration
public class RestTemplateConfig {

    @Value("${http.maxTotal}")
    private Integer maxTotal;

    @Value("${http.defaultMaxPerRoute}")
    private Integer defaultMaxPerRoute;

    @Value("${http.connectTimeout}")
    private Integer connectTimeout;

    @Value("${http.connectionRequestTimeout}")
    private Integer connectionRequestTimeout;

    @Value("${http.socketTimeout}")
    private Integer socketTimeout;

    @Value("${http.staleConnectionCheckEnabled}")
    private boolean staleConnectionCheckEnabled;

    @Value("${http.validateAfterInactivity}")
    private Integer validateAfterInactivity;

    @Bean
    public RestTemplate restTemplate() {
        return new RestTemplate(httpRequestFactory());
    }

    /**
     * 使用 Apache HttpClient 作为底层客户端
     * 效率: OkHttp > Apache HttpClient > JDK HttpURLConnection
     */
    @Bean
    public ClientHttpRequestFactory httpRequestFactory() {
        return new HttpComponentsClientHttpRequestFactory(httpClient());
    }

    @Bean
    public HttpClient httpClient() {
        Registry<ConnectionSocketFactory> registry = RegistryBuilder.<ConnectionSocketFactory>create()
            .register("http", PlainConnectionSocketFactory.getSocketFactory())
            .register("https", SSLConnectionSocketFactory.getSocketFactory())
            .build();
        PoolingHttpClientConnectionManager connectionManager = new PoolingHttpClientConnectionManager(registry);
        connectionManager.setMaxTotal(maxTotal); // 最大连接数
        connectionManager.setDefaultMaxPerRoute(defaultMaxPerRoute);    // 单个路由最大连接数
        connectionManager.setValidateAfterInactivity(validateAfterInactivity); // 最大空间时间

        RequestConfig requestConfig = RequestConfig.custom()
            .setSocketTimeout(socketTimeout)        // 服务器返回数据(response)的时间,超过抛出read timeout
            .setConnectTimeout(connectTimeout)      // (握手成功)的时间,超出抛出connect timeout
            .setStaleConnectionCheckEnabled(staleConnectionCheckEnabled) // 提交前检测是否可用
            .setConnectionRequestTimeout(connectionRequestTimeout)// 从连接池中获取连接的超时时间,超时间未拿到可用连接,会抛出org.apache.http.conn.ConnectionPoolTimeoutException: Timeout waiting for connection from pool
            .build();
        // headers
        List<Header> headers = new ArrayList<>();
        headers.add(new BasicHeader("User-Agent", "Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/31.0.1650.16 Safari/537.36"));
        headers.add(new BasicHeader("Accept-Encoding", "gzip,deflate"));
        headers.add(new BasicHeader("Accept-Language", "zh-CN"));
        headers.add(new BasicHeader("Connection", "Keep-Alive"));
        headers.add(new BasicHeader("Content-type", "application/json;charset=UTF-8"));
        return HttpClientBuilder.create()
            .setDefaultRequestConfig(requestConfig)
            .setConnectionManager(connectionManager)
            .setDefaultHeaders(headers)
            // 保持长连接配置,需要在头添加Keep-Alive
            .setKeepAliveStrategy(new DefaultConnectionKeepAliveStrategy())
            // 重试次数,默认是3次,没有开启
            .setRetryHandler(new DefaultHttpRequestRetryHandler(2, true))
            .build();
    }
}

 2.2.3、用 OkHttpClient 为底层客户端  建 Bean 注入 IOC 

@Configuration
public class RestTemplateConfig {
	
    /**
     * 没有实例化RestTemplate时,初始化RestTemplate
     */
	/*
    @ConditionalOnMissingBean(RestTemplate.class)
    @Bean
    public RestTemplate restTemplate(){
		// JDK自带的 HttpURLConnection 为底层HTTP客户端实现。
        RestTemplate restTemplate = new RestTemplate();
        return restTemplate;
    }
	*/
	
    @ConditionalOnMissingBean(RestTemplate.class)
    @Bean
    public RestTemplate restTemplate(){
        RestTemplate restTemplate = new RestTemplate(getClientHttpRequestFactory());
        restTemplate.setInterceptors(Collections.singletonList(restTemplateInterceptor()));
		// 手动补充text/html格式
        MappingJackson2HttpMessageConverter jackson2HttpMessageConverter = new MappingJackson2HttpMessageConverter();
        jackson2HttpMessageConverter.setSupportedMediaTypes(Arrays.asList(MediaType.TEXT_HTML, MediaType.TEXT_PLAIN));
        restTemplate.getMessageConverters().add(jackson2HttpMessageConverter);
        return restTemplate;
    }
	
    /**
     * 用 HttpClient 为底层客户端
     */
	/* 
    private ClientHttpRequestFactory getClientHttpRequestFactory() {
        int timeout = 5000;
        RequestConfig config = RequestConfig.custom()
                .setConnectTimeout(timeout)
                .setConnectionRequestTimeout(timeout)
                .setSocketTimeout(timeout)
                .build();
        CloseableHttpClient client = HttpClientBuilder
                .create()
                .setDefaultRequestConfig(config)
                .build();
        return new HttpComponentsClientHttpRequestFactory(client);
    }
    */
	 
	/**
	 * 用 OkHttpClient 为底层客户端
	 */
	private ClientHttpRequestFactory getClientHttpRequestFactory(){
		OkHttpClient okHttpClient = new OkHttpClient.Builder()
				.connectTimeout(5, TimeUnit.SECONDS)
				.writeTimeout(5, TimeUnit.SECONDS)
				.readTimeout(5, TimeUnit.SECONDS)
				.build();
		return new OkHttp3ClientHttpRequestFactory(okHttpClient);
	}
	
    @Bean
    public RestTemplateInterceptor restTemplateInterceptor() {
        return new RestTemplateInterceptor();
    }
	
}

public class LogFilter implements Filter {
	
    @Override
    public void init(FilterConfig filterConfig) throws ServletException { }
	
    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
        MDC.put("TRACE_ID",UUID.randomUUID().toString());
        System.out.println("请求日志");
        chain.doFilter(request, response);
        System.out.println("响应日志");
    }
	
    @Override
    public void destroy() { }
}

public class RestTemplateInterceptor implements ClientHttpRequestInterceptor {
	
    @Override
    public ClientHttpResponse intercept(HttpRequest request, byte[] body, ClientHttpRequestExecution execution) throws IOException { 
        request.getHeaders().set("traceId", MDC.get("TRACE_ID"));
		
	    // headers
        List<Header> headers = new ArrayList<>();
    	headers.add(new BasicHeader("User-Agent", "Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/31.0.1650.16 Safari/537.36"));
        headers.add(new BasicHeader("Accept-Encoding", "gzip,deflate"));
        headers.add(new BasicHeader("Accept-Language", "zh-CN"));
        headers.add(new BasicHeader("Connection", "Keep-Alive"));
        headers.add(new BasicHeader("Content-type", "application/json;charset=UTF-8"));
		
        ClientHttpResponse response = execution.execute(request, body);
        return response;
    }
}

2.3、API 使用 

2.3.1、GET请求 

public <T> T getForObject(...):返回值是HTTP协议的响应体内容。
public <T> ResponseEntity<T> getForEntity(...):返回的是ResponseEntity,ResponseEntity包含响应体、HTTP状态码、contentType、contentLength、Header等信息。

//不带参的 get 请求
ResponseBean responseBean = restTemplate.getForObject(url, ResponseBean.class);

//带参的get请求 , 直接返回对象(restful风格)
ResponseBean responseBean = restTemplate.getForObject(url, ResponseBean.class, "001", "张三");

//带参的get请求 , 直接返回对象(restful风格)
Map<String, String> uriVariables = new HashMap<>();
uriVariables.put("userName", "唐三藏");
uriVariables.put("userPwd", "123456");
ResponseBean responseBean = restTemplate.getForObject(url, ResponseBean.class, uriVariables);

2.3.2、POST请求

public URI postForLocation(...):用POST创建一个新资源,返回UTI对象,可从响应中返回Location报头。
public <T> T postForObject(...):返回HTTP协议的响应体body内容。
public <T> ResponseEntity<T> postForEntity(...):返回ResponseEntity,ResponseEntity包含响应体、HTTP状态码、contentType、contentLength、Header等信息。

// post请求,模拟表单提交
// 请求头设置,x-www-form-urlencoded 格式的数据
HttpHeaders headers = new HttpHeaders();
headers.setContentType(MediaType.APPLICATION_FORM_URLENCODED);
// 提交参数设置 
MultiValueMap<String, String> map = new LinkedMultiValueMap<>();
map.add("userName", "唐三藏");
map.add("userPwd", "123456");
// 组装请求体
HttpEntity<MultiValueMap<String, String>> request = new HttpEntity<>(map, headers);
// 发起请求
ResponseBean responseBean = restTemplate.postForObject(url, request, ResponseBean.class);

// post请求,模拟JSON提交
RequestBean request = new RequestBean();
request.setUserName("唐三藏");
request.setUserPwd("123456789");
//发送post请求,并打印结果,以String类型接收响应结果JSON字符串
ResponseBean responseBean = restTemplate.postForObject(url, request, ResponseBean.class);

// post请求,重定向
//入参
RequestBean request = new RequestBean();
request.setUserName("唐三藏");
request.setUserPwd("123456789");
//用于提交完成数据之后的页面跳转,返回跳转url
URI uri = restTemplate.postForLocation(url, request);
System.out.println(uri.toString());

2.3.3、PUT、DELETE请求 

PUT请求
一般用来新增或修改资源,没返回值。
	String url = "......";
	User user = new User();
	user.setName("鲁大师");
	restTemplate.put(url,user);

DELETE请求
没返回值,与PUT请求一样。

2.3.4、EXECUTE

RestTemplate restTemplate = new RestTemplate();
//请求地址
String url = "http://172.18.20.200:8080/cms/work/queryWorks";
User user = new User();
user.setName("aaa");
HttpHeaders httpHeaders = new HttpHeaders();
httpHeaders.setContentType(MediaType.APPLICATION_JSON);
HttpEntity<User> userHttpEntity = new HttpEntity<User>(user, httpHeaders);
ResponseEntity<Object> execute = restTemplate.execute(url, HttpMethod.POST, restTemplate.httpEntityCallback(userHttpEntity), restTemplate.responseEntityExtractor(Object.class));
System.out.println(execute);

2.3.5、上传、下载文件

上传文件
	//需要上传的文件
	String filePath = "/C:/cms/aaa.jpg";
	//请求地址
	String url = "http://localhost:8080/rest/file/test/post/upload";
	// 请求头设置,multipart/form-data格式的数据
	HttpHeaders headers = new HttpHeaders();
	headers.setContentType(MediaType.MULTIPART_FORM_DATA);
	//提交参数设置
	MultiValueMap<String, Object> param = new LinkedMultiValueMap<>();
	param.add("multipartFile", new FileSystemResource(new File(filePath)));
	param.add("userCode", "anightmonarch");
	param.add("userName", "一宿君");
	// 组装请求体
	HttpEntity<MultiValueMap<String, Object>> request = new HttpEntity<>(param, headers);
	//发起请求
	ResponseBean responseBean = restTemplate.postForObject(url, request, ResponseBean.class);
	System.out.println(responseBean);

下载文件
	//需要下载的文件
	String fileName = "121dfga0ab3ba.jpg";
	//用户信息
	String userId = "123456789";
	//请求地址
	String url = "......";
	//提交参数设置
	Map<String,Object> paramMap = new HashMap<String,Object>();
	paramMap.put("fileName",fileName);
	paramMap.put("userId",userId);
	//发起请求(响应内容为字节文件)
	ResponseEntity<byte[]> response = restTemplate.getForEntity(url, byte[].class, paramMap);
	System.out.println("HTTP 响应状态:" + response.getStatusCode());
	//下载文件保存路径
	File savePath = new File("C:\\cms\\download\\img\\");
	if (!savePath.isDirectory()){
		savePath.mkdirs();
	}
	Path path = Paths.get(savePath.getPath() + File.separator + fileName);
	Files.write(path, Objects.requireNonNull(response.getBody(),"下载文件失败!"));
  • 8
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Boot使用RestTemplate,可以通过以下步骤进行配置和使用。 首先,需要在Spring Boot应用程序的配置类中创建一个RestTemplate的实例,并将其声明为一个Bean。可以使用注解`@Bean`来实现这一点。例如,在配置类中添加以下代码: ```java @Bean public RestTemplate restTemplate() { return new RestTemplate(); } ``` 然后,可以在需要使用RestTemplate的地方通过依赖注入的方式引入RestTemplate实例。可以使用`@Autowired`注解来实现依赖注入。例如: ```java @Autowired private RestTemplate restTemplate; ``` 接下来,可以使用RestTemplate的各种方法发送HTTP请求。根据请求的类型和参数的传递方式,可以选择使用不同的方法,比如`getForObject()`、`postForObject()`等。具体的使用方法可以参考中的参考资料。 总结起来,使用RestTemplate的步骤可以概括为以下几点: 1. 在配置类中创建RestTemplate的实例,并将其声明为一个Bean。 2. 在需要使用RestTemplate的地方通过依赖注入的方式引入RestTemplate实例。 3. 使用RestTemplate的方法发送HTTP请求,根据请求类型和参数的传递方式选择合适的方法。 以上就是使用RestTemplate的简单介绍和使用步骤。希望能对您有所帮助。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [SpringBoot使用RestTemplate](https://blog.csdn.net/weixin_42408447/article/details/117560736)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值