您的位置 首页 > 数码极客

okhttp如何设置缓存

第一步,添加依赖库'com.;

自动下载两个库,OkHttp和Okio,后者是前者的通信基础

第二步,具体用法

创建OkHttpClient实例

实现一个默认的客户端,没有连接时间限制

OkHttpClient client=new OkHttpClient();

实现一个对连接时间读取时间有限制的客户端,超过时间限制,强制失败

OkHttpClient client=new OkH().connectTimeout(8000, TimeUnit.MILLISECONDS).readTimeout(8000,TimeUnit.MILLISECONDS).build();

发起Http请求(GET)

有同步和异步两种

同步需要自己开一个线程,执行网络连接

创建request请求,并在build()方法前连缀属性;

调用client的newCall()方法创建Call对象;

调用Call对象的excute()方法获取服务器返回的数据response;

返回结果数据格式依赖服务器实现


  1. new Thread(new Runnable() {
  2. @Override
  3. public void run() {
  4. OkHttpClient client=new OkH().connectTimeout(8000, TimeUnit.MILLISECONDS).readTimeout(8000,TimeUnit.MILLISECONDS).build();
  5. Request request= new Reque().url(address).build();
  6. try {
  7. Response response=client.newCall(request).execute();
  8. if () != null) {
  9. String strResponse = re().string();
  10. }
  11. } catch (IOException e) {
  12. e.printStackTrace();
  13. }
  14. }
  15. }).start();

异步get请求,因为onResponse和onFailure()默认开启子线程,需要更新UI时,需要跳转主UI线程


  1. OkHttpClient client=new OkH().connectTimeout(8000, TimeUnit.MILLISECONDS).readTimeout(8000,TimeUnit.MILLISECONDS).build();
  2. Request request= new Reque().url(address).build();
  3. client.newCall(request).enqueue(new callback() {
  4. @Override
  5. public void onFailure(Call call, IOException e) {
  6. // do something
  7. Log.d("http connect","失败");
  8. Log.d("http connect","e"+e.toString());
  9. }
  10. @Override
  11. public void onResponse(Call call, Response response) throws IOException {
  12. if ()){
  13. // do something
  14. Log.d("http connect","获取数据");
  15. Log.d("http connect","re()=="+re());
  16. Log.d("http connect","re().string()=="+re().string());
  17. }
  18. }
  19. });

发起POST请求

创建RequestBody对象来存放待提交数据;

新建request请求,在build()前调用post()方法,并把body对象传入;

后续操作与get并没有区别


  1. OkHttpClient client=new OkH().connectTimeout(8000, TimeUnit.MILLISECONDS).readTimeout(8000,TimeUnit.MILLISECONDS).build();
  2. RequestBody body=new FormBody.Builder().add("account",account).add("password",password).build();
  3. Request request= new Reque().url(address).post(body).build();
  4. client.newCall(request).enqueue(new Callback() {
  5. @Override
  6. public void onFailure(Call call, IOException e) {
  7. // do something
  8. Log.d("http connect","失败");
  9. Log.d("http connect","e"+e.toString());
  10. }
  11. @Override
  12. public void onResponse(Call call, Response response) throws IOException {
  13. if ()){
  14. // do something
  15. Log.d("http connect","获取数据");
  16. Log.d("http connect","re()=="+re());
  17. Log.d("http connect","re().string()=="+re().string());
  18. }
  19. }
  20. });

re(),这个是http协议自带的,200表示连接成功

re().string()要放在子线程,且只执行一次

做一个小小的封装


  1. public class OkHttpUtils {
  2. private static OkHttpClient client ;
  3. private static final String TAG = "OkHttpUtils";
  4. private static ConcurrentHashMap<String, List<Cookie>> cookiestore = new ConcurrentHashMap<>();
  5. //单例模式返回一个实例
  6. public static OkHttpClient getInstance(){
  7. if (client==null)
  8. synchronized ){
  9. if (client==null){
  10. //添加cookieJar,自动化管理cookie,获得一致的sessions值
  11. //添加连接超时和读取超时,在网络状况不好的时候可以做出提示.
  12. client=new OkH().cookieJar(new CookieJar() {
  13. @Override
  14. public void saveFromResponse(HttpUrl url, List<Cookie> cookies) {
  15. Log.i(TAG, "saveFromResponse: cookies="+cookies);
  16. Log.i(TAG, "saveFromResponse: cookieStore="+cookieS());
  17. cookieS(), cookies);
  18. }
  19. @Override
  20. public List<Cookie> loadForRequest(HttpUrl url) {
  21. List<Cookie> cookies = cookieS());
  22. Log.i(TAG, "loadForRequest: cookies="+cookies);
  23. return cookies != null ? cookies : new ArrayList<Cookie>();
  24. }
  25. }).connectTimeout(8000, TimeUnit.MILLISECONDS).readTimeout(8000,TimeUnit.MILLISECONDS).build();
  26. }
  27. }
  28. return client;
  29. }
  30. //封装了一个静态方法.用来实现登录,url是你要访问的网址或者接口.在使用时可以通过匿名类,根据实际情况实现一个callback,在callback方法中,分别对成功和失败做处理
  31. public static void sendHttpLoginRequest( String account, String password, Callback callback) {
  32. OkHttpClient client=OkH();
  33. RequestBody body=new FormBody.Builder().add("account",account).add("password",password).build();
  34. Request request= new Reque().url("url").post(body).build();
  35. client.newCall(request).enqueue(callback);
  36. }
  37. }

