OkHttp使用进阶 译自OkHttp Github官方教程

没有使用过OkHttp的,可以先看OkHttp使用介绍

英文版原版地址

Recipes · square/okhttp Wiki

同步get

下载一个文件,打印他的响应头,以string形式打印响应体。
响应体的 string()
方法对于小文档来说十分方便、高效。但是如果响应体太大(超过1MB),应避免适应 string()方法
,因为他会将把整个文档加载到内存中。
对于超过1MB的响应body,应使用流的方式来处理body。

private final OkHttpClient client = new OkHttpClient();

public void run() throws Exception {
	Request request = new Request.Builder()
	    .url("http://publicobject.com/helloworld.txt")
	    .build();

	Response response = client.newCall(request).execute();
	if (!response.isSuccessful()) throw new IOException("Unexpected code " + response);

	Headers responseHeaders = response.headers();
	for (int i = 0; i < responseHeaders.size(); i++) {
	  System.out.println(responseHeaders.name(i) + ": " + responseHeaders.value(i));
	}

	System.out.println(response.body().string());
}

异步get

在一个工作线程中下载文件,当响应可读时回调Callback接口。读取响应时会阻塞当前线程。OkHttp现阶段不提供异步api来接收响应体。

private final OkHttpClient client = new OkHttpClient();

public void run() throws Exception {
	Request request = new Request.Builder()
	    .url("http://publicobject.com/helloworld.txt")
	    .build();

	client.newCall(request).enqueue(new Callback() {
	  @Override public void onFailure(Request request, Throwable throwable) {
	    throwable.printStackTrace();
	  }

	  @Override public void onResponse(Response response) throws IOException {
	    if (!response.isSuccessful()) throw new IOException("Unexpected code " + response);

	    Headers responseHeaders = response.headers();
	    for (int i = 0; i < responseHeaders.size(); i++) {
	      System.out.println(responseHeaders.name(i) + ": " + responseHeaders.value(i));
	    }

	    System.out.println(response.body().string());
	  }
	});
}

提取响应头

典型的HTTP头 像是一个 Map<String, String> :每个字段都有一个或没有值。但是一些头允许多个值,像Guava的Multimap。例如:HTTP响应里面提供的Vary响应头,就是多值的。OkHttp的api试图让这些情况都适用。
当写请求头的时候,使用header(name,
value)
可以设置唯一的name、value。如果已经有值,旧的将被移除,然后添加新的。使用addHeader(name,
value)
可以添加多值(添加,不移除已有的)。
当读取响应头时,使用header(name)返回最后出现的name、value。通常情况这也是唯一的name、value。如果没有值,那么header(name)将返回null。如果想读取字段对应的所有值,使用headers(name)会返回一个list。
为了获取所有的Header,Headers类支持按index访问。

private final OkHttpClient client = new OkHttpClient();

public void run() throws Exception {
	Request request = new Request.Builder()
	    .url("https://api.github.com/repos/square/okhttp/issues")
	    .header("User-Agent", "OkHttp Headers.java")
	    .addHeader("Accept", "application/json; q=0.5")
	    .addHeader("Accept", "application/vnd.github.v3+json")
	    .build();

	Response response = client.newCall(request).execute();
	if (!response.isSuccessful()) throw new IOException("Unexpected code " + response);

	System.out.println("Server: " + response.header("Server"));
	System.out.println("Date: " + response.header("Date"));
	System.out.println("Vary: " + response.headers("Vary"));
}

Post方式提交String

使用HTTP POST提交请求到服务。这个例子提交了一个markdown文档到web服务,以HTML方式渲染markdown。因为整个请求体都在内存中,因此避免使用此api提交大文档(大于1MB)。

public static final MediaType MEDIA_TYPE_MARKDOWN
  = MediaType.parse("text/x-markdown; charset=utf-8");

private final OkHttpClient client = new OkHttpClient();

