對于android中的異步操作,有人說:2B青年用handler+thread逃糟,屌絲用AsyncTask吼鱼,普通青年用Loader蓬豁,而用rxjava的我都沒聽過Loader,為了做一個(gè)普通青年菇肃,所以決定學(xué)一學(xué)loader
網(wǎng)上的資料一大票的抄襲官方文檔地粪,全是通訊錄的demo,所以我寫一個(gè)不一樣的琐谤,對loader進(jìn)行一下簡單封裝蟆技,同時(shí)體會(huì)他的精髓。當(dāng)然斗忌,用完之后质礼,還是他媽的rxjava爽。
封裝一個(gè)用于做get請求的loader织阳,代碼如下:
package com.xk.androiddemo;
import android.content.Context;
import android.os.AsyncTask;
import android.os.Bundle;
import android.support.v4.content.AsyncTaskLoader;
import com.google.gson.Gson;
import com.google.gson.reflect.TypeToken;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Type;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
import java.nio.charset.Charset;
import java.util.Iterator;
/**
* Created by xuekai on 2017/12/6.
*/
public class NetGetLoader<T> extends AsyncTaskLoader<T> {
private String url;
public NetGetLoader(Context context, String baseUrl, Bundle params) {
super(context);
this.url = baseUrl;
Iterator<String> iterator = params.keySet().iterator();
//拼接參數(shù)的操作眶蕉,具體如何寫,隨意發(fā)揮
while (iterator.hasNext()) {
String next = iterator.next();
url = url + next + "=" + params.getString(next) + "&";
}
}
@Override
protected void onStartLoading() {
//在這里可以做緩存處理:例如陈哑,對請求url妻坝、返回結(jié)果伸眶、請求時(shí)間進(jìn)行存儲(chǔ)惊窖,這樣就可以實(shí)現(xiàn)同意請求,某段時(shí)間內(nèi)不走網(wǎng)絡(luò)厘贼。如果滿足條件界酒,直接調(diào)用deliverResult(返回結(jié)果)即可把結(jié)果傳給主線程,否則調(diào)用forceLoad()走網(wǎng)絡(luò)嘴秸。我這里方便起見毁欣,直接走網(wǎng)絡(luò)了。
//調(diào)用該方法 走網(wǎng)絡(luò)
forceLoad();
}
@Override
public T loadInBackground() {
//做網(wǎng)絡(luò)請求岳掐,把返回的json串轉(zhuǎn)成T類型凭疮,我這里用的是谷歌的gson,網(wǎng)絡(luò)請求是隨便copy的串述,具體根據(jù)情況發(fā)揮
return doGet(url);
}
//自由發(fā)揮區(qū)
private T doGet(String urlStr) {
String result = null;
HttpURLConnection conn = null;
ByteArrayOutputStream bout = null;
InputStream in = null;
try {
URL url = new URL(urlStr);
conn = (HttpURLConnection) url.openConnection();
conn.setReadTimeout(8000 * 2);
conn.setConnectTimeout(8000 * 2);
byte[] buffer = new byte[1024];
bout = new ByteArrayOutputStream();
in = conn.getInputStream();
int len = -1;
while ((len = in.read(buffer)) > 0) {
bout.write(buffer, 0, len);
}
result = new String(bout.toByteArray(), Charset.forName("UTF-8"));
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
if (in != null)
in.close();
if (bout != null)
bout.close();
} catch (Exception e) {
}
if (conn != null)
conn.disconnect();
}
try {
Gson gson = new Gson();
Type jsonType = new TypeToken<T>() {
}.getType();
return gson.fromJson(result, jsonType);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
}
這樣就封裝了一個(gè)簡單的用于做get請求的loader执解,使用起來也很簡單
private final int netget = 1;
authorParams = new Bundle();
authorParams.putString("id", "571");
//指明loader類型,參數(shù)列表纲酗,callback中指明返回結(jié)果的類型和baseurl
getSupportLoaderManager().initLoader(netget, authorParams, new Callback<Author>("http://baobab.kaiyanapp.com/api/v4/pgcs/detail/tab?"));
class Callback<T> implements LoaderManager.LoaderCallbacks<T> {
private String baseUrl;
public Callback(String baseUrl) {
this.baseUrl = baseUrl;
}
@Override
public Loader<T> onCreateLoader(int id, Bundle args) {
//根據(jù)標(biāo)記衰腌,創(chuàng)建對應(yīng)的loader,這里id是netget觅赊,就創(chuàng)建之前封裝好的NetGetLoader
switch (id) {
case netget:
return new NetGetLoader(MainActivity.this, baseUrl, args);
}
return null;
}
@Override
public void onLoadFinished(Loader<T> loader, T data) {
Log.i("Callback", "onLoadFinished-->" + data);
}
@Override
public void onLoaderReset(Loader<T> loader) {
}
}
loadermanager會(huì)被activity右蕊、fragment關(guān)聯(lián),所以我們不用考慮生命周期吮螺,不用擔(dān)心內(nèi)存泄漏饶囚,這是他的好處帕翻。
個(gè)人不太喜歡handler+thread的寫法,感覺把請求操作和處理請求結(jié)果的操作分離開萝风,看起來很累熊咽,而這個(gè)loader也是類似的,所以還是覺的rxjava更好一點(diǎn)闹丐。