okHttp封裝,使用超簡單的網路請求實現
文章內容概要:
- 框架需要解決的問題,及思路;
- 具體實施。
說明
okHttp是一款實用高效,使用簡單的網路模組實現,雖然使用簡單,但要在專案中使用還是得進行簡單的封裝,不可能每次請求都去寫十幾,甚至幾十行相同的程式碼,這樣是相當低效的,說這麼多都算是廢話。
來看看本次封裝我想要達到的效果:
1. 呼叫起來要簡單,我可以選擇請求方式:getAsyn、getSync、post、fileUpload、fileDownload、displayImage。呼叫者只提供請求url,請求引數。至於內部如何發起請求,肯定是呼叫者所不關心的。
2. 返回值是呼叫者所關心的,呼叫成功或失敗是呼叫者關心的,成功後我可以拿到返回結果,呼叫失敗我可以選擇是否處理,或者我不處理的時候可以預設提示請求失敗。像顯示圖片這種請求,請求完後希望能直接更新UI。當然也可以選擇顯示完後進行回撥。反正一句話,呼叫者想做什麼都能做,什麼都不做的時候程式不會出錯。
實現
首先想到下面第一種方法。
1.在okHttp幾大方法已經封裝在一個思HttpUtil的類裡,而且假設都是靜態方法。這裡在Activity中只需要呼叫譬如,HttpUtil.getAsyn(url,param);類似這樣的呼叫即可,看起來呼叫還是挺方便的。來看返回,返回有成功和失敗兩種情況,okHttp本身提供了Callback介面,我們是否可以實現這個介面就可以拿到返回了,於是呼叫就成這樣了
HttpUtil.getAsyn(url,param,new Callback() {
@Override
public void onFailure(Request request, IOException e) {
}
@Override
public void onResponse(final Response response) throws IOException {
}
});
但是這樣有一個問題,就是我呼叫方法的時候,我必須去實現這個介面,如果我不想去關心返回結果,那我還要去實現這個介面就太不友好了。那如果你不關心,大不了Activity實現這個介面,實現兩個空方法,呼叫方法就變成了
HttpUtil.getAsyn(url,param,this);//this指的是Activity,必須實現Callback介面。
這樣看起來是比匿名內部類簡潔一點。如果實在不願意去實現介面,那大不了HttpUtil多過載一個方法出來,不要求傳Callback實現就行了。這也不失為一種解決辦法。呵呵,這是我寫之前沒有想到的,似乎這麼簡單封裝就能達到我們的要求了。
HttpUtil.getAsyn(url,param);//不關心返回時呼叫。
下面把我的封裝說一下,用的java三大特性:封閉、繼承、多型。呵呵,瞬間高大上了。
給所有 Activity 建立了一個父類:BaseActivity,父類中定義了一個屬性:HttpUtil(非靜態),在父類OnCreate中初始化HttpUtil。HttpUtil中保有BaseActivity的引用。
//BaseActivity部分程式碼
public abstract class BaseActivity extends Activity {
public static final String TAG = BaseActivity.class.getSimpleName();
protected Handler mHandler = null;
public BaseUtil baseUtil;
public HttpUtil httpUtil;
@Override
protected void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
baseUtil = new BaseUtil(this);
//初始化httpUtil,並將this引用作為引數
httpUtil = new HttpUtil(this);
//初始化mHandler,用於子執行緒向主執行緒發訊息
initHandler();
}
//實現了一個http請求返回呼叫的空方法
protected void httpResponse(String url, Map<String, Object> params, Object result) {
}
//檔案下載完成後回撥,空方法
protected void completeDownload(String abstractPath) {
}
////請求出錯回撥,預設提示網路請求失敗
void httpError(String url, Map<String, Object> params) {
mHandler.post(new Runnable() {
@Override
public void run() {
Toast.makeText(BaseActivity.this, "網路請求失敗", 300).show();
}
});
}
.
.
.
}
//HttpUtil部分程式碼,這裡也並非面對真正的okHttp封裝,而在之上又包了一層。
//邊寫邊想的時候發現這一層如果拿掉也不是不可以,但這樣至少我們直接面對的程式碼是比較簡潔的。
public class HttpUtil {
private BaseActivity activity;
public HttpUtil(BaseActivity activity) {
this.activity = activity;
}
public Map<String, Object> getSyncHttp(String url) {
try {
Response response = OkHttpEngine.getInstance().getSyncHttp(activity, url);
Map<String, Object> result = new HashMap<>();
return result;
} catch (IOException e) {
return null;
}
}
.
.
.
}
進行了這兩層封裝,請求的返回方法已經被BaseActivity實現,子類Activity如果不關心,完全不用實現任何介面,請求之後的事情完全可以由BaseActivity消化吸收。若需要對返回值進行處理,子類實現httpResponse()方法即可。
Activity中呼叫網路請求方法:
public void httpTest(){
this.httpUtil.getAsynHttp("https://www.baidu.com", p1);
ImageView view = findView(R.id.imageview);
// 載入圖片
this.httpUtil.displayImage(mHandler, view, "https://www.baidu.com/img/bd_logo1.png", R.drawable.no_image);
String p = MyApplication.mkdirs("/net_image/a/b");
//下載檔案儲存到sdcard/net_image/a/b資料夾下。
this.httpUtil.downloadFile("http://a4.att.hudong.com/23/09/01300000165488122547095974400.jpg", p);
}
直接上程式碼了。來一張程式碼結構圖
從上到下
BaseActivity.java
建立了mHandler,保證瞭如果子類不建立handler時,子類呼叫http請求後,父類可以代為處理響應,更新UI,顯示提示資訊等。
建立了httpUtil,將父類的引用傳遞到httpUtil中,請求返回後,直接呼叫響應成功或失敗方法,如果子類實現了響應方法,呼叫會直達子類實現,這就是繼承時多型。
實現了響應回撥方法,若子類未實現時,父類可以代為處理。
package com.mydemo.common;
import java.util.Map;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.os.Handler;
import android.view.View;
import android.widget.Toast;
public abstract class BaseActivity extends Activity {
public static final String TAG = BaseActivity.class.getSimpleName();
protected Handler mHandler = null;
public BaseUtil baseUtil;
public HttpUtil httpUtil;
@Override
protected void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
baseUtil = new BaseUtil(this);
httpUtil = new HttpUtil(this);
initHandler();
}
protected void initHandler() {
mHandler = new Handler();
}
public <T> T findView(int id) {
View view = findViewById(id);
if (view == null || view.toString().equals("")) {
return null;
}
return (T) view;
}
/**
* 通過類名啟動Activity
*
* @param pClass
*/
protected void openActivity(Class<?> pClass) {
openActivity(pClass, null);
}
/**
* 通過類名啟動Activity,並且含有Bundle資料
*
* @param pClass
* @param pBundle
*/
protected void openActivity(Class<?> pClass, Bundle pBundle) {
Intent intent = new Intent(this, pClass);
if (pBundle != null) {
intent.putExtras(pBundle);
}
startActivity(intent);
}
/**
* 通過Action啟動Activity
*
* @param pAction
*/
protected void openActivity(String pAction) {
openActivity(pAction, null);
}
/**
* 通過Action啟動Activity,並且含有Bundle資料
*
* @param pAction
* @param pBundle
*/
protected void openActivity(String pAction, Bundle pBundle) {
Intent intent = new Intent(pAction);
if (pBundle != null) {
intent.putExtras(pBundle);
}
startActivity(intent);
}
protected void httpResponse(String url, Map<String, Object> params, Object result) {
}
protected void completeDownload(String abstractPath) {
}
protected void httpError(String url, Map<String, Object> params, final Exception e) {
mHandler.post(new Runnable() {
@Override
public void run() {
Toast.makeText(BaseActivity.this, "網路請求失敗:" + e.getMessage(), 4000).show();
}
});
}
void httpError(String url, Map<String, Object> params) {
mHandler.post(new Runnable() {
@Override
public void run() {
Toast.makeText(BaseActivity.this, "網路請求失敗", 300).show();
}
});
}
}
BaseUtil
這個類是工具類,一併承上,非本人原創。
package com.mydemo.common;
import java.util.Locale;
import android.content.Context;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.telephony.TelephonyManager;
import android.view.inputmethod.InputMethodManager;
import android.widget.EditText;
import android.widget.Toast;
public class BaseUtil {
private Context context;
protected InputMethodManager imm;
private TelephonyManager tManager;
public BaseUtil(Context context) {
this.context = context;
tManager = (TelephonyManager) context.getSystemService(context.TELEPHONY_SERVICE);
imm = (InputMethodManager) context.getSystemService(context.INPUT_METHOD_SERVICE);
}
public void DisPlay(String content) {
Toast.makeText(context, content, 1).show();
}
public void DisplayToast(String str) {
Toast.makeText(context, str, Toast.LENGTH_SHORT).show();
}
public void hideOrShowSoftInput(boolean isShowSoft, EditText editText) {
if (isShowSoft) {
imm.showSoftInput(editText, 0);
} else {
imm.hideSoftInputFromWindow(editText.getWindowToken(), 0);
}
}
// 獲得當前程式版本資訊
public String getVersionName() throws Exception {
// 獲取packagemanager的例項
PackageManager packageManager = context.getPackageManager();
// getPackageName()是你當前類的包名,0代表是獲取版本資訊
PackageInfo packInfo = packageManager.getPackageInfo(context.getPackageName(), 0);
return packInfo.versionName;
}
// 獲得裝置資訊
public String getDeviceId() throws Exception {
String deviceId = tManager.getDeviceId();
return deviceId;
}
/**
* 獲取SIM卡序列號
*
* @return
*/
public String getToken() {
return tManager.getSimSerialNumber();
}
/* 獲得系統版本 */
public String getClientOs() {
return android.os.Build.ID;
}
/* 獲得系統版本號 */
public String getClientOsVer() {
return android.os.Build.VERSION.RELEASE;
}
// 獲得系統語言包
public String getLanguage() {
return Locale.getDefault().getLanguage();
}
public String getCountry() {
return Locale.getDefault().getCountry();
}
}
HttpUtil
該類在BaseActivity中建立,並且儲存BaseActivity物件引用。
1.使得呼叫都不需要將自身作為引數進行傳遞,減小了呼叫複雜度,
2.使得OkHttpEngine不需要BaseActivity進行繫結,使OkHttpEngine作為完全獨立的存在。
package com.mydemo.common;
import java.io.File;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
import android.os.Handler;
import android.widget.ImageView;
import com.squareup.okhttp.Response;
public class HttpUtil {
private BaseActivity activity;
public HttpUtil(BaseActivity activity) {
this.activity = activity;
}
public void getAsynHttp(String url, Map<String, Object> params) {
OkHttpEngine.getInstance().getAsynHttp(activity, url);
}
public Map<String, Object> getSyncHttp(String url) {
try {
Response response = OkHttpEngine.getInstance().getSyncHttp(activity, url);
Map<String, Object> result = new HashMap<>();
return result;
} catch (IOException e) {
return null;
}
}
public void post(String url, Map<String, Object> params) {
OkHttpEngine.getInstance().postAsynHttp(activity, url, params);
}
public void downloadFile(final String url, final String destFileDir) {
OkHttpEngine.getInstance().downloadFile(activity, url, destFileDir);
}
public void displayImage(final Handler mHandler, final ImageView view, final String imgUrl, final int errorResId) {
OkHttpEngine.getInstance().displayImage(mHandler, view, imgUrl, errorResId);
}
public void fileUpload(final String url, final Map<String, Object> params, File[] files, String[] fileKeys) {
OkHttpEngine.getInstance().fileUpload(activity, url, params, files, fileKeys);
}
}
ImageUtil
與圖片處理相關工具類,非本人原創,一併承上。
package com.mydemo.common;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.InputStream;
import java.lang.reflect.Field;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.util.DisplayMetrics;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
/**
*/
public class ImageUtils {
/**
* 根據InputStream獲取圖片實際的寬度和高度
*/
public static ImageSize getImageSize(InputStream imageStream) {
BitmapFactory.Options options = new BitmapFactory.Options();
options.inJustDecodeBounds = true;
BitmapFactory.decodeStream(imageStream, null, options);
return new ImageSize(options.outWidth, options.outHeight);
}
public static class ImageSize {
int width;
int height;
public ImageSize() {
}
public ImageSize(int width, int height) {
this.width = width;
this.height = height;
}
@Override
public String toString() {
return "ImageSize{" + "width=" + width + ", height=" + height + '}';
}
}
public static int calculateInSampleSize(ImageSize srcSize, ImageSize targetSize) {
// 源圖片的寬度
int width = srcSize.width;
int height = srcSize.height;
int inSampleSize = 1;
int reqWidth = targetSize.width;
int reqHeight = targetSize.height;
if (width > reqWidth && height > reqHeight) {
// 計算出實際寬度和目標寬度的比率
int widthRatio = Math.round((float) width / (float) reqWidth);
int heightRatio = Math.round((float) height / (float) reqHeight);
inSampleSize = Math.max(widthRatio, heightRatio);
}
return inSampleSize;
}
/**
* 根據ImageView獲適當的壓縮的寬和高
*/
public static ImageSize getImageViewSize(View view) {
ImageSize imageSize = new ImageSize();
imageSize.width = getExpectWidth(view);
imageSize.height = getExpectHeight(view);
return imageSize;
}
/**
* 根據view獲得期望的高度
*/
private static int getExpectHeight(View view) {
int height = 0;
if (view == null) return 0;
final ViewGroup.LayoutParams params = view.getLayoutParams();
// 如果是WRAP_CONTENT,此時圖片還沒載入,getWidth根本無效
if (params != null && params.height != ViewGroup.LayoutParams.WRAP_CONTENT) {
height = view.getWidth(); // 獲得實際的寬度
}
if (height <= 0 && params != null) {
height = params.height; // 獲得佈局檔案中的宣告的寬度
}
if (height <= 0) {
height = getImageViewFieldValue(view, "mMaxHeight");// 獲得設定的最大的寬度
}
// 如果寬度還是沒有獲取到,憋大招,使用螢幕的寬度
if (height <= 0) {
DisplayMetrics displayMetrics = view.getContext().getResources().getDisplayMetrics();
height = displayMetrics.heightPixels;
}
return height;
}
/**
* 根據view獲得期望的寬度
*/
private static int getExpectWidth(View view) {
int width = 0;
if (view == null) return 0;
final ViewGroup.LayoutParams params = view.getLayoutParams();
// 如果是WRAP_CONTENT,此時圖片還沒載入,getWidth根本無效
if (params != null && params.width != ViewGroup.LayoutParams.WRAP_CONTENT) {
width = view.getWidth(); // 獲得實際的寬度
}
if (width <= 0 && params != null) {
width = params.width; // 獲得佈局檔案中的宣告的寬度
}
if (width <= 0)
{
width = getImageViewFieldValue(view, "mMaxWidth");// 獲得設定的最大的寬度
}
// 如果寬度還是沒有獲取到,憋大招,使用螢幕的寬度
if (width <= 0)
{
DisplayMetrics displayMetrics = view.getContext().getResources().getDisplayMetrics();
width = displayMetrics.widthPixels;
}
return width;
}
/**
* 通過反射獲取imageview的某個屬性值
*/
private static int getImageViewFieldValue(Object object, String fieldName) {
int value = 0;
try {
Field field = ImageView.class.getDeclaredField(fieldName);
field.setAccessible(true);
int fieldValue = field.getInt(object);
if (fieldValue > 0 && fieldValue < Integer.MAX_VALUE) {
value = fieldValue;
}
} catch (Exception e) {
}
return value;
}
public static Bitmap getLoacalBitmap(String url) {
try {
FileInputStream fis = new FileInputStream(url);
return BitmapFactory.decodeStream(fis);
} catch (FileNotFoundException e) {
e.printStackTrace();
return null;
}
}
}
okHttp封裝
說明,這裡方法呼叫裡,使用BaseActivity作為引數,可能不是最好的方法,可以考慮使用實現了介面,可能更為妥當。本次暫不深究了。
package com.mydemo.common;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.FileNameMap;
import java.net.URLConnection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.Handler;
import android.widget.ImageView;
import com.squareup.okhttp.Cache;
import com.squareup.okhttp.Call;
import com.squareup.okhttp.Callback;
import com.squareup.okhttp.FormEncodingBuilder;
import com.squareup.okhttp.Headers;
import com.squareup.okhttp.MediaType;
import com.squareup.okhttp.MultipartBuilder;
import com.squareup.okhttp.OkHttpClient;
import com.squareup.okhttp.Request;
import com.squareup.okhttp.RequestBody;
import com.squareup.okhttp.Response;
public class OkHttpEngine {
private static OkHttpEngine okHttpEngine;
private OkHttpClient mOkHttpClient;
public static synchronized void initEngine() {
okHttpEngine = new OkHttpEngine();
}
public static OkHttpEngine getInstance() {
if (okHttpEngine == null) {
initEngine();
}
return okHttpEngine;
}
private OkHttpEngine() {
mOkHttpClient = new OkHttpClient();
mOkHttpClient.setConnectTimeout(15, TimeUnit.SECONDS);
mOkHttpClient.setWriteTimeout(20, TimeUnit.SECONDS);
mOkHttpClient.setReadTimeout(20, TimeUnit.SECONDS);
}
public OkHttpEngine setCache(Context mContext) {
File sdcache = mContext.getExternalCacheDir();
int cacheSize = 10 * 1024 * 1024;
mOkHttpClient.setCache(new Cache(sdcache.getAbsoluteFile(), cacheSize));
return okHttpEngine;
}
/**
* @param currentActivity
* 當前活動的activity物件
* @param url
* 請求地址
* @param params
* 引數
*/
public void getAsynHttp(final BaseActivity currentActivity, final String url) {
final Request request = new Request.Builder().url(url).build();
Call call = mOkHttpClient.newCall(request);
call.enqueue(new Callback() {
@Override
public void onFailure(Request request, IOException e) {
}
@Override
public void onResponse(final Response response) throws IOException {
String str = response.body().string();
currentActivity.httpResponse(url, null, str);
}
});
}
public Response getSyncHttp(final BaseActivity currentActivity, final String url) throws IOException {
// OkHttpClient mOkHttpClient = new OkHttpClient();
final Request request = new Request.Builder().url(url).build();
Call call = mOkHttpClient.newCall(request);
Response mResponse = call.execute();
if (mResponse.isSuccessful()) {
currentActivity.httpResponse(url, null, mResponse.body().string());
return mResponse;
} else {
currentActivity.httpError(url, null);
}
return null;
}
public void postAsynHttp(final BaseActivity currentActivity, final String url, final Map<String, Object> params) {
// OkHttpClient mOkHttpClient = new OkHttpClient();
FormEncodingBuilder builer = _getFormBuilder(params);
Request request = new Request.Builder().url(url).post(builer.build()).build();
Call call = mOkHttpClient.newCall(request);
call.enqueue(new Callback() {
@Override
public void onFailure(Request request, IOException e) {
currentActivity.httpError(url, params);
}
@Override
public void onResponse(Response response) throws IOException {
currentActivity.httpResponse(url, params, response.body().string());
}
});
}
public void fileUpload(final BaseActivity currentActivity, final String url, final Map<String, Object> params, File[] files, String[] fileKeys) {
Request request = buildMultipartFormRequest(url, files, fileKeys, params);
Call call = mOkHttpClient.newCall(request);
call.enqueue(new Callback() {
@Override
public void onFailure(Request request, IOException e) {
currentActivity.httpError(url, params);
}
@Override
public void onResponse(Response response) throws IOException {
String str = response.body().string();
currentActivity.httpResponse(url, params, response.body().string());
}
});
}
/**
* 下載檔案,返回檔案絕對路徑
*
* @param currentActivity
* @param url
* 檔案網路地址
* @param destFileDir
* 檔案儲存位置
*/
public void downloadFile(final BaseActivity currentActivity, final String url, final String destFileDir) {
final Request request = new Request.Builder().url(url).build();
final Call call = mOkHttpClient.newCall(request);
call.enqueue(new Callback() {
@Override
public void onFailure(final Request request, final IOException e) {
currentActivity.httpError(url, new HashMap<String, Object>());
}
@Override
public void onResponse(Response response) {
InputStream is = null;
byte[] buf = new byte[2048];
int len = 0;
FileOutputStream fos = null;
try {
is = response.body().byteStream();
File file = new File(destFileDir, getFileName(url));
fos = new FileOutputStream(file);
while ((len = is.read(buf)) != -1) {
fos.write(buf, 0, len);
}
fos.flush();
Map<String, Object> params = new HashMap<String, Object>();
currentActivity.completeDownload(file.getAbsolutePath());
} catch (IOException e) {
e.printStackTrace();
currentActivity.httpError(url, new HashMap<String, Object>(), e);
} finally {
try {
if (is != null)
is.close();
} catch (IOException e) {
}
try {
if (fos != null)
fos.close();
} catch (IOException e) {
}
}
}
});
}
public void displayImage(final Handler mHandler, final ImageView view, final String imgUrl, final int errorResId) {
final Request request = new Request.Builder().url(imgUrl).build();
Call call = mOkHttpClient.newCall(request);
call.enqueue(new Callback() {
@Override
public void onFailure(Request request, IOException e) {
mHandler.post(new Runnable() {
@Override
public void run() {
view.setImageResource(errorResId);
}
});
}
@Override
public void onResponse(Response response) {
InputStream is = null;
try {
is = response.body().byteStream();
ImageUtils.ImageSize actualImageSize = ImageUtils.getImageSize(is);
ImageUtils.ImageSize imageViewSize = ImageUtils.getImageViewSize(view);
int inSampleSize = ImageUtils.calculateInSampleSize(actualImageSize, imageViewSize);
try {
is.reset();
} catch (IOException e) {
response = _getAsyn(imgUrl);
is = response.body().byteStream();
}
BitmapFactory.Options ops = new BitmapFactory.Options();
ops.inJustDecodeBounds = false;
ops.inSampleSize = inSampleSize;
final Bitmap bm = BitmapFactory.decodeStream(is, null, ops);
mHandler.post(new Runnable() {
@Override
public void run() {
view.setImageBitmap(bm);
}
});
} catch (Exception e) {
setErrorResId(mHandler, view, errorResId);
} finally {
if (is != null)
try {
is.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
});
}
/**
* 同步的Get請求
*
* @param url
* @return Response
*/
private Response _getAsyn(String url) throws IOException {
final Request request = new Request.Builder().url(url).build();
Call call = mOkHttpClient.newCall(request);
Response execute = call.execute();
return execute;
}
private FormEncodingBuilder _getFormBuilder(Map<String, Object> params) {
FormEncodingBuilder builder = new FormEncodingBuilder();
if (params != null && params.size() > 0) {
Set<Entry<String, Object>> set = params.entrySet();
Iterator<Entry<String, Object>> iterator = set.iterator();
while (iterator.hasNext()) {
Entry<String, Object> entry = iterator.next();
builder.add(entry.getKey(), entry.getValue().toString());
}
}
return builder;
}
private Request buildMultipartFormRequest(String url, File[] files, String[] fileKeys, final Map<String, Object> params) {
MultipartBuilder builder = new MultipartBuilder().type(MultipartBuilder.FORM);
if (params != null && params.size() > 0) {
Set<Entry<String, Object>> set = params.entrySet();
Iterator<Entry<String, Object>> iterator = set.iterator();
while (iterator.hasNext()) {
Entry<String, Object> entry = iterator.next();
builder.addPart(Headers.of(String.format("Content-Disposition", "form-data; name=\"%s\"", entry.getKey())), RequestBody.create(null, entry.getValue().toString()));
}
}
if (files != null) {
RequestBody fileBody = null;
for (int i = 0; i < files.length; i++) {
File file = files[i];
String fileName = file.getName();
fileBody = RequestBody.create(MediaType.parse(guessMimeType(fileName)), file);
// TODO 根據檔名設定contentType
builder.addPart(Headers.of("Content-Disposition", "form-data; name=\"" + fileKeys[i] + "\"; filename=\"" + fileName + "\""), fileBody);
}
}
RequestBody requestBody = builder.build();
return new Request.Builder().url(url).post(requestBody).build();
}
private String guessMimeType(String path) {
FileNameMap fileNameMap = URLConnection.getFileNameMap();
String contentTypeFor = fileNameMap.getContentTypeFor(path);
if (contentTypeFor == null) {
contentTypeFor = "application/octet-stream";
}
return contentTypeFor;
}
private String getFileName(String path) {
int separatorIndex = path.lastIndexOf("/");
return (separatorIndex < 0) ? path : path.substring(separatorIndex + 1, path.length());
}
private void setErrorResId(final Handler mHandler, final ImageView view, final int errorResId) {
mHandler.post(new Runnable() {
@Override
public void run() {
view.setImageResource(errorResId);
}
});
}
}
MainActivity
package com.mydemo.ui;
import java.util.HashMap;
import java.util.Map;
import com.mydemo.MyApplication;
import com.mydemo.common.BaseActivity;
import com.mydemo.common.ImageUtils;
import com.mydemo.okhttp.R;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.widget.ImageView;
import android.widget.TextView;
public class MainActivity extends BaseActivity {
private final String BAIDU_INDEX1 = "http://www.baidu.com";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mHandler = new Handler(getMainLooper()) {
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
switch (msg.what) {
case 1:
break;
}
}
};
Map<String, Object> p1 = new HashMap<String, Object>();
this.httpUtil.getAsynHttp(BAIDU_INDEX1, p1);
// this.getAsynHttp(BAIDU_INDEX2, p1);
//
// this.getAsynHttp(BAIDU_INDEX3, p1);
// 要呼叫同步GET方法,但需要另起一個子執行緒發起請求,而不能直接在主執行緒發起請求。
/*
* Thread t = new Thread(new Runnable() {
* @Override
* public void run() {
* Map<String, Object> p1 = new HashMap<>();
* p1.put("s", System.currentTimeMillis());
* MainActivity.this.getSyncHttp(BAIDU_INDEX4, p1);
* }
* });
* t.start();
*/
this.httpUtil.getAsynHttp(BAIDU_INDEX1, p1);
ImageView view = findView(R.id.imageview);
// 載入圖片
this.httpUtil.displayImage(mHandler, view, "https://www.baidu.com/img/bd_logo1.png", R.drawable.no_image);
String p = MyApplication.mkdirs("/net_image/a/b");
this.httpUtil.downloadFile("http://a4.att.hudong.com/23/09/01300000165488122547095974400.jpg", p);
}
@Override
public void httpResponse(String url, Map<String, Object> params, Object result) {
Message msg = new Message();
switch (url) {
case BAIDU_INDEX1:
msg.what = 1;
msg.obj = result;
mHandler.sendMessage(msg);
break;
}
}
@Override
public void completeDownload(final String abstractPath) {
super.completeDownload(abstractPath);
mHandler.post(new Runnable() {
@Override
public void run() {
ImageView view5 = findView(R.id.imageview2);
view5.setImageBitmap(ImageUtils.getLoacalBitmap(abstractPath));
TextView view1 = findView(R.id.result5);
view1.setText("儲存位置:" + abstractPath);
}
});
}
@Override
protected void httpError(String url, Map<String, Object> params, final Exception e) {
mHandler.post(new Runnable() {
@Override
public void run() {
TextView view4 = findView(R.id.result5);
view4.setText(e.getMessage());
}
});
}
}
Application
package com.mydemo;
import java.io.File;
import android.app.Application;
import android.os.Environment;
public class MyApplication extends Application {
static MyApplication application = new MyApplication();
@Override
public void onCreate