下面实际使用下,demo,点击获取按钮,会把获取到的信息设置到TextView中


  1. public class Main2Activity extends AppCompatActivity implements View.OnClickListener{
  2. private TextView mContent;
  3. private Button bInsquire;
  4. private Button bInfor;
  5. private Button bModifyPhone;
  6. private static final String TAG = "Main2Activity";
  7. //定义常量
  8. public static final int GET_CONTACT=1;
  9. public static final int GET_INFOR=2;
  10. public static final int MODIFY_PHONE=3;
  11. //定义一个内部Handler,用来获取callback发送的消息,重写内部方法,并处理消息
  12. private Handler handler=new Handler(){
  13. @Override
  14. public void handleMessage(Message msg) {
  15. if (msg!=null){
  16. switch ){
  17. case GET_CONTACT:
  18. mCon((String);
  19. case GET_INFOR:
  20. mCon((String);
  21. case MODIFY_PHONE:
  22. mCon((String);
  23. }
  24. }else{
  25. Log.i(TAG, "hanlderNews: msg null");
  26. }
  27. }
  28. });
  29. @Override
  30. protected void onCreate(Bundle savedInstanceState) {
  31. (savedInstanceState);
  32. setContentView);
  33. mContent=findViewById);
  34. bInsquire=findViewById);
  35. bInfor=findViewById);
  36. bModifyPhone=findViewById);
  37. bIn(this);
  38. bIn(this);
  39. bModi(this);
  40. }
  41. @Override
  42. public void onClick(View view) {
  43. switch ()){
  44. case R.id.b_inquire_contacts:
  45. Log.i(TAG, "onClick: bInsquire");
  46. //匿名类,重写了失败和成功的处理,成功的时候,给Handler发送消息,消息内容为获得的相应体Response的内容
  47. OkH(new Callback() {
  48. @Override
  49. public void onFailure(Call call, IOException e) {
  50. Log.i(TAG, "onFailure: connect failed");
  51. }
  52. @Override
  53. public void onResponse(Call call, Response response) throws IOException {
  54. if (response!=null){
  55. String strResponse=re().string();
  56. Log.d("http connect","re()=="+re());
  57. Log.d("http connect","re().toString()=="+strResponse);
  58. Message msg=Me();
  59. m;
  60. m;
  61. mHanlderU(msg);
  62. }else {
  63. Log.i(TAG, "onResponse: response failed");
  64. }
  65. }
  66. });
  67. break;
  68. case R.id.b_Information:
  69. case R.id.b_modify_phone:
  70. }
  71. }
  72. }

注意,在新建Handler的时候,AS3.0版本以后会出现告警信息

This Handler class should be static or leaks might occur (anonymous android.os.Handler)

这是由于handler获取到activity的引用,可能由于消息处理的不及时,延时任务会导致activity不能被成功finish(),出现内存泄漏的风险,这里参考

对hanlder进行封装,持有当前活动的弱引用,由于封装的hanlder会处理不同的message,所以在封装Handler的handlerMessage()方法中回调接口方法,在不同的活动里根据情况实现这个接口即可

接口方法


  1. public interface HandlerNewsInterface {
  2. void hanlderNews(Message msg);
  3. }

封装handler


  1. public class HandlerUtils extends Handler {
  2. private WeakReference<Activity> mActivityReference;
  3. private HandlerNewsInterface mHanlderNews;
  4. HandlerUtils(Activity mActivity,HandlerNewsInterface mHanlderNews) {
  5. = new WeakReference<>(mActivity);
  6. ;
  7. }
  8. @Override
  9. public void handleMessage(Message msg) {
  10. mHanlderNews.hanlderNews(msg);
  11. (msg);
  12. }
  13. }

