相关文章
Android网络编程(一)HTTP协议原理
Android网络编程(二)HttpClient与HttpURLConnection
Android网络编程(三)Volley用法全解析
Android网络编程(四)从源码解析volley
Android网络编程(五)OkHttp2.x用法全解析
前言
上一篇介绍了OkHttp2.x的用法,这一篇文章我们来对照OkHttp2.x版本来看看,OkHttp3使用起来有那些变化。当然,看这篇文章前建议看一下前一篇文章Android网络编程(五)OkHttp2.x用法全解析。
1.使用前准备
Android Studio 配置gradle:
<code class="hljs bash has-numbering"> compile <span class="hljs-string">'com.squareup.okhttp3:okhttp:3.2.0'</span> compile <span class="hljs-string">'com.squareup.okio:okio:1.7.0'</span></code><ul style="" class="pre-numbering"><li>1</li><li>2</li></ul>
添加网络权限:
<code class="hljs xml has-numbering"> <span class="hljs-tag"><<span class="hljs-title">uses-permission</span> <span class="hljs-attribute">android:name</span>=<span class="hljs-value">"android.permission.INTERNET"</span>/></span></code><ul style="" class="pre-numbering"><li>1</li></ul>
2.异步GET请求
惯例,请求百度:
<code class="hljs java has-numbering"> <span class="hljs-keyword">private</span> <span class="hljs-keyword">void</span> <span class="hljs-title">getAsynHttp</span>() {
mOkHttpClient=<span class="hljs-keyword">new</span> OkHttpClient();
Request.Builder requestBuilder = <span class="hljs-keyword">new</span> Request.Builder().url(<span class="hljs-string">"http://www.baidu.com"</span>);
<span class="hljs-comment">//可以省略,默认是GET请求</span>
requestBuilder.method(<span class="hljs-string">"GET"</span>,<span class="hljs-keyword">null</span>);
Request request = requestBuilder.build();
Call mcall= mOkHttpClient.newCall(request);
mcall.enqueue(<span class="hljs-keyword">new</span> Callback() {
<span class="hljs-annotation">@Override</span>
<span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">onFailure</span>(Call call, IOException e) {
}
<span class="hljs-annotation">@Override</span>
<span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">onResponse</span>(Call call, Response response) <span class="hljs-keyword">throws</span> IOException {
<span class="hljs-keyword">if</span> (<span class="hljs-keyword">null</span> != response.cacheResponse()) {
String str = response.cacheResponse().toString();
Log.i(<span class="hljs-string">"wangshu"</span>, <span class="hljs-string">"cache---"</span> + str);
} <span class="hljs-keyword">else</span> {
response.body().string();
String str = response.networkResponse().toString();
Log.i(<span class="hljs-string">"wangshu"</span>, <span class="hljs-string">"network---"</span> + str);
}
runOnUiThread(<span class="hljs-keyword">new</span> Runnable() {
<span class="hljs-annotation">@Override</span>
<span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">run</span>() {
Toast.makeText(getApplicationContext(), <span class="hljs-string">"请求成功"</span>, Toast.LENGTH_SHORT).show();
}
});
}
});
}</code><ul style="" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li><li>28</li><li>29</li><li>30</li><li>31</li><li>32</li></ul>
与2.x版本并没有什么不同,比较郁闷的是回调仍然不在UI线程。
2.异步POST请求
OkHttp3异步POST请求和OkHttp2.x有一些差别就是没有FormEncodingBuilder这个类,替代它的是功能更加强大的FormBody:
<code class="hljs vbscript has-numbering"> <span class="hljs-keyword">private</span> void postAsynHttp() {
mOkHttpClient=<span class="hljs-keyword">new</span> OkHttpClient();
RequestBody formBody = <span class="hljs-keyword">new</span> FormBody.Builder()
.add(<span class="hljs-string">"size"</span>, <span class="hljs-string">"10"</span>)
.build();
<span class="hljs-built_in">Request</span> <span class="hljs-built_in">request</span> = <span class="hljs-keyword">new</span> <span class="hljs-built_in">Request</span>.Builder()
.url(<span class="hljs-string">"http://api.1-blog.com/biz/bizserver/article/list.do"</span>)
.post(formBody)
.build();
<span class="hljs-keyword">Call</span> <span class="hljs-keyword">call</span> = mOkHttpClient.newCall(<span class="hljs-built_in">request</span>);
<span class="hljs-keyword">call</span>.enqueue(<span class="hljs-keyword">new</span> Callback() {
@Override
<span class="hljs-keyword">public</span> void onFailure(<span class="hljs-keyword">Call</span> <span class="hljs-keyword">call</span>, IOException e) {
}
@Override
<span class="hljs-keyword">public</span> void onResponse(<span class="hljs-keyword">Call</span> <span class="hljs-keyword">call</span>, <span class="hljs-built_in">Response</span> <span class="hljs-built_in">response</span>) throws IOException {
<span class="hljs-built_in">String</span> str = <span class="hljs-built_in">response</span>.body().<span class="hljs-built_in">string</span>();
<span class="hljs-built_in">Log</span>.i(<span class="hljs-string">"wangshu"</span>, str);
runOnUiThread(<span class="hljs-keyword">new</span> Runnable() {
@Override
<span class="hljs-keyword">public</span> void run() {
Toast.makeText(getApplicationContext(), <span class="hljs-string">"请求成功"</span>, Toast.LENGTH_SHORT).show();
}
});
}
});
}</code><ul style="" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li><li>28</li><li>29</li><li>30</li><li>31</li></ul>
3.异步上传文件
上传文件本身也是一个POST请求,上一篇没有讲,这里我们补上。首先定义上传文件类型:
<code class="hljs java has-numbering"> <span class="hljs-keyword">public</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">final</span> MediaType MEDIA_TYPE_MARKDOWN
= MediaType.parse(<span class="hljs-string">"text/x-markdown; charset=utf-8"</span>);</code><ul style="" class="pre-numbering"><li>1</li><li>2</li></ul>
将sdcard根目录的wangshu.txt文件上传到服务器上:
<code class="hljs vbscript has-numbering"> <span class="hljs-keyword">private</span> void postAsynFile() {
mOkHttpClient=<span class="hljs-keyword">new</span> OkHttpClient();
File file = <span class="hljs-keyword">new</span> File(<span class="hljs-string">"/sdcard/wangshu.txt"</span>);
<span class="hljs-built_in">Request</span> <span class="hljs-built_in">request</span> = <span class="hljs-keyword">new</span> <span class="hljs-built_in">Request</span>.Builder()
.url(<span class="hljs-string">"https://api.github.com/markdown/raw"</span>)
.post(RequestBody.create(MEDIA_TYPE_MARKDOWN, file))
.build();
mOkHttpClient.newCall(<span class="hljs-built_in">request</span>).enqueue(<span class="hljs-keyword">new</span> Callback() {
@Override
<span class="hljs-keyword">public</span> void onFailure(<span class="hljs-keyword">Call</span> <span class="hljs-keyword">call</span>, IOException e) {
}
@Override
<span class="hljs-keyword">public</span> void onResponse(<span class="hljs-keyword">Call</span> <span class="hljs-keyword">call</span>, <span class="hljs-built_in">Response</span> <span class="hljs-built_in">response</span>) throws IOException {
<span class="hljs-built_in">Log</span>.i(<span class="hljs-string">"wangshu"</span>,<span class="hljs-built_in">response</span>.body().<span class="hljs-built_in">string</span>());
}
});
}
</code><ul style="" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li></ul>
当然如果想要改为同步的上传文件只要调用 mOkHttpClient.newCall(request).execute()就可以了。
在wangshu.txt文件中有一行字“Android网络编程(六)OkHttp3用法全解析”我们运行程序点击发送文件按钮,最终请求网络返回的结果就是我们txt文件中的内容 :
当然不要忘了添加如下权限:
<code class="hljs xml has-numbering"> <span class="hljs-tag"><<span class="hljs-title">uses-permission</span> <span class="hljs-attribute">android:name</span>=<span class="hljs-value">"android.permission.READ_EXTERNAL_STORAGE"</span>/></span> <span class="hljs-tag"><<span class="hljs-title">uses-permission</span> <span class="hljs-attribute">android:name</span>=<span class="hljs-value">"android.permission.WRITE_EXTERNAL_STORAGE"</span>/></span></code><ul style="" class="pre-numbering"><li>1</li><li>2</li></ul>
4.异步下载文件
下载文件同样在上一篇没有讲到,实现起来比较简单,在这里下载一张图片,我们得到Response后将流写进我们指定的图片文件中就可以了。
<code class="hljs vbscript has-numbering"> <span class="hljs-keyword">private</span> void downAsynFile() {
mOkHttpClient = <span class="hljs-keyword">new</span> OkHttpClient();
<span class="hljs-built_in">String</span> url = <span class="hljs-string">"https://img-my.youkuaiyun.com/uploads/201603/26/1458988468_5804.jpg"</span>;
<span class="hljs-built_in">Request</span> <span class="hljs-built_in">request</span> = <span class="hljs-keyword">new</span> <span class="hljs-built_in">Request</span>.Builder().url(url).build();
mOkHttpClient.newCall(<span class="hljs-built_in">request</span>).enqueue(<span class="hljs-keyword">new</span> Callback() {
@Override
<span class="hljs-keyword">public</span> void onFailure(<span class="hljs-keyword">Call</span> <span class="hljs-keyword">call</span>, IOException e) {
}
@Override
<span class="hljs-keyword">public</span> void onResponse(<span class="hljs-keyword">Call</span> <span class="hljs-keyword">call</span>, <span class="hljs-built_in">Response</span> <span class="hljs-built_in">response</span>) {
InputStream inputStream = <span class="hljs-built_in">response</span>.body().byteStream();
FileOutputStream fileOutputStream = <span class="hljs-literal">null</span>;
try {
fileOutputStream = <span class="hljs-keyword">new</span> FileOutputStream(<span class="hljs-keyword">new</span> File(<span class="hljs-string">"/sdcard/wangshu.jpg"</span>));
byte[] buffer = <span class="hljs-keyword">new</span> byte[<span class="hljs-number">2048</span>];
<span class="hljs-built_in">int</span> <span class="hljs-built_in">len</span> = <span class="hljs-number">0</span>;
<span class="hljs-keyword">while</span> ((<span class="hljs-built_in">len</span> = inputStream.read(buffer)) != -<span class="hljs-number">1</span>) {
fileOutputStream.write(buffer, <span class="hljs-number">0</span>, <span class="hljs-built_in">len</span>);
}
fileOutputStream.flush();
} catch (IOException e) {
<span class="hljs-built_in">Log</span>.i(<span class="hljs-string">"wangshu"</span>, <span class="hljs-string">"IOException"</span>);
e.printStackTrace();
}
<span class="hljs-built_in">Log</span>.d(<span class="hljs-string">"wangshu"</span>, <span class="hljs-string">"文件下载成功"</span>);
}
});
}</code><ul style="" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li><li>28</li><li>29</li><li>30</li><li>31</li></ul>
5.异步上传Multipart文件
这种场景很常用,我们有时会上传文件同时还需要传其他类型的字段,OkHttp3实现起来很简单,需要注意的是没有服务器接收我这个Multipart文件,所以这里只是举个例子,具体的应用还要结合实际工作中对应的服务器。
首先定义上传文件类型:
<code class="hljs java has-numbering"><span class="hljs-keyword">private</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">final</span> MediaType MEDIA_TYPE_PNG = MediaType.parse(<span class="hljs-string">"image/png"</span>);</code><ul style="" class="pre-numbering"><li>1</li></ul>
<code class="hljs avrasm has-numbering">private void sendMultipart(){
mOkHttpClient = new OkHttpClient()<span class="hljs-comment">;</span>
RequestBody requestBody = new MultipartBody<span class="hljs-preprocessor">.Builder</span>()
<span class="hljs-preprocessor">.setType</span>(MultipartBody<span class="hljs-preprocessor">.FORM</span>)
<span class="hljs-preprocessor">.addFormDataPart</span>(<span class="hljs-string">"title"</span>, <span class="hljs-string">"wangshu"</span>)
<span class="hljs-preprocessor">.addFormDataPart</span>(<span class="hljs-string">"image"</span>, <span class="hljs-string">"wangshu.jpg"</span>,
RequestBody<span class="hljs-preprocessor">.create</span>(MEDIA_TYPE_PNG, new File(<span class="hljs-string">"/sdcard/wangshu.jpg"</span>)))
<span class="hljs-preprocessor">.build</span>()<span class="hljs-comment">;</span>
Request request = new Request<span class="hljs-preprocessor">.Builder</span>()
<span class="hljs-preprocessor">.header</span>(<span class="hljs-string">"Authorization"</span>, <span class="hljs-string">"Client-ID "</span> + <span class="hljs-string">"..."</span>)
<span class="hljs-preprocessor">.url</span>(<span class="hljs-string">"https://api.imgur.com/3/image"</span>)
<span class="hljs-preprocessor">.post</span>(requestBody)
<span class="hljs-preprocessor">.build</span>()<span class="hljs-comment">;</span>
mOkHttpClient<span class="hljs-preprocessor">.newCall</span>(request)<span class="hljs-preprocessor">.enqueue</span>(new Callback() {
@Override
public void onFailure(<span class="hljs-keyword">Call</span> <span class="hljs-keyword">call</span>, IOException e) {
}
@Override
public void onResponse(<span class="hljs-keyword">Call</span> <span class="hljs-keyword">call</span>, Response response) throws IOException {
Log<span class="hljs-preprocessor">.i</span>(<span class="hljs-string">"wangshu"</span>, response<span class="hljs-preprocessor">.body</span>()<span class="hljs-preprocessor">.string</span>())<span class="hljs-comment">;</span>
}
})<span class="hljs-comment">;</span>
}</code><ul style="" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li></ul>
6.设置超时时间和缓存
和OkHttp2.x有区别的是不能通过OkHttpClient直接设置超时时间和缓存了,而是通过OkHttpClient.Builder来设置,通过builder配置好OkHttpClient后用builder.build()来返回OkHttpClient,所以我们通常不会调用new OkHttpClient()来得到OkHttpClient,而是通过builder.build():
<code class="hljs avrasm has-numbering"> File sdcache = getExternalCacheDir()<span class="hljs-comment">;</span>
int cacheSize = <span class="hljs-number">10</span> * <span class="hljs-number">1024</span> * <span class="hljs-number">1024</span><span class="hljs-comment">;</span>
OkHttpClient<span class="hljs-preprocessor">.Builder</span> builder = new OkHttpClient<span class="hljs-preprocessor">.Builder</span>()
<span class="hljs-preprocessor">.connectTimeout</span>(<span class="hljs-number">15</span>, TimeUnit<span class="hljs-preprocessor">.SECONDS</span>)
<span class="hljs-preprocessor">.writeTimeout</span>(<span class="hljs-number">20</span>, TimeUnit<span class="hljs-preprocessor">.SECONDS</span>)
<span class="hljs-preprocessor">.readTimeout</span>(<span class="hljs-number">20</span>, TimeUnit<span class="hljs-preprocessor">.SECONDS</span>)
<span class="hljs-preprocessor">.cache</span>(new Cache(sdcache<span class="hljs-preprocessor">.getAbsoluteFile</span>(), cacheSize))<span class="hljs-comment">;</span>
OkHttpClient mOkHttpClient=builder<span class="hljs-preprocessor">.build</span>()<span class="hljs-comment">; </span></code><ul style="" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li></ul>
7.关于取消请求和封装
取消请求仍旧可以调用call.cancel(),这个没有变化,不明白的可以查看上一篇文章Android网络编程(五)OkHttp2.x用法全解析,这里就不赘述了,封装上一篇也讲过仍旧推荐OkHttpFinal,它目前是基于OkHttp3来进行封装的。
8.关于源码Demo
源码Demo很简单就四个按钮用来测试上面讲到的内容:
本文详细介绍OkHttp3在网络编程中的应用,包括异步GET、POST请求,文件上传下载及多部分文件上传等操作,并讲解如何设置超时时间和缓存。
1211

被折叠的 条评论
为什么被折叠?



