OkHttp
OkHttp 是由鼎鼎大名的 Square 公司的一款非常出色的网络通信库。
准备工作 这里使用的 OkHttp 的版本为 3.14.x 的最后一个版本,OkHttp 的 4.0.x 版本已经全部由 Java 替换到了 Kotlin。
同时还需要再添加 Okio 的依赖库,而 Okio 在 1.x 版本是基于 Java 实现的,2.x 则是 Kotlin 实现的。
1 2 3 4 5 6 7 8 dependencies { implementation "com.squareup.okhttp3:okhttp:3.14.2" implementation 'com.squareup.okio:okio:1.17.4' }
OkHttp 在 3.13.x 以上的版本需要在 Android 5.0+ (API level 21+) 和 Java 1.8 的环境开发。3.12.x 以及以下的版本支持 Android 2.3+ (API level 9+) 和 Java 1.7 的开发环境。
1 2 3 4 5 compileOptions{ sourceCompatibility JavaVersion.VERSION_1_8 targetCompatibility JavaVersion.VERSION_1_8 }
网络权限:
1 <uses-permission android:name ="android.permission.INTERNET" />
同步 GET 请求 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 OkHttpClient client = new OkHttpClient ();Request request = new Request .Builder() .url(STR) .build(); new Thread (new Runnable () { @Override public void run () { Response response = null ; try { response = client .newCall(request) .execute(); if (response.isSuccessful()){ Log.e(TAG,response.body()!=null ?response.body().string():"response.body == null" ); }else { Log.e(TAG,String.valueOf(response)); } } catch (IOException e) { e.printStackTrace(); } } }).start();
异步 GET 请求 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 Request.Builder requestBuilder = new Request .Builder().url(STR); requestBuilder.method("GET" ,null ); Request request = requestBuilder.build();OkHttpClient okHttpClient = new OkHttpClient ();Call call = okHttpClient.newCall(request);call.enqueue(new Callback () { @Override public void onFailure (Call call, IOException e) { } @Override public void onResponse (Call call, Response response) throws IOException { String str = response.body().string(); Log.e(TAG,str); } });
异步 POST 请求 RequestBody
是一个抽象类,分别有FormBody
和MultipartBody
两个子类,前者用于传输表单类型的参数,后者则支持多类型的参数传递,例如:在传输表单类型的参数的同时,还可以传输文件。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 RequestBody formBody = new FormBody .Builder() .add("" ,"" ) .build(); Request request = new Request .Builder() .url(STR) .post(formBody) .build(); OkHttpClient okHttpClient = new OkHttpClient ();Call call = okHttpClient.newCall(request);call.enqueue(new Callback () { @Override public void onFailure (Call call, IOException e) { } @Override public void onResponse (Call call, Response response) throws IOException { String str = response.body().string(); Log.e(TAG,str); } });
异步上传 Multipart 文件 Content-Type的类型使用对照表
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 public class MainActivity extends AppCompatActivity { private static final String STR = "https://jianghouren.com" ; private static final String TAG = "TAGMainActivity" ; public static final MediaType MEDIA_TYPE_PNG = MediaType.parse("image/png" ); @Override protected void onCreate (Bundle savedInstanceState) { super .onCreate(savedInstanceState); setContentView(R.layout.activity_main); OkHttpClient okHttpClient = new OkHttpClient (); RequestBody requestBody = new MultipartBody .Builder() .setType(MultipartBody.FORM) .addFormDataPart("title" ,"a" ) .addFormDataPart("image" ,"a.png" ,RequestBody.create(MEDIA_TYPE_PNG,new File ("/sdcard/a.png" ))) .build(); Request request = new Request .Builder() .header("Authorization" ,"Client-ID " +"..." ) .url(STR) .post(requestBody) .build(); okHttpClient.newCall(request).enqueue(new Callback () { @Override public void onFailure (Call call, IOException e) { Log.e(TAG,"onFailure" ); } @Override public void onResponse (Call call, Response response) throws IOException { Log.e(TAG,response.body().string()); } }); } }
异步上传文件 1 2 <uses-permission android:name ="android.permission.WRITE_EXTERNAL_STORAGE" /> <uses-permission android:name ="android.permission.READ_EXTERNAL_STORAGE" />
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 public class MainActivity extends AppCompatActivity { private static final String STR = "https://jianghouren.com" ; private static final String TAG = "TAGMainActivity" ; public static final MediaType MEDIA_TYPE_MARKDOWN = MediaType.parse("text/x-markdown; charset=utf-8" ); @Override protected void onCreate (Bundle savedInstanceState) { super .onCreate(savedInstanceState); setContentView(R.layout.activity_main); if (ContextCompat.checkSelfPermission(this , Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) { if (ActivityCompat.shouldShowRequestPermissionRationale(this , Manifest.permission.WRITE_EXTERNAL_STORAGE)) { initData(); } else { ActivityCompat.requestPermissions(this , new String []{Manifest.permission.WRITE_EXTERNAL_STORAGE}, 11 ); } }else { initData(); } } @Override public void onRequestPermissionsResult (int requestCode, String permissions[], int [] grantResults) { switch (requestCode) { case 11 : { if (grantResults.length > 0 && grantResults[0 ] == PackageManager.PERMISSION_GRANTED) { initData(); } else { } return ; } } } private void initData () { String filepath = "" ; if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){ String inPath = Environment.getExternalStorageDirectory()+"/a.txt" ; boolean isSave=saveFile(inPath); if (isSave){ Toast.makeText(this ,"写入完成" ,Toast.LENGTH_LONG).show(); } filepath = Environment.getExternalStorageDirectory().getAbsolutePath(); }else { return ; } Log.e(TAG,"开始上传文件" ); File file = new File (filepath,"a.txt" ); Request request = new Request .Builder() .url("https://api.github.com/markdown/raw" ) .post(RequestBody.create(MEDIA_TYPE_MARKDOWN,file)) .build(); OkHttpClient okHttpClient = new OkHttpClient (); okHttpClient.newCall(request).enqueue(new Callback () { @Override public void onFailure (Call call, IOException e) { Log.e(TAG,"onFailure" ); } @Override public void onResponse (Call call, Response response) throws IOException { Log.e(TAG,response.body().string()); Log.e(TAG,String.valueOf(response)); } }); } public boolean saveFile (String inPath) { File file = new File (inPath); try { FileOutputStream fos = new FileOutputStream (file); fos.write(("OkHttp" ).getBytes()); fos.flush(); fos.close(); return true ; } catch (Exception e) { e.printStackTrace(); return false ; } } }
异步下载文件 Android P(API 28)全面禁止了非https链接,并严格审查网站的CA证书 请参考
解决办法:
1、在 res/xml 下建立 network_security_config文件,名字可以任意。
2、对文件进行配置
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 <network-security-config > <domain-config cleartextTrafficPermitted ="false" > <domain includeSubdomains ="true" > example.com</domain > <domain includeSubdomains ="true" > cdn.example2.com</domain > </domain-config > <base-config cleartextTrafficPermitted ="true" /> </network-security-config >
3、在 AndroidManifest 的 application 标签中增加如下属性android:networkSecurityConfig="@xml/network_security_config"
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 String url = "http://imgsw.cn/static/images/mingyunzhiye.webp" ;Request request = new Request .Builder().url(url).build();OkHttpClient okHttpClient = new OkHttpClient ();okHttpClient.newCall(request).enqueue(new Callback () { @Override public void onFailure (Call call, IOException e) { Log.e(TAG,"onFailure" ); } @Override public void onResponse (Call call, Response response) { Log.e(TAG,String.valueOf(response)); InputStream inputStream = response.body().byteStream(); FileOutputStream fileOutputStream = null ; String filepath = "" ; if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){ filepath = Environment.getExternalStorageDirectory().getAbsolutePath(); }else { filepath = getFilesDir().getAbsolutePath(); } File file = new File (filepath,"a.webp" ); if (null != file){ try { fileOutputStream = new FileOutputStream (file); byte [] buffer = new byte [2048 ]; int len = 0 ; while ((len = inputStream.read(buffer)) != -1 ){ fileOutputStream.write(buffer,0 ,len); } fileOutputStream.flush(); } catch (FileNotFoundException e) { Log.e(TAG,"FileNotFoundException" ); e.printStackTrace(); } catch (IOException e) { Log.e(TAG,"IOException" ); e.printStackTrace(); } } } });
设置超时时间和缓存 1 2 3 4 5 6 7 8 File sdcache = getExternalCacheDir();int cacheSize = 10 * 1024 * 1024 ;OkHttpClient.Builder builder = new OkHttpClient .Builder() .connectTimeout(15 , TimeUnit.SECONDS) .writeTimeout(20 ,TimeUnit.SECONDS) .readTimeout(20 , TimeUnit.SECONDS) .cache(new Cache (sdcache.getAbsoluteFile(),cacheSize)); OkHttpClient okHttpClient = builder.build();
1 2 3 4 5 6 Request request = new Request .Builder() .header("Accept" ,"image/webp" ) .addHeader("Charset" ,"UTF-8" ) .removeHeader("Charset" ) .url(URL) .build();
取消请求 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 public class MainActivity extends AppCompatActivity { private static final String STR = "https://jianghouren.com" ; private static final String TAG = "TAGMainActivity" ; private ScheduledExecutorService executorService = Executors.newScheduledThreadPool(1 ); @Override protected void onCreate (Bundle savedInstanceState) { super .onCreate(savedInstanceState); setContentView(R.layout.activity_main); final Request request = new Request .Builder() .url(STR) .cacheControl(CacheControl.FORCE_NETWORK) .build(); Call call = null ; OkHttpClient okHttpClient = new OkHttpClient (); call = okHttpClient.newCall(request); final Call finalCall = call; executorService.schedule(new Runnable () { @Override public void run () { finalCall.cancel(); } },100 ,TimeUnit.MILLISECONDS); call.enqueue(new Callback () { @Override public void onFailure (Call call, IOException e) { Log.e(TAG,"onFailure" ); } @Override public void onResponse (Call call, Response response) throws IOException { if (null != response.cacheResponse()){ String str = response.cacheResponse().toString(); Log.e(TAG,"cache---" +str); }else { String str = response.networkResponse().toString(); Log.e(TAG,"network---" +str); } } }); } }
拦截器 Interceptors 拦截器可以监视、重写和重试调用请求。它可以设置多个,并且调用是有顺序的。 拦截器还分为应用拦截器(Application Interceptors)和网络拦截器(Network Interceptors)。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 public class LoggingInterceptor implements Interceptor { @Override public Response intercept (Chain chain) throws IOException { Request request = chain.request(); String host = request.url().host(); Log.e("TAG" ,"intercept host: " + host); long t1 = System.nanoTime(); Log.e("TAG" , String.format("Sending request %s on %s%n%s" , request.url(), chain.connection(), request.headers())); if (host.equals("www.baidu.com" )){ Request newRequest = request.newBuilder().url("https://jianghouren.com" ).build(); Response response = chain.proceed(newRequest); long t2 = System.nanoTime(); Log.e("TAG" , String.format("Received response for %s in %.1fms%n%s" , response.request().url(), (t2 - t1) / 1e6d , response.headers())); return response; } else { return chain.proceed(request); } } }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 OkHttpClient client = new OkHttpClient .Builder() .addInterceptor(new LoggingInterceptor ()) .build(); Request request = new Request .Builder() .url("https://www.baidu.com" ) .header("User-Agent" , "OkHttp Example" ) .build(); Call call = client.newCall(request); call.enqueue(new Callback () { @Override public void onFailure (Call call, IOException e) { Log.e(TAG,"onFailure" ); } @Override public void onResponse (Call call, Response response) throws IOException { Log.e(TAG,String.valueOf(response)); } });
OkHttp 封装 1 2 3 4 5 6 7 8 9 10 public abstract class ResultCallback { public abstract void onError (Request request,Exception e) ; public abstract void onResponse (String str) throws IOException; }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 public class OkHttpEngine { private static volatile OkHttpEngine mInstance; private OkHttpClient mOkHttpClient; private Handler mHandler; public static OkHttpEngine getInstance (Context context) { if (mInstance == null ){ synchronized (OkHttpEngine.class){ if (mInstance == null ){ mInstance = new OkHttpEngine (context); } } } return mInstance; } private OkHttpEngine (Context context) { File sdcache = context.getExternalCacheDir(); int cacheSize = 10 * 1024 * 1024 ; OkHttpClient.Builder builder = new OkHttpClient .Builder() .connectTimeout(15 , TimeUnit.SECONDS) .writeTimeout(20 ,TimeUnit.SECONDS) .readTimeout(20 , TimeUnit.SECONDS) .cache(new Cache (sdcache.getAbsoluteFile(),cacheSize)); mOkHttpClient = builder.build(); mHandler = new Handler (); } public void getAsynHttp (String url,ResultCallback callback) { final Request request = new Request .Builder() .url(url) .build(); Call call = mOkHttpClient.newCall(request); dealResult(call,callback); } private void dealResult (Call call,final ResultCallback callback) { call.enqueue(new Callback () { @Override public void onFailure (Call call, IOException e) { sendFailedCallback(call.request(),e,callback); } private void sendFailedCallback (final Request request,final Exception e,final ResultCallback callback) { mHandler.post(new Runnable () { @Override public void run () { if (callback != null ){ callback.onError(request,e); } } }); } @Override public void onResponse (Call call, Response response) throws IOException { sendSuccessCallback(response.body().string(),callback); } private void sendSuccessCallback (final String string,final ResultCallback callback) { mHandler.post(new Runnable () { @Override public void run () { if (callback != null ){ try { callback.onResponse(string); } catch (IOException e) { e.printStackTrace(); } } } }); } }); } }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 public class MainActivity extends AppCompatActivity { private static final String STR = "https://jianghouren.com" ; private static final String TAG = "TAGMainActivity" ; @Override protected void onCreate (Bundle savedInstanceState) { super .onCreate(savedInstanceState); setContentView(R.layout.activity_main); OkHttpEngine.getInstance(MainActivity.this ).getAsynHttp(STR, new ResultCallback () { @Override public void onError (Request request, Exception e) { e.printStackTrace(); Log.e(TAG,"onError" ); } @Override public void onResponse (String str) throws IOException { Log.e(TAG,str); Toast.makeText(getApplicationContext(),"请求成功" ,Toast.LENGTH_SHORT).show(); } }); } }
Kotlin 写法 1、创建一个 OkHttpClient 的实例
1 val client = OkHttpClient()
2、创建一个 Request 对象来发送一条 HTTP 请求
1 val request = Request.Builder().build()
3、在 build() 之前连缀其它方法来丰富 Request 对象
1 2 3 4 val request = Request.Builder() .url("https://www.baidu.com" ) .build()
4、调用 OkHttpClient 的 newCall() 来创建一个 Call 对象,并调用它的 execute() 来发送请求并获取服务器返回的数据。
1 val response = client.newCall(request).execute()
5、Response 对象就是服务器返回的数据了,可通过如下写法来得到返回的具体内容。
1 val responseData = response.body?.string()
6、发起一条 POST 请求
1 2 3 4 5 6 7 8 9 10 11 12 val requestBody = FormBody.Builder() .add("username" ,"admin" ) .add("password" ,123456 ) .build() val request = Request.Builder() .url("https://www.baidu.com" ) .post(requestBody) .build()
示例(Kotlin):
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 class NetworkActivity : BaseActivity () { override fun onCreate (savedInstanceState: Bundle ?) { super .onCreate(savedInstanceState) setContentView(R.layout.activity_network) initView() } private fun showResponse (response: String ) { runOnUiThread{ tvRequest.text = response } } private fun initView () { btnOkHttp.setOnClickListener{ sendRequestWithOkHttp() } } private fun sendRequestWithOkHttp () { thread { try { val client = OkHttpClient() val request = Request.Builder() .url("https://jianghouren.com/" ) .build() val response = client.newCall(request).execute() val responseData = response.body?.string() if (responseData != null ){ showResponse(responseData) } }catch (e:Exception){ e.printStackTrace() } } } }
示例(Kotlin):网络请求工具类
1 2 3 4 5 6 7 8 9 10 11 12 13 14 object HttpUtil{ fun sendOkHttpRequest (address: String ,callback:okhttp3 .Callback ) { val client = OkHttpClient() val request = Request.Builder() .url(address) .build() client.newCall(request).enqueue(callback) } }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 btnOkHttp.setOnClickListener{ val address = "https://jianghouren.com/" HttpUtil.sendOkHttpRequest(address,object : Callback{ override fun onResponse (call: Call , response: Response ) { val responseData = response.body?.string() if (responseData!=null ){ showResponse(responseData) } } override fun onFailure (call: Call , e: IOException ) { } }) }
备注
参考资料:
Android 进阶之光
第一行代码(第3版)
OkHttp 项目主页地址