所以上面的Demo中的Handler就可以这样实现


  1. private HandlerUtils mHanlderUtils=new HandlerUtils(this, new HandlerNewsInterface() {
  2. @Override
  3. public void hanlderNews(Message msg) {
  4. if (msg!=null){
  5. switch ){
  6. case GET_CONTACT:
  7. mCon((String);
  8. case GET_INFOR:
  9. mCon((String);
  10. case MODIFY_PHONE:
  11. mCon((String);
  12. }
  13. }else{
  14. Log.i(TAG, "hanlderNews: msg null");
  15. }
  16. }
  17. });

记得要在活动消失前,移除hanlder中添加的消息


  1. @Override
  2. protected void onDestroy() {
  3. mHandlerU(GET_CONTACT);
  4. mHandlerU(GET_INFOR);
  5. mHandlerU(MODIFY_PHONE);
  6. ();
  7. }

注意,

A.网络请求,需要在配置文件中声明权限

记得在AndroidMani中配置网络权限。

<uses-permission android:name="android.;></uses-permission>

B.对okhttp进行底层封装

对okHttp的更多用法,参考

1.单例模式,获取客户端实例保证

这个唯一的实例,可以在初始化时就设置连接超时,读取超时和写入超时,cookieJar自动管理cookie,可以设置缓存大小和目录

2 对网络连接超时和读取超时的处理

首先,在设置客户端的时候,我们在build()方法中添加了connectTimOut()和ReadTimeOut()方法,添加了超时属性

接下来,在callBack中返回失败的接口中分情况处理


  1. @Override
  2. public void onFailure(Call call, IOException e) {
  3. // do something
  4. Log.d("http connect","失败");
  5. //连接失败,一般是没有网络或者服务器问题
  6. if (e instanceof ConnectException){
  7. Log.d("http connect","ConnectException e="+e.toString());
  8. }
  9. //超时异常,一般是网络不好,可以设置重连以及重连次数
  10. if (e instanceof SocketTimeoutException){
  11. Log.d("http connect","SocketTimeoutException e="+e.toString());
  12. }
  13. }

3 psot数据的类型

表单数据(键值对)

RequestBody body=new FormBody.Builder().add("pwd",pwd).add("newPwd1",new1).add("newPwd2",new2).build();

json数组


  1. //设置媒体类型。application/json表示传递的是一个json格式的对象
  2. MediaType mediaType = MediaTy("application/json");
  3. //使用JSONObject封装参数
  4. JSONObject jsonObject = new JSONObject();
  5. try {
  6. j("参数名","参数值");
  7. } catch (JSONException e) {
  8. e.printStackTrace();
  9. }
  10. //创建RequestBody对象,将参数按照指定的MediaType封装
  11. RequestBody requestBody = Reque(mediaType,j());

更多的媒体类型(MediaType)信息

图片格式 参考


  1. private void sendMessage(String fileName,File file) {
  2. RequestBody requestBody = new Mul()
  3. .setType)
  4. .addFormDataPart("headPic", fileName,
  5. Reque(MediaTy("image/jpg"), file))
  6. .build();
  7. OkHttpClient client=OkH();
  8. Request request=new Reque().url).post(requestBody).build();
  9. client.newCall(request).enqueue(new Callback() {
  10. @Override
  11. public void onFailure(Call call, IOException e) {
  12. Log.i(TAG, "onFailure: failed");
  13. }
  14. @Override
  15. public void onResponse(Call call, Response response) throws IOException {
  16. if ()){
  17. String result=re().string();
  18. Log.i(TAG, "onResponse: result="+result);
  19. }
  20. }
  21. });

关于FileName和File的获取

需要知道路径信息imagePath;


  1. String fileName=imagePa(imagePath.lastIndexOf("/")+1);
  2. File file=new File(imagePath);

表单数据提交能完成大部分的网络请求,毕竟一般发送少,下载多

如果需要发送大量的数据,或者需要接受不同格式的数据,就需要包装成json数组

4.添加拦截器

squareup公司自己提供了一个okhttp拦截器方便我们观察网络请求中的数据

使用方法

添加依赖 版本号 与okhttp一致

implementation 'com.;

新建一个实例并配置参数


  1. HttpLoggingInterceptor logging = new HttpLoggingInterceptor();
  2. logging.setLevel);

setlevel用来设置日志打印的级别,共包括了四个级别:NONE,BASIC,HEADER,BODY

BASEIC:请求/响应行

HEADER:请求/响应行 + 头

BODY:请求/响应航 + 头 + 体

给ok添加拦截器


  1. client=new OkH()
  2. .addInterceptor(logging)
  3. .build();

5 cookie的持久化

移动端和服务端连接的过程中,为了保证安全,

在ok以后,新增了专门的CookieJar类来保存Cookie


  1. new CookieJar() {
  2. private HashMap<String, List<Cookie>> cookieStore = new HashMap<>();
  3. @Override
  4. public void saveFromResponse(HttpUrl url, List<Cookie> cookies) {
  5. cookieS(), cookies);
  6. }
  7. @Override
  8. public List<Cookie> loadForRequest(HttpUrl url) {
  9. List<Cookie> cookies = cookieS());
  10. return cookies != null ? cookies : new ArrayList<Cookie>();
  11. }
  12. }

主要原理:

Url的不变的一部分,比如Url.host()作为key,服务器返回的cookie作为value

saveFromResponse()中建立一个Map,并保存

loadForRequest()中根据url.host()来取出保存的cookie

责任编辑: 鲁达

1.内容基于多重复合算法人工智能语言模型创作,旨在以深度学习研究为目的传播信息知识,内容观点与本网站无关,反馈举报请
2.仅供读者参考,本网站未对该内容进行证实,对其原创性、真实性、完整性、及时性不作任何保证;
3.本站属于非营利性站点无毒无广告,请读者放心使用!

“okhttp如何设置缓存”边界阅读