public void run() throws Exception {
	String postBody = ""
	    + "Releases\n"
	    + "--------\n"
	    + "\n"
	    + " * _1.0_ May 6, 2013\n"
	    + " * _1.1_ June 15, 2013\n"
	    + " * _1.2_ August 11, 2013\n";

	Request request = new Request.Builder()
	    .url("https://api.github.com/markdown/raw")
	    .post(RequestBody.create(MEDIA_TYPE_MARKDOWN, postBody))
	    .build();

	Response response = client.newCall(request).execute();
	if (!response.isSuccessful()) throw new IOException("Unexpected code " + response);

	System.out.println(response.body().string());
}

Post方式提交流

以流的方式POST提交请求体。请求体的内容由流写入产生。这个例子是流直接写入Okio的BufferedSink。你的程序可能会使用OutputStream,你可以使用BufferedSink.outputStream()来获取。

public static final MediaType MEDIA_TYPE_MARKDOWN
      = MediaType.parse("text/x-markdown; charset=utf-8");

private final OkHttpClient client = new OkHttpClient();

public void run() throws Exception {
	RequestBody requestBody = new RequestBody() {
	  @Override public MediaType contentType() {
	    return MEDIA_TYPE_MARKDOWN;
	  }

	  @Override public void writeTo(BufferedSink sink) throws IOException {
	    sink.writeUtf8("Numbers\n");
	    sink.writeUtf8("-------\n");
	    for (int i = 2; i <= 997; i++) {
	      sink.writeUtf8(String.format(" * %s = %s\n", i, factor(i)));
	    }
	  }

	  private String factor(int n) {
	    for (int i = 2; i < n; i++) {
	      int x = n / i;
	      if (x * i == n) return factor(x) + " × " + i;
	    }
	    return Integer.toString(n);
	  }
	};

	Request request = new Request.Builder()
	    .url("https://api.github.com/markdown/raw")
	    .post(requestBody)
	    .build();

	Response response = client.newCall(request).execute();
	if (!response.isSuccessful()) throw new IOException("Unexpected code " + response);

	System.out.println(response.body().string());
}

Post方式提交文件

以文件作为请求体是十分简单的。

public static final MediaType MEDIA_TYPE_MARKDOWN
  = MediaType.parse("text/x-markdown; charset=utf-8");

private final OkHttpClient client = new OkHttpClient();

public void run() throws Exception {
	File file = new File("README.md");

	Request request = new Request.Builder()
	    .url("https://api.github.com/markdown/raw")
	    .post(RequestBody.create(MEDIA_TYPE_MARKDOWN, file))
	    .build();

	Response response = client.newCall(request).execute();
	if (!response.isSuccessful()) throw new IOException("Unexpected code " + response);

	System.out.println(response.body().string());
}

Post方式提交表单

使用FormEncodingBuilder来构建和HTML<form>标签相同效果的请求体。键值对将使用一种HTML兼容形式的URL编码来进行编码。

private final OkHttpClient client = new OkHttpClient();

public void run() throws Exception {
	RequestBody formBody = new FormEncodingBuilder()
	    .add("search", "Jurassic Park")
	    .build();
	Request request = new Request.Builder()
	    .url("https://en.wikipedia.org/w/index.php")
	    .post(formBody)
	    .build();

	Response response = client.newCall(request).execute();
	if (!response.isSuccessful()) throw new IOException("Unexpected code " + response);

	System.out.println(response.body().string());
}

Post方式提交分块请求

MultipartBuilder可以构建复杂的请求体,与HTML文件上传形式兼容。多块请求体中每块请求都是一个请求体,可以定义自己的请求头。这些请求头可以用来描述这块请求,例如他的Content-Disposition。如果Content-LengthContent-Type可用的话,他们会被自动添加到请求头中。

private static final String IMGUR_CLIENT_ID = "...";
private static final MediaType MEDIA_TYPE_PNG = MediaType.parse("image/png");

private final OkHttpClient client = new OkHttpClient();

