OkHttp的简单使用

来源:互联网 发布:硬盘坏了恢复数据 编辑:程序博客网 时间:2024/05/29 23:44
OKHTTP:
1.为什么要使用okhttp?
使用范围
OkHttp支持Android 2.3及其以上版本。 对于java JDK1.7以上。
官方英文网站
https://github.com/square/okhttp/wiki/Recipes
中文社区
http://www.cnblogs.com/ct2011/p/3997368.html
Android系统提供了两种HTTP通信类,HttpURLConnection和HttpClient。
关于HttpURLConnection和HttpClient的选择>>官方博客
尽管Google在大部分安卓版本中推荐使用HttpURLConnection,但是这个类相比HttpClient实在是太难用,太弱爆了。
OkHttp是一个相对成熟的解决方案,据说Android4.4的源码中可以看到HttpURLConnection已经替换成OkHttp实现了。所以我们更有理由相信OkHttp的强大。

okhttp的优点和不足
1、优点
支持SPDY, 可以合并多个到同一个主机的请,使用连接池技术减少请求的延迟(如果SPDY是可用的
话) ,
SPDY协议是Google提出的基于传输控制协议(TCP)的应用层协议,通过压缩、多路复用和优先级来缩短加载时间。该协议是一种更加快速的内容传输协议。
使用GZIP压缩减少传输的数据量,缓存响应避免重复的网络请求、拦截器等等。
2、缺点
第一缺点是消息回来需要切到主线程,主线程要自己去写,第二传入调用比较复杂。

okhttp 的机制原理

一、介绍
OKHttp是一款高效的HTTP客户端,支持连接同一地址的链接共享同一个socket,通过连接池来减小响应延迟,
还有透明的GZIP压缩,请求缓存等优势,其核心主要有路由、连接协议、拦截器、代理、安全性认证、连接池以
及网络适配,拦截器主要是指添加,移除或者转换请求或者回应的头部信息,总流程图如下:

基础使用:
// okhttp实现HttpPost 网络访问
//square 公司出的Http请求库;
//android 原始类 HttpUrlconnection和httpClient;

//高级用法
// 基于Http Post的文件上传(类似表单)
// 多文件和多参数同时上传
MultipartBuilder表单传入
// 大文件下载和下载进度回调
// 大文件上传和上传进度回调
// 支持session的保持
// 支持自签名网站https的访问,提供方法设置下证书就行
// 支持根据Tag取消请求


// okhttp实现HttpGet 网络访问
1.实例化:OkHttpClient;
2.获取对象 Request (Request 是okhttp中访问请求builder是一个辅助类);
3.获取对象Response ,获取call对象调用execute;
4.//获取Responsedbody对象。通过//获取Responsed对象body方法获取数据;
5.从Responsedbody对象中获取服务器返回的数据
// okhttp实现HttpPost 网络访问(json,键值对)
1.builder放入键值对
2.获取requestbody对象
3.获取request对象,将reqbodybody放入它
4.获取response对象
5.获取responsebody对象
6从response中获取服务器返回数据

OkHttp支持同步&异步获得数据
1. 同步get
new Thread(new Runnable() { @Overridepublicvoid run() {//1. 得到OKHttpClient对象 OkHttpClient okHttpClient =new OkHttpClient();//2. 获取Request对象 ,内部也是通过建造者模式去封装的一些请求参数 Request request =new Request.Builder() .url(url) .build();//3. 获取Call对象 Call call = okHttpClient.newCall(request);//4. 获取Response对象try { Response response = call.execute();//响应体 ResponseBody body = response.body();//可以根据自己的需要,返回相应的类型://可以返回byte数组, 可以返回 InputStream ,可以返回 字符串// byte[] bytes = body.bytes();//InputStream inputStream = body.byteStream();final String result = body.string();//注意这里是string()方法,不要写成toString()// 这个方法执行在主线程当中// 如果 当前的动作,在主线程中会立即执行// 如果 当前的动作在子线程中,会先发送到主线程中,然后去执行 runOnUiThread(new Runnable() { @Overridepublicvoid run() { mTvResult.setText(result); } }); }catch (IOException e) { e.printStackTrace(); } }}).start();

2. 异步get
//1. 得到OKHttpClient对象OkHttpClient okHttpClient =new OkHttpClient();//2. 获取Request对象Request request = new Request.Builder() .url(Constant.URL_GET_ASYNC) .build();//3. 获取Call对象Call call = okHttpClient.newCall(request);//4. 获取Response对象, 通过接口回调方式返回Responsecall.enqueue(new Callback() { @Overridepublicvoid onFailure(Call call, IOException e) {//TODO 请求失败逻辑在这里处理//TODO 注意:这里边都是子线程,所以要更改UI的时候需要发送到主线程才OK }@Overridepublicvoid onResponse(Call call, Response response)throws IOException {//TODO 请求成功时候//TODO 注意:这里边都是子线程,所以要更改UI的时候需要发送到主线程才OK String result = response.body().string(); }});

3.同步post
new Thread(new Runnable() { @Overridepublicvoid run() {//1. 得到OKHttpClient OkHttpClient okHttpClient =new OkHttpClient();//2. 得到RequestBody对象,通过FormBody.Builder()来获取,通过add方法提交键值对 RequestBody body =new FormBody.Builder() .add("key1","value1") .add("key2","value2") .build();//3. 创建Request对象 Request request =new Request.Builder() .url(url) .post(body) .build();//4. 得到Call对象 Call call = okHttpClient.newCall(request);try { Response response = call.execute();if (response.isSuccessful()) {final String result = response.body().string(); runOnUiThread(new Runnable() { @Overridepublicvoid run() { mTvResult.setText(result); } }); } }catch (IOException e) { e.printStackTrace(); } }}).start();

4.异步post
OkHttpClient okHttpClient =new OkHttpClient();RequestBody requestBody =new FormBody.Builder() .add("key1","value1") .add("key2","value2") .build();Request request =new Request.Builder() .url(url) .post(requestBody) .build();Call call = okHttpClient.newCall(request);call.enqueue(new Callback() { @Overridepublicvoid onFailure(Call call, IOException e) {//TODO 失败的,在子线程中 }@Overridepublicvoid onResponse(Call call, Response response)throws IOException {//TODO 成功时,在子线程中。 }});

提交json数据
OkHttpClient okHttpClient =new OkHttpClient();//创建RequetBody对象, 这里注意:不论什么类型的Post请求,只是 RequestBody获取的方式不一样,其他步骤的一模一样MediaType mediaType = MediaType.parse("application/json; charset=utf-8");RequestBody requestBody = RequestBody.create(mediaType, YOURJSONSTRING);//获取Request对象Request request = new Request.Builder() .url(url) .post(requestBody) .build();//获取Call对象Call call = okHttpClient.newCall(request);call.enqueue(new Callback() { @Overridepublicvoid onFailure(Call call, IOException e) { }@Overridepublicvoid onResponse(Call call, Response response)throws IOException { }});


解决okhttp 报java.lang.IllegalStateException: closed,java.lang.IllegalStateException: closed,原因为OkHttp请求回调中response.body().string()只能有效调用一次
原创粉丝点击