public void run() throws Exception {
	// Use the imgur image upload API as documented at https://api.imgur.com/endpoints/image
	RequestBody requestBody = new MultipartBuilder()
	    .type(MultipartBuilder.FORM)
	    .addPart(
	        Headers.of("Content-Disposition", "form-data; name=\"title\""),
	        RequestBody.create(null, "Square Logo"))
	    .addPart(
	        Headers.of("Content-Disposition", "form-data; name=\"image\""),
	        RequestBody.create(MEDIA_TYPE_PNG, new File("website/static/logo-square.png")))
	    .build();

	Request request = new Request.Builder()
	    .header("Authorization", "Client-ID " + IMGUR_CLIENT_ID)
	    .url("https://api.imgur.com/3/image")
	    .post(requestBody)
	    .build();

	Response response = client.newCall(request).execute();
	if (!response.isSuccessful()) throw new IOException("Unexpected code " + response);

	System.out.println(response.body().string());
}

使用Gson来解析JSON响应

Gson是一个在JSON和Java对象之间转换非常方便的api。这里我们用Gson来解析Github API的JSON响应。
注意:ResponseBody.charStream()使用响应头Content-Type指定的字符集来解析响应体。默认是UTF-8。

private final OkHttpClient client = new OkHttpClient();
private final Gson gson = new Gson();

public void run() throws Exception {
	Request request = new Request.Builder()
	    .url("https://api.github.com/gists/c2a7c39532239ff261be")
	    .build();
	Response response = client.newCall(request).execute();
	if (!response.isSuccessful()) throw new IOException("Unexpected code " + response);

	Gist gist = gson.fromJson(response.body().charStream(), Gist.class);
	for (Map.Entry<String, GistFile> entry : gist.files.entrySet()) {
	  System.out.println(entry.getKey());
	  System.out.println(entry.getValue().content);
	}
}

static class Gist {
	Map<String, GistFile> files;
}

static class GistFile {
	String content;
}

响应缓存

为了缓存响应,你需要一个你可以读写的缓存目录,和缓存大小的限制。这个缓存目录应该是私有的,不信任的程序应不能读取缓存内容。
一个缓存目录同时拥有多个缓存访问是错误的。大多数程序只需要调用一次new
OkHttp()
,在第一次调用时配置好缓存,然后其他地方只需要调用这个实例就可以了。否则两个缓存示例互相干扰,破坏响应缓存,而且有可能会导致程序崩溃。
响应缓存使用HTTP头作为配置。你可以在请求头中添加Cache-Control:
max-stale=3600
,OkHttp缓存会支持。你的服务通过响应头确定响应缓存多长时间,例如使用Cache-Control:
max-age=9600

private final OkHttpClient client;

public CacheResponse(File cacheDirectory) throws Exception {
	int cacheSize = 10 * 1024 * 1024; // 10 MiB
	Cache cache = new Cache(cacheDirectory, cacheSize);

	client = new OkHttpClient();
	client.setCache(cache);
}

public void run() throws Exception {
	Request request = new Request.Builder()
	    .url("http://publicobject.com/helloworld.txt")
	    .build();

	Response response1 = client.newCall(request).execute();
	if (!response1.isSuccessful()) throw new IOException("Unexpected code " + response1);

	String response1Body = response1.body().string();
	System.out.println("Response 1 response:          " + response1);
	System.out.println("Response 1 cache response:    " + response1.cacheResponse());
	System.out.println("Response 1 network response:  " + response1.networkResponse());

	Response response2 = client.newCall(request).execute();
	if (!response2.isSuccessful()) throw new IOException("Unexpected code " + response2);

	String response2Body = response2.body().string();
	System.out.println("Response 2 response:          " + response2);
	System.out.println("Response 2 cache response:    " + response2.cacheResponse());
	System.out.println("Response 2 network response:  " + response2.networkResponse());

	System.out.println("Response 2 equals Response 1? " + response1Body.equals(response2Body));
}

扩展

在这一节还提到了下面一句:
There are cache headers to force a cached response, force a
network response, or force the network response to be validated with a
conditional GET.

我不是很懂cache,平时用到的也不多,所以把Google在Android Developers一段相关的解析放到这里吧。

Force a Network Response

In some situations, such as after a user clicks a ‘refresh‘ button, it may be
necessary to skip the cache, and fetch data directly from the server. To force a
full refresh, add the no-cache directive:

connection.addRequestProperty("Cache-Control", "no-cache");

If it is only necessary to force a cached response to be validated by the server, use the more efficient max-age=0 instead:

connection.addRequestProperty("Cache-Control", "max-age=0");

Force a Cache Response

Sometimes you‘ll want to show resources if they are available immediately, but not otherwise. This can be used so your application can show something while waiting for the latest data to be downloaded. To restrict a request to locally-cached resources, add the only-if-cached directive:

try {
     connection.addRequestProperty("Cache-Control", "only-if-cached");
     InputStream cached = connection.getInputStream();
     // the resource was cached! show it
  catch (FileNotFoundException e) {
     // the resource was not cached
 }
}

This technique works even better in situations where a stale response is better than no response. To permit stale cached responses, use the max-stale directive with the maximum staleness in seconds:

int maxStale = 60 * 60 * 24 * 28; // tolerate 4-weeks stale
connection.addRequestProperty("Cache-Control", "max-stale=" + maxStale);

以上信息来自:HttpResponseCache - Android SDK | Android Developers

取消一个Call

使用Call.cancel()可以立即停止掉一个正在执行的call。如果一个线程正在写请求或者读响应,将会引发IOException。当call没有必要的时候,使用这个api可以节约网络资源。例如当用户离开一个应用时。不管同步还是异步的call都可以取消。
你可以通过tags来同时取消多个请求。当你构建一请求时,使用RequestBuilder.tag(tag)来分配一个标签。之后你就可以用OkHttpClient.cancel(tag)来取消所有带有这个tag的call。

private final ScheduledExecutorService executor = Executors.newScheduledThreadPool(1);
private final OkHttpClient client = new OkHttpClient();

public void run() throws Exception {
	Request request = new Request.Builder()
	    .url("http://httpbin.org/delay/2") // This URL is served with a 2 second delay.
	    .build();

	final long startNanos = System.nanoTime();
	final Call call = client.newCall(request);

	// Schedule a job to cancel the call in 1 second.
	executor.schedule(new Runnable() {
	  @Override public void run() {
	    System.out.printf("%.2f Canceling call.%n", (System.nanoTime() - startNanos) / 1e9f);
	    call.cancel();
	    System.out.printf("%.2f Canceled call.%n", (System.nanoTime() - startNanos) / 1e9f);
	  }
	}, 1, TimeUnit.SECONDS);

	try {
	  System.out.printf("%.2f Executing call.%n", (System.nanoTime() - startNanos) / 1e9f);
	  Response response = call.execute();
	  System.out.printf("%.2f Call was expected to fail, but completed: %s%n",
	      (System.nanoTime() - startNanos) / 1e9f, response);
	} catch (IOException e) {
	  System.out.printf("%.2f Call failed as expected: %s%n",
	      (System.nanoTime() - startNanos) / 1e9f, e);
	}
}

超时

没有响应时使用超时结束call。没有响应的原因可能是客户点链接问题、服务器可用性问题或者这之间的其他东西。OkHttp支持连接,读取和写入超时。

private final OkHttpClient client;

public ConfigureTimeouts() throws Exception {
	client = new OkHttpClient();
	client.setConnectTimeout(10, TimeUnit.SECONDS);
	client.setWriteTimeout(10, TimeUnit.SECONDS);
	client.setReadTimeout(30, TimeUnit.SECONDS);
}

public void run() throws Exception {
	Request request = new Request.Builder()
	    .url("http://httpbin.org/delay/2") // This URL is served with a 2 second delay.
	    .build();

	Response response = client.newCall(request).execute();
	System.out.println("Response completed: " + response);
}

每个call的配置

使用OkHttpClient,所有的HTTP Client配置包括代理设置、超时设置、缓存设置。当你需要为单个call改变配置的时候,clone 一个 OkHttpClient。这个api将会返回一个浅拷贝(shallow copy),你可以用来单独自定义。下面的例子中,我们让一个请求是500ms的超时、另一个是3000ms的超时。

private final OkHttpClient client = new OkHttpClient();

public void run() throws Exception {
	Request request = new Request.Builder()
	    .url("http://httpbin.org/delay/1") // This URL is served with a 1 second delay.
	    .build();

	try {
	  Response response = client.clone() // Clone to make a customized OkHttp for this request.
	      .setReadTimeout(500, TimeUnit.MILLISECONDS)
	      .newCall(request)
	      .execute();
	  System.out.println("Response 1 succeeded: " + response);
	} catch (IOException e) {
	  System.out.println("Response 1 failed: " + e);
	}

	try {
	  Response response = client.clone() // Clone to make a customized OkHttp for this request.
	      .setReadTimeout(3000, TimeUnit.MILLISECONDS)
	      .newCall(request)
	      .execute();
	  System.out.println("Response 2 succeeded: " + response);
	} catch (IOException e) {
	  System.out.println("Response 2 failed: " + e);
	}
}

处理验证

这部分和HTTP AUTH有关。
相关资料:HTTP AUTH 那些事 - 王绍全的博客
- 博客频道 - CSDN.NET

OkHttp会自动重试未验证的请求。当响应是401 Not
Authorized
时,Authenticator会被要求提供证书。Authenticator的实现中需要建立一个新的包含证书的请求。如果没有证书可用,返回null来跳过尝试。

public List<Challenge> challenges()
Returns the authorization challenges appropriate for this response‘s code. If the response code is 401 unauthorized, this returns the "WWW-Authenticate" challenges. If the response code is 407 proxy unauthorized, this returns the "Proxy-Authenticate" challenges. Otherwise this returns an empty list of challenges.

当需要实现一个Basic challenge, 使用Credentials.basic(username, password)来编码请求头。

private final OkHttpClient client = new OkHttpClient();

public void run() throws Exception {
	client.setAuthenticator(new Authenticator() {
	  @Override public Request authenticate(Proxy proxy, Response response) {
	    System.out.println("Authenticating for response: " + response);
	    System.out.println("Challenges: " + response.challenges());
	    String credential = Credentials.basic("jesse", "password1");
	    return response.request().newBuilder()
	        .header("Authorization", credential)
	        .build();
	  }

	  @Override public Request authenticateProxy(Proxy proxy, Response response) {
	    return null; // Null indicates no attempt to authenticate.
	  }
	});

	Request request = new Request.Builder()
	    .url("http://publicobject.com/secrets/hellosecret.txt")
	    .build();

	Response response = client.newCall(request).execute();
	if (!response.isSuccessful()) throw new IOException("Unexpected code " + response);

	System.out.println(response.body().string());
}
时间: 2024-12-14 23:26:22

OkHttp使用进阶 译自OkHttp Github官方教程的相关文章

【转】OkHttp使用进阶 译自OkHttp Github官方教程

作者:GavinCT 出处:http://www.cnblogs.com/ct2011/ 英文版原版地址 Recipes · square/okhttp Wiki 同步get 下载一个文件,打印他的响应头,以string形式打印响应体.响应体的 string() 方法对于小文档来说十分方便.高效.但是如果响应体太大(超过1MB),应避免适应 string()方法 ,因为他会将把整个文档加载到内存中.对于超过1MB的响应body,应使用流的方式来处理body. private final OkHt

GitHub官方Markdown语法教程

说明:Markdown随着编译器不一样,语法也都不一样,但这份GitHub提供的官方教程,基本学会这份就够了. https://guides.github.com/features/mastering-markdown/ https://guides.github.com/pdfs/markdown-cheatsheet-online.pdf 原文地址:https://www.cnblogs.com/EasonJim/p/8463510.html

GitHub Desktop for Mac(Github官方桌面客户端) v2.2.4

哪里有Github官方桌面客户端?GitHub Desktop for Mac是mac上一款功能强大的github客户端工具,可以帮助您以更直观和简化的方式连接到您的github帐户并管理您的存储库和其他相关活动,而且github Desktop附带一个设置助手,可帮助您输入github凭据并自动填充Git配置.这样您就可以在几分钟内开始编码.感兴趣的朋友们可以尝试下载使用哦! Github官方桌面客户端 https://www.macdown.com/mac/4687.html 属性可以轻松地

Git和Github简单教程

网络上关于Git和GitHub的教程不少,但是这些教程有的命令太少不够用,有的命令太多,使得初期学习的时候需要额外花不少时间在一些当前用不到的命令上.这篇文章主要的目标是用较少的时间学习Git和GitHub的基本使用.在足够一般使用的前提下,尽量减少命令.如果需要其他命令,到时候再去其他地方了解就行了. 目录: 零.Git是什么 一.Git的主要功能:版本控制 二.概览 三.Git for Windows软件安装 四.本地Git的使用 五.Github与Git的关联 六.团队合作开发 七.Git

GitHub入门教程 Hello World for GitHub

      Intro                              1.简介 What is GitHub?           2.什么是github? Create a Repository     3.创建一个库 Create a Branch            4.创建一个分支 Make a Commit            5.做一次提交 Open a Pull Request     6.提出一次Pull Request Merge Pull Request  

Nutch教程中文翻译1(官方教程,中英对照)——Nutch的编译、安装和简单运行

本教程是Nutch官方教程的翻译,采用逐段翻译的方法,并加上自己的解释. 本文由精简导航提供. 本文原版发布在CSDN博客和精简导航,并且文章在持续修改和更新.其他网站出现皆为转载,转载的文章不一定完整.请浏览原网页. 本教程虽然是Nutch 1.x的教程,但是官网上Nutch2.x的教程只是告诉我们怎么去配置一些新特性.Nutch2.x的基础教程,仍在在本教程中. Introduction Apache Nutch is an open source Web crawler written i

SwiftUI 官方教程(一)

完整中文教程及代码请查看 https://github.com/WillieWangWei/SwiftUI-Tutorials 创建和组合 View 此部分将指引你构建一个发现和分享您喜爱地方的 iOS app —— Landmarks .首先我们来构建显示地标详细信息的 view. Landmarks 使用 stacks 将 image.text 等组件进行组合和分层,以此来给 view 布局.如果想给视图添加地图,我们需要引入标准 MapKit 组件.在我们调整设计时,Xcode 可以作出实

内含star项目推荐:程序员小白人手必备,GitHub搜索教程

说起GitHub,相信在座的各位都不会感到陌生,作为全球最大的社交编程及代码托管网站,GitHub几乎对于每一个程序员来说都是一个必不可少的工具. GitHub被喻为全球最大的"同兴趣交友网站"以及代码仓库,不管你是想上传自己的代码,以此保存代码或炫技,还是想要跟全世界优秀的技术大牛学习编程经验,GitHub统统都能满足你. 技术界有一个说法,GitHub越来越成为衡量程序员能力的指标了,虽然并不是决定性的,但关注数.Fork数和star数在一定程度上也反映了一个程序员的编程实力.现在

学习ASP .NET MVC5官方教程总结(七)Edit方法和Edit视图详解

学习ASP .NET MVC5官方教程总结(七)Edit方法和Edit视图详解 在本章中,我们研究生成的Edit方法和视图.但在研究之前,我们先将 release date 弄得好看一点.打开Models\Movie.cs 文件.先添加一个引用: <span style="font-size:14px;">using System.ComponentModel.DataAnnotations;</span> 然后在Movie类中添加以下代码: [Display(