安卓屬於小團隊開發,架構的重要性在不少公司其實不是那麼的明顯,加上如今的開源框架層出不窮,更好的幫助咱們上手android項目的開發。我前兩年也在公司主導過項目開發,搭建過很多項目,之前主要的傾向是MVC,致使了activity/fragment過大,並且不少公共功能雜亂在項目中,後期維護起來不方便,最近恰好有時間,從新搭建了一個新的框架。(ps:有建議或者更好想法的能夠留言。)java
UI----面向對象
數據交互----MVP模式
數據庫------GreenDao
網絡圖片加載-----picasso
json解析-----gson
http請求----OKHttp
事件總線----eventbusandroid
以上知識點不熟悉的,能夠先熟悉下基本知識,若是已經瞭解過,能夠直接跳過下面的連接,直接看下文的使用。 MVP模式 講解地址:blog.csdn.net/dfskhgalshg…
GreenDao 講解地址:blog.csdn.net/dfskhgalshg…
picasso 講解地址:blog.csdn.net/dfskhgalshg…
OKHttp 講解地址:鴻神的博客講解地址: blog.csdn.net/lmj62356579…
eventbus 講解地址:blog.csdn.net/dfskhgalshg…數據庫
bean---------------------------------------------------存放java model對象
biz-----------------------------------------------------業務模塊,根據不能業務創建子模塊
bridge-------------------------------------------------底層功能實現跟UI層的銜接層
capabilities--------------------------------------------底層功能具體實現(後期項目迭代到必定程度穩定後會考慮以jar形式導入)
constant-----------------------------------------------常量
ui------------------------------------------------------界面,根據不一樣業務創建子模塊
util-----------------------------------------------------業務層公共方法
view---------------------------------------------------自定義view實現json
仍是按照你們的習慣思惟,從界面--->數據---->網絡----->交互,這樣的層次講解。數組
1)UI層緩存
UI層其實比較簡單,主要就是用到面向對象的封裝,BaseActivity爲基類,同時BaseActivity實現三個接口,分別爲CreateInit, PublishActivityCallBack, PresentationLayerFunc,這三個接口的做用依次是:界面初始化,頁面跳轉封裝,頁面交互封裝。PresentationLayerFunc的具體實現是在PresentationLayerFuncHelper裏面,BaseActivity類會初始化該類,把複雜的功能實現抽象出去,輕量化基類。安全
BaseActivity代碼以下所示:markdown
/**
* <基礎activity>
*
* @author caoyinfei
* @version [版本號, 2014-3-24]
* @see [相關類/方法]
* @since [V1]
*/
public abstract class BaseActivity extends Activity implements CreateInit, PublishActivityCallBack, PresentationLayerFunc, IMvpView, OnClickListener {
private PresentationLayerFuncHelper presentationLayerFuncHelper;
/**
* 返回按鈕
*/
private LinearLayout back;
/**
* 標題,右邊字符
*/
protected TextView title, right;
public BasePresenter presenter;
public final String TAG = this.getClass().getSimpleName();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
presentationLayerFuncHelper = new PresentationLayerFuncHelper(this);
initViews();
initListeners();
initData();
setHeader();
EBApplication.ebApplication.addActivity(this);
EventBus.getDefault().register(this);
}
@Override
public void setHeader() {
back = (LinearLayout) findViewById(R.id.ll_back);
title = (TextView) findViewById(R.id.tv_title);
right = (TextView) findViewById(R.id.tv_right);
back.setOnClickListener(this);
}
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.ll_back:
finish();
break;
}
}
public void onEventMainThread(Event event) {
}
@Override
protected void onResume() {
EBApplication.ebApplication.currentActivityName = this.getClass().getName();
super.onResume();
}
@Override
public void startActivity(Class<?> openClass, Bundle bundle) {
Intent intent = new Intent(this, openClass);
if (null != bundle)
intent.putExtras(bundle);
startActivity(intent);
}
@Override
public void openActivityForResult(Class<?> openClass, int requestCode, Bundle bundle) {
Intent intent = new Intent(this, openClass);
if (null != bundle)
intent.putExtras(bundle);
startActivityForResult(intent, requestCode);
}
@Override
public void setResultOk(Bundle bundle) {
Intent intent = new Intent();
if (bundle != null) ;
intent.putExtras(bundle);
setResult(RESULT_OK, intent);
finish();
}
@Override
public void showToast(String msg) {
presentationLayerFuncHelper.showToast(msg);
}
@Override
public void showProgressDialog() {
presentationLayerFuncHelper.showProgressDialog();
}
@Override
public void hideProgressDialog() {
presentationLayerFuncHelper.hideProgressDialog();
}
@Override
public void showSoftKeyboard(View focusView) {
presentationLayerFuncHelper.showSoftKeyboard(focusView);
}
@Override
public void hideSoftKeyboard() {
presentationLayerFuncHelper.hideSoftKeyboard();
}
@Override
protected void onDestroy() {
EBApplication.ebApplication.deleteActivity(this);
EventBus.getDefault().unregister(this);
if (presenter != null) {
presenter.detachView(this);
}
OkHttpManager httpManager = BridgeFactory.getBridge(Bridges.HTTP);
httpManager.cancelActivityRequest(TAG);
super.onDestroy();
}
}
複製代碼
PresentationLayerFuncHelper代碼以下所示:網絡
/**
* <頁面基礎公共功能實現>
*
* @author caoyinfei
* @version [版本號, 2016/6/6]
* @see [相關類/方法]
* @since [V1]
*/
public class PresentationLayerFuncHelper implements PresentationLayerFunc {
private Context context;
public PresentationLayerFuncHelper(Context context) {
this.context = context;
}
@Override
public void showToast(String msg) {
ToastUtil.makeText(context, msg);
}
@Override
public void showProgressDialog() {
}
@Override
public void hideProgressDialog() {
}
@Override
public void showSoftKeyboard(View focusView) {
}
@Override
public void hideSoftKeyboard() {
}
}
複製代碼
三個接口,分別爲CreateInit, PublishActivityCallBack, PresentationLayerFunc代碼以下所示:架構
/**
* <公共方法抽象>
*
* @author caoyinfei
* @version [版本號, 2014-3-24]
* @see [相關類/方法]
* @since [V1]
*/
public interface CreateInit {
/**
* 初始化佈局組件
*/
public void initViews();
/**
* 增長按鈕點擊事件
*/
void initListeners();
/**
* 初始化數據
*/
public void initData();
/**
* 初始化公共頭部
*/
public void setHeader();
}
複製代碼
/**
* <頁面跳轉封裝>
*
* @author caoyinfei
* @version [版本號, 2016/6/6]
* @see [相關類/方法]
* @since [V1]
*/
public interface PublishActivityCallBack {
/**
* 打開新界面
*
* @param openClass 新開頁面
* @param bundle 參數
*/
public void startActivity(Class<?> openClass, Bundle bundle);
/**
* 打開新界面,期待返回
*
* @param openClass 新界面
* @param requestCode 請求碼
* @param bundle 參數
*/
public void openActivityForResult(Class<?> openClass, int requestCode, Bundle bundle);
/**
* 返回到上個頁面
*
* @param bundle 參數
*/
public void setResultOk(Bundle bundle);
}
複製代碼
/**
* <頁面基礎公共功能抽象>
*
* @author caoyinfei
* @version [版本號, 2016/6/6]
* @see [相關類/方法]
* @since [V1]
*/
public interface PresentationLayerFunc {
/**
* 彈出消息
*
* @param msg
*/
public void showToast(String msg);
/**
* 網絡請求加載框
*/
public void showProgressDialog();
/**
* 隱藏網絡請求加載框
*/
public void hideProgressDialog();
/**
* 顯示軟鍵盤
*
* @param focusView
*/
public void showSoftKeyboard(View focusView);
/**
* 隱藏軟鍵盤
*/
public void hideSoftKeyboard();
}
複製代碼
對於上層開發而言,工做就比較簡單了,好比登陸界面(LoginActivity),只要繼承BaseActivity則能夠了,而後用IDE工具,自動導入必要的override方法。 代碼以下:
public class LoginActivity extends BaseActivity implements IUserLoginView {
/**
* 用戶名
*/
private EditText userName;
/**
* 用戶密碼
*/
private EditText password;
/**
* 登陸
*/
private Button login;
private LoginPresenter mUserLoginPresenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
setContentView(R.layout.activity_main);
super.onCreate(savedInstanceState);
presenter = mUserLoginPresenter = new LoginPresenter();
mUserLoginPresenter.attachView(this);
}
@Override
public void initViews() {
userName = (EditText) findViewById(R.id.username);
password = (EditText) findViewById(R.id.passowrd);
login = (Button) findViewById(R.id.login);
}
@Override
public void initListeners() {
login.setOnClickListener(this);
}
@Override
public void initData() {
}
@Override
public void setHeader() {
super.setHeader();
title.setText("登陸");
}
@Override
public void onEventMainThread(Event event) {
super.onEventMainThread(event);
switch (event){
case IMAGE_LOADER_SUCCESS:
clearEditContent();
break;
}
}
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.login:
//13914786934 123456 能夠登陸
mUserLoginPresenter.login(userName.getText().toString(), password.getText().toString());
break;
}
super.onClick(v);
}
@Override
public void clearEditContent() {
userName.setText("");
password.setText("");
}
@Override
public void onError(String errorMsg, String code) {
showToast(errorMsg);
}
@Override
public void onSuccess() {
startActivity(HomeActivity.class,null);
}
@Override
public void showLoading() {
}
@Override
public void hideLoading() {
}
}
複製代碼
你們應該看得出,acitivty裏面全是接口,開發gg只要把想應實現填到對應的接口中便可,這樣實現的好處有幾個: 1.每一個頁面都是這種統一的格式,後期人員流動後維護方便。 2.公共處理,好比title欄,每一個頁面都有,各個頁面去單獨實現,代碼冗餘,這邊抽到BaseActivity 裏面setHeader()方法去統一處理,當時各個子類也能夠自定義特殊格式,好比title欄上面的titleName的不一樣。 3.公共方法抽象,避免每一個activity重複大量代碼。
2)數據交互層 可能有人會看到上面的代碼中有MVP的代碼,會看不太懂?別急,接下來說解MVP的做用。 以前activity層既作界面,又作業務邏輯,代碼量特別大,動不動幾百上千行,以前項目上線的時候,領導讓我混淆一下,我當時說,這種代碼,過幾個月咱們本身都看不懂了,還須要混淆嗎?哈哈~~固然是開玩笑。 言歸正傳,咱們這邊用MVP代替了MVC,從上面activity能夠看出,activity只作兩件事:一、view的建立。二、用戶交互。那業務邏輯咱們放在哪裏呢?這裏咱們引入Presenter層,用來專門處理業務邏輯,並經過IMvpView接口實現跟activity的交互(mvp具體講解,前面已經很詳細的介紹過,地址:blog.csdn.net/dfskhgalshg…
代碼以下: 上面咱們說過,Presenter與View交互是經過接口。因此咱們這裏須要定義一個IUserLoginView ,難點就在於應該有哪些方法,咱們這個是登陸頁面,其實有哪些功能,就應該有哪些方法,好比登陸成功,失敗,彈出加載框這些都要通知ui(Activity)去更新。因此定義了以下方法:
/**
* <功能詳細描述>
*
* @author caoyinfei
* @version [版本號, 2016/5/4]
* @see [相關類/方法]
* @since [V1]
*/
public interface IMvpView {
void onError(String errorMsg, String code);
void onSuccess();
void showLoading();
void hideLoading();
}
複製代碼
/**
* <功能詳細描述>
*
* @author caoyinfei
* @version [版本號, 2016/5/4]
* @see [相關類/方法]
* @since [產品/模塊版本]
*/
public interface IUserLoginView extends IMvpView {
void clearEditContent();
}
複製代碼
LoginPresenter 爲登陸的業務實現類,他須要作兩件事:一、業務處理。2.通知頁面數據刷新。業務處理很簡單,這邊不作介紹了。Presenter與頁面交互是經過接口實現的,這邊經過繼承基類BasePresenter,從而實現接口attachView(V view),這邊的view是個泛型,在這裏,他實際上是IUserLoginView,LoginActivity會實現這個接口,在初始化LoginPresenter 的時候,會把自身傳過來mUserLoginPresenter.attachView(this);-----這段代碼是在LoginActivity的onCreate中,這樣 Presenter通知頁面刷新就只要經過接口就能夠了。
/**
* <基礎業務類>
*
* @author caoyinfei
* @version [版本號, 2016/6/6]
* @see [相關類/方法]
* @since [V1]
*/
public interface Presenter<V> {
void attachView(V view);
void detachView(V view);
}
複製代碼
/**
* <基礎業務類>
*
* @author caoyinfei
* @version [版本號, 2016/6/6]
* @see [相關類/方法]
* @since [V1]
*/
public abstract class BasePresenter<V extends IMvpView> implements Presenter<V> {
protected V mvpView;
public void attachView(V view) {
mvpView = view;
}
@Override
public void detachView(V view) {
mvpView = null;
}
}
複製代碼
/**
* <功能詳細描述>
*
* @author caoyinfei
* @version [版本號, 2016/5/4]
* @see [相關類/方法]
* @since [產品/模塊版本]
*/
public class LoginPresenter extends BasePresenter<IUserLoginView> {
public LoginPresenter() {
}
public void login(String useName, String password) {
//網絡層
mvpView.showLoading();
SecurityManager securityManager = BridgeFactory.getBridge(Bridges.SECURITY);
OkHttpManager httpManager = BridgeFactory.getBridge(Bridges.HTTP);
httpManager.requestAsyncPostByTag(URLUtil.USER_LOGIN, getName(), new ITRequestResult<LoginResp>() {
@Override
public void onCompleted() {
mvpView.hideLoading();
}
@Override
public void onSuccessful(LoginResp entity) {
mvpView.onSuccess();
EBSharedPrefManager manager = BridgeFactory.getBridge(Bridges.SHARED_PREFERENCE);
manager.getKDPreferenceUserInfo().saveString(EBSharedPrefUser.USER_NAME, "abc");
}
@Override
public void onFailure(String errorMsg) {
mvpView.onError(errorMsg, "");
}
}, LoginResp.class, new Param("username", useName),
new Param("pas", securityManager.get32MD5Str(password)));
}
}
複製代碼
public class LoginActivity extends BaseActivity implements IUserLoginView {
/**
* 用戶名
*/
private EditText userName;
/**
* 用戶密碼
*/
private EditText password;
/**
* 登陸
*/
private Button login;
private LoginPresenter mUserLoginPresenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
setContentView(R.layout.activity_main);
super.onCreate(savedInstanceState);
presenter = mUserLoginPresenter = new LoginPresenter();
mUserLoginPresenter.attachView(this);
}
@Override
public void initViews() {
userName = (EditText) findViewById(R.id.username);
password = (EditText) findViewById(R.id.passowrd);
login = (Button) findViewById(R.id.login);
}
@Override
public void initListeners() {
login.setOnClickListener(this);
}
@Override
public void initData() {
}
@Override
public void setHeader() {
super.setHeader();
title.setText("登陸");
}
@Override
public void onEventMainThread(Event event) {
super.onEventMainThread(event);
switch (event){
case IMAGE_LOADER_SUCCESS:
clearEditContent();
break;
}
}
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.login:
//13914786934 123456 能夠登陸
mUserLoginPresenter.login(userName.getText().toString(), password.getText().toString());
break;
}
super.onClick(v);
}
@Override
public void clearEditContent() {
userName.setText("");
password.setText("");
}
@Override
public void onError(String errorMsg, String code) {
showToast(errorMsg);
}
@Override
public void onSuccess() {
startActivity(HomeActivity.class,null);
}
@Override
public void showLoading() {
}
@Override
public void hideLoading() {
}
}
複製代碼
3)網絡層 網絡因爲google在6.0後再也不使用httpclient,以前項目中經過httpclient實現了網絡通訊,如今跟隨google,換成OKHttp框架。這個框架的講解再也不介紹了,比較簡單,我貼一個鴻神的博客講解地址: blog.csdn.net/lmj62356579… 我這邊作的事情是,對OKHttp再作了一層封裝,更方便咱們使用。
/**
* <http公共解析庫>
*
* @author caoyinfei
* @version [版本號, 2016/6/6]
* @see [相關類/方法]
* @since [V1]
*/
public class OkHttpUtil {
Handler handler = new Handler() {
};
private final String TAG = OkHttpUtil.class.getSimpleName();
private static OkHttpUtil manager;
private OkHttpClient mOkHttpClient;
public final int TIMEOUT = 20;
public final int WRITE_TIMEOUT = 20;
public final int READ_TIMEOUT = 20;
/**
* 請求url集合
*/
private HashMap<String, Set<String>> requestMap;
public OkHttpUtil() {
requestMap = new HashMap<String, Set<String>>();
mOkHttpClient = new OkHttpClient();
mOkHttpClient.setConnectTimeout(TIMEOUT, TimeUnit.SECONDS);
mOkHttpClient.setWriteTimeout(WRITE_TIMEOUT, TimeUnit.SECONDS);
mOkHttpClient.setReadTimeout(READ_TIMEOUT, TimeUnit.SECONDS);
}
public static OkHttpUtil getInstance() {
if (manager == null) {
synchronized (OkHttpUtil.class) {
if (manager == null) {
return new OkHttpUtil();
}
}
}
return manager;
}
/*********************************************************** get請求*********************************************************/
/**
* 異步Get請求 具體實現
*
* @param url 請求url
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
* @param <T> 泛型模板
*/
public <T> void requestAsyncGetEnqueue(String url, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
String constructUrl = constructUrl(url, params);
Request request = new Request.Builder()
.get()
.url(constructUrl)
.build();
mOkHttpClient.newCall(request).enqueue(new TRequestCallBack(iTRequestResult, clazz));
}
/**
* 異步Get請求 具體實現(可取消)
*
* @param url 請求url
* @param activityName 請求activityName
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
* @param <T> 泛型模板
*/
public <T> void requestAsyncGetEnqueueByTag(String url, String activityName, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
addRequestUrl(activityName, url);
String constructUrl = constructUrl(url, params);
Request request = new Request.Builder()
.get()
.url(constructUrl)
.tag(url)
.build();
mOkHttpClient.newCall(request).enqueue(new TRequestCallBack(iTRequestResult, clazz, activityName));
}
/**
* 構造get請求的url
*
* @param url 不帶參數的url
* @param params 參數
* @return 帶參數的url
*/
private String constructUrl(String url, Param... params) {
StringBuilder sb = new StringBuilder();
sb.append(url);
if (params.length != 0) {
sb.append("?");
} else {
return sb.toString();
}
for (Param param :
params) {
sb.append(param.key + "=" + param.value + "&");
}
return sb.toString().substring(0, sb.length() - 1);
}
/*********************************************************** post請求*********************************************************/
/**
* 異步POST請求 具體實現
*
* @param url 請求url
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
* @param <T> 泛型模板
*/
public <T> void requestAsyncPost(String url, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
FormEncodingBuilder builder = new FormEncodingBuilder();
for (Param param :
params) {
builder.add(param.key, param.value);
}
RequestBody body = builder.build();
Request request = new Request.Builder().post(body).url(url).build();
mOkHttpClient.newCall(request).enqueue(new TRequestCallBack(iTRequestResult, clazz));
}
/**
* 異步POST請求 具體實現(可取消)
*
* @param url 請求url
* @param activityName 請求activityName
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
* @param <T> 泛型模板
*/
public <T> void requestAsyncPostByTag(String url, String activityName, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
addRequestUrl(activityName, url);
FormEncodingBuilder builder = new FormEncodingBuilder();
for (Param param :
params) {
builder.add(param.key, param.value);
}
RequestBody body = builder.build();
Request request = new Request.Builder().post(body).url(url).tag(url).build();
mOkHttpClient.newCall(request).enqueue(new TRequestCallBack(iTRequestResult, clazz, activityName));
}
/**
* 異步DELETE請求 具體實現
*
* @param url 請求url
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
* @param <T> 泛型模板
*/
public <T> void requestAsyncDelete(String url, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
String finalUrl = constructUrl(url, params);
Request request = new Request.Builder()
.delete()
.url(finalUrl)
.build();
mOkHttpClient.newCall(request).enqueue(new TRequestCallBack(iTRequestResult, clazz));
}
/*********************************************************** 文件請求*********************************************************/
/**
* 異步POST請求 單文件上傳
*
* @param url 請求url
* @param file 待上傳的文件
* @param key 待上傳的key
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
*/
public <T> void requestAsyncPost(String url, File file, String key, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
MultipartBuilder builder = new MultipartBuilder().type(MultipartBuilder.FORM);
for (Param param :
params) {
builder.addFormDataPart(param.key, param.value);
}
builder = constructMultipartBuilder(builder, file, key);
RequestBody body = builder.build();
Request request = new Request.Builder().post(body).url(url).build();
mOkHttpClient.newCall(request).enqueue(new TRequestCallBack(iTRequestResult, clazz));
}
/**
* 異步POST請求 單文件上傳(可取消)
*
* @param url 請求url
* @param activityName 請求activityName
* @param file 待上傳的文件
* @param key 待上傳的key
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
*/
public <T> void requestAsyncPostByTag(String url, String activityName, File file, String key, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
addRequestUrl(activityName, url);
MultipartBuilder builder = new MultipartBuilder().type(MultipartBuilder.FORM);
for (Param param :
params) {
builder.addFormDataPart(param.key, param.value);
}
builder = constructMultipartBuilder(builder, file, key);
RequestBody body = builder.build();
Request request = new Request.Builder().post(body).url(url).tag(url).build();
mOkHttpClient.newCall(request).enqueue(new TRequestCallBack(iTRequestResult, clazz, activityName));
}
/**
* 異步POST請求 多文件上傳
*
* @param url 請求url
* @param files 待上傳的文件s
* @param keys 待上傳文件的keys
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
*/
public <T> void requestAsyncPost(String url, File[] files, String[] keys, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
MultipartBuilder builder = new MultipartBuilder().type(MultipartBuilder.FORM);
for (Param param :
params) {
builder.addFormDataPart(param.key, param.value);
}
for (int i = 0; i < files.length; i++) {
builder = constructMultipartBuilder(builder, files[i], keys[i]);
}
RequestBody body = builder.build();
Request request = new Request.Builder().post(body).url(url).build();
mOkHttpClient.newCall(request).enqueue(new TRequestCallBack(iTRequestResult, clazz));
}
/**
* 異步POST請求 多文件上傳(可取消)
*
* @param url 請求url
* @param activityName 請求activityName
* @param files 待上傳的文件s
* @param keys 待上傳文件的keys
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
*/
public <T> void requestAsyncPostByTag(String url, String activityName, File[] files, String[] keys, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
addRequestUrl(activityName, url);
MultipartBuilder builder = new MultipartBuilder().type(MultipartBuilder.FORM);
for (Param param :
params) {
builder.addFormDataPart(param.key, param.value);
}
for (int i = 0; i < files.length; i++) {
builder = constructMultipartBuilder(builder, files[i], keys[i]);
}
RequestBody body = builder.build();
Request request = new Request.Builder().post(body).url(url).tag(url).build();
mOkHttpClient.newCall(request).enqueue(new TRequestCallBack(iTRequestResult, clazz, activityName));
}
/**
* 異步POST請求 單圖片上傳上傳
*
* @param url 請求url
* @param files 待上傳圖片數組
* @param fileName 待上傳圖片名
* @param key 待上傳的key
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
*/
public <T> void requestAsyncPost(String url, byte[] files, String fileName, String key, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
MultipartBuilder builder = new MultipartBuilder().type(MultipartBuilder.FORM);
for (Param param :
params) {
builder.addFormDataPart(param.key, param.value);
}
RequestBody requestBody = RequestBody.create(MediaType.parse("image/*"), files);
builder.addFormDataPart(key, fileName, requestBody);
RequestBody body = builder.build();
Request request = new Request.Builder().post(body).url(url).build();
mOkHttpClient.newCall(request).enqueue(new TRequestCallBack(iTRequestResult, clazz));
}
/**
* 異步POST請求 單圖片上傳上傳(可取消)
*
* @param url 請求url
* @param activityName 請求activityName
* @param files 待上傳圖片數組
* @param fileName 待上傳圖片名
* @param key 待上傳的key
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
*/
public <T> void requestAsyncPostByTag(String url, String activityName, byte[] files, String fileName, String key, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
addRequestUrl(activityName, url);
MultipartBuilder builder = new MultipartBuilder().type(MultipartBuilder.FORM);
for (Param param :
params) {
builder.addFormDataPart(param.key, param.value);
}
RequestBody requestBody = RequestBody.create(MediaType.parse("image/*"), files);
builder.addFormDataPart(key, fileName, requestBody);
RequestBody body = builder.build();
Request request = new Request.Builder().post(body).url(url).tag(url).build();
mOkHttpClient.newCall(request).enqueue(new TRequestCallBack(iTRequestResult, clazz, activityName));
}
/**
* 構造多部件builer
*
* @param builder 當前實例化MultipartBuilder
* @param file 待上傳文件
* @param key 對應的參數名
* @return 構造後的MultipartBuilder
*/
private MultipartBuilder constructMultipartBuilder(MultipartBuilder builder, File file, String key) {
String name = file.getName();
RequestBody requestBody = RequestBody.create(MediaType.parse(guessMimeType(name)), file);
builder.addFormDataPart(key, name, requestBody);
return builder;
}
/**
* 獲取文件類型
*
* @param path
* @return
*/
private String guessMimeType(String path) {
FileNameMap fileNameMap = URLConnection.getFileNameMap();
String contentTypeFor = fileNameMap.getContentTypeFor(path);
if (contentTypeFor == null) {
contentTypeFor = "application/octet-stream";
}
return contentTypeFor;
}
/**
* 增長請求標誌
*
* @param activityName
* @param url
*/
private void addRequestUrl(String activityName, String url) {
if (requestMap.containsKey(activityName)) {
requestMap.get(activityName).add(url);
} else {
Set<String> urlSet = new HashSet<String>();
urlSet.add(url);
requestMap.put(activityName, urlSet);
}
}
/**
* 取消正在請求的url
*
* @param url 請求url
*/
public void cancelRequest(String url) {
try {
mOkHttpClient.getDispatcher().cancel(url);
} catch (Exception ex) {
ex.printStackTrace();
}
}
/**
* 取消當前頁面正在的請求
*
* @param activityName
*/
public void cancelActivityRequest(String activityName) {
try {
if (requestMap.containsKey(activityName)) {
Set<String> urlSet = requestMap.get(activityName);
for (String url : urlSet) {
mOkHttpClient.getDispatcher().cancel(url);
}
requestMap.remove(activityName);
}
} catch (Exception ex) {
ex.printStackTrace();
}
}
/*************************************************************
* 回調方法
*********************************************************/
class TRequestCallBack<T> implements Callback {
private ITRequestResult<T> mITRequestResult;
private Class<T> clazz;
private String notifyMsg = "";
private String activityName;
public TRequestCallBack(ITRequestResult<T> mITRequestResult, Class<T> clazz) {
this.mITRequestResult = mITRequestResult;
this.clazz = clazz;
}
public TRequestCallBack(ITRequestResult<T> mITRequestResult, Class<T> clazz, String activityName) {
this.mITRequestResult = mITRequestResult;
this.clazz = clazz;
this.activityName = activityName;
}
@Override
public void onFailure(Request request, IOException e) {
EBLog.e(TAG, request.toString() + e.toString());
if (!isHaveActivtyName(activityName)) return;
notifyMsg = NETWORK_ERROR;
postErrorMsg();
}
@Override
public void onResponse(Response response) throws IOException {
if (!isHaveActivtyName(activityName)) return;
if (response.isSuccessful()) {
String result = response.body().string(); //方法只能調用一次
EBLog.i(TAG, result);
final T res = GsonHelper.toType(result, clazz);
int code = -1;
if (res != null && res instanceof BaseResp) {
code = ((BaseResp) res).getRetcode();
switch (code) {
case 000000:
postSucessMsg(res);
break;
case 10005:
case 10011:
//自動登陸
default:
notifyMsg = ((BaseResp) res).getRetinfo();
postErrorMsg();
break;
}
} else {
notifyMsg = SERVER_ERROR;
postErrorMsg();
}
} else {
notifyMsg = NETWORK_ERROR;
postErrorMsg();
}
}
/**
* 主線程發送錯誤消息
*/
private void postErrorMsg() {
handler.post(new Runnable() {
@Override
public void run() {
mITRequestResult.onCompleted();
mITRequestResult.onFailure(notifyMsg);
}
});
}
/**
* 主線程發送正確消息
*/
private void postSucessMsg(final T res) {
handler.post(new Runnable() {
@Override
public void run() {
mITRequestResult.onCompleted();
mITRequestResult.onSuccessful(res);
}
});
}
/**
* 當前activity是否存在
*
* @param activityName
*/
private boolean isHaveActivtyName(String activityName) {
if (GeneralUtils.isNotNullOrZeroLenght(activityName)) {
return requestMap.containsKey(activityName);
} else {
return true;
}
}
}
public static String SERVER_ERROR = "請求失敗,請稍後再試";
public static String NETWORK_ERROR = "您的網絡情況不佳,請檢查網絡鏈接";
public void destory() {
manager = null;
}
}
複製代碼
/**
* <功能詳細描述>
*
* @author caoyinfei
* @version [版本號, 2016/6/8]
* @see [相關類/方法]
* @since [產品/模塊版本]
*/
public interface ITRequestResult<T> {
public void onSuccessful(T entity);
public void onFailure(String errorMsg);
}
複製代碼
/**
* <參數類>
*
* @author caoyinfei
* @version [版本號, 2016/6/8]
* @see [相關類/方法]
* @since [V1]
*/
public class Param {
public Param() {
}
public Param(String key, String value) {
this.key = key;
this.value = value != null ? value : "";
}
public Param(String key, int value) {
this.key = key;
this.value = value + "";
}
String key;
String value;
}
複製代碼
應該已經很清楚了,個人作的事情有三個: 1.定義ITRequestResult接口,用於處理網絡請求後的回調,而且此接口中的回調在主線程中(OKHttp返回接口Callback是在子線程中 )。 2.TRequestCallBack接口實現。 集中統一處理網絡層異常碼而後返回到UI層。 集中統一處理網絡層正常狀況,經過json庫,把網絡返回解析成java model返回給UI層。 3.get,post,cancel方法封裝,方便調用。
4)Bridge層抽象
每一個項目中的重複代碼特別多,不少項目喜歡抽象公共方法類,可是項目的時間一久,可能你本身都不清楚,這個方法是否認義過,寫在哪裏,勤快的人會全局搜一遍,有些同窗可能會嫌麻煩,本身新建一個util類,寫上本身的名字,頓時感受本身萌萌的。
這邊,咱們引入了BridgeFactory,用來統一管理基礎功能,相似本地服務的實現原理。 BridgeFactory裏面實現了文件,網絡,數據庫,安全等等管理類的實現,並保存了各種管理類的引用。業務層或者上層調用底層實現時,一概經過BridgeFactory去訪問,而不是直接的調用。
/**
* <中間鏈接層>
*
* @author caoyinfei
* @version [版本號, 2016/6/6]
* @see [相關類/方法]
* @since [V1]
*/
public class BridgeFactory {
private static BridgeFactory model;
private HashMap<String, Object> mBridges;
private BridgeFactory() {
mBridges = new HashMap<String, Object>();
}
public static void init(Context context) {
model = new BridgeFactory();
model.iniLocalFileStorageManager();
model.initPreferenceManager();
model.initSecurityManager();
model.initUserSession();
model.initCoreServiceManager(context);
model.initOkHttpManager();
}
public static void destroy() {
model.mBridges = null;
model = null;
}
/**
* 初始化本地存儲路徑管理類
*/
private void iniLocalFileStorageManager() {
LocalFileStorageManager localFileStorageManager = new LocalFileStorageManager();
model.mBridges.put(Bridges.LOCAL_FILE_STORAGE, localFileStorageManager);
BridgeLifeCycleSetKeeper.getInstance().trustBridgeLifeCycle(localFileStorageManager);
}
/**
* 初始化SharedPreference管理類
*/
private void initPreferenceManager() {
EBSharedPrefManager ebSharedPrefManager = new EBSharedPrefManager();
model.mBridges.put(Bridges.SHARED_PREFERENCE, ebSharedPrefManager);
BridgeLifeCycleSetKeeper.getInstance().trustBridgeLifeCycle(ebSharedPrefManager);
}
/**
* 網絡請求管理類
*/
private void initOkHttpManager() {
OkHttpManager mOkHttpManager = new OkHttpManager();
model.mBridges.put(Bridges.HTTP, mOkHttpManager);
BridgeLifeCycleSetKeeper.getInstance().trustBridgeLifeCycle(mOkHttpManager);
}
/**
* 初始化安全模塊
*/
private void initSecurityManager() {
SecurityManager securityManager = new SecurityManager();
model.mBridges.put(Bridges.SECURITY, securityManager);
BridgeLifeCycleSetKeeper.getInstance().trustBridgeLifeCycle(securityManager);
}
/**
* 初始化用戶信息模塊
*/
private void initUserSession() {
}
/**
* 初始化Tcp服務
*
* @param context
*/
private void initCoreServiceManager(Context context) {
}
private void initDBManager() {
}
/**
* 經過bridgeKey {@link Bridges}來獲取對應的Bridge模塊
*
* @param bridgeKey {@link Bridges}
* @return
*/
@SuppressWarnings("unchecked")
public static <V extends Object> V getBridge(String bridgeKey) {
final Object bridge = model.mBridges.get(bridgeKey);
if (bridge == null) {
throw new NullPointerException("-no defined bridge-");
}
return (V) bridge;
}
}
複製代碼
而且,經過BridgeLifeCycleListener 接口,實現各個底層功能管理類的統一初始化跟銷燬工做,保持跟app的生命週期一致。代碼以下:
/**
* 若是Bridge層的生命週期和App的生命週期相關(在Application
* onCreate的時候初始化,在用戶雙擊back鍵退出),則實現此接口,屆時統一初始化和銷燬
*/
public interface BridgeLifeCycleListener {
public void initOnApplicationCreate(Context context);
public void clearOnApplicationQuit();
}
複製代碼
Manager類代碼以下:
/**
* <http公共解析庫>
*
* @author caoyinfei
* @version [版本號, 2016/6/6]
* @see [相關類/方法]
* @since [V1]
*/
public class OkHttpManager implements BridgeLifeCycleListener {
@Override
public void initOnApplicationCreate(Context context) {
}
/**
* 異步Get請求 泛型返回
*
* @param url 請求url
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
* @param <T> 泛型模板
*/
public <T> void requestAsyncGet(String url, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
OkHttpUtil.getInstance().requestAsyncGetEnqueue(url, iTRequestResult, clazz, params);
}
/**
* 異步Get請求 帶tag(關閉頁面則取消請求)
*
* @param url 請求url
* @param activityName 請求activityName
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
* @param <T> 泛型模板
*/
public <T> void requestAsyncGetByTag(String url, String activityName, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
OkHttpUtil.getInstance().requestAsyncGetEnqueueByTag(url, activityName, iTRequestResult, clazz, params);
}
/**
* 異步POST請求
*
* @param url 請求url
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
* @param <T> 泛型模板
*/
public <T> void requestAsyncPost(String url, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
OkHttpUtil.getInstance().requestAsyncPost(url, iTRequestResult, clazz, params);
}
/**
* 異步POST請求 帶tag(關閉頁面則取消請求)
*
* @param url 請求url
* @param activityName 請求activityName
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
* @param <T> 泛型模板
*/
public <T> void requestAsyncPostByTag(String url, String activityName, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
OkHttpUtil.getInstance().requestAsyncPostByTag(url, activityName, iTRequestResult, clazz, params);
}
/**
* 異步DELETE請求
*
* @param url 請求url
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
* @param <T> 泛型模板
*/
public <T> void requestAsyncDelete(String url, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
OkHttpUtil.getInstance().requestAsyncDelete(url, iTRequestResult, clazz, params);
}
/**
* 異步POST請求 單文件上傳
*
* @param url 請求url
* @param file 待上傳的文件
* @param key 待上傳的key
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
*/
public <T> void requestAsyncPost(String url, File file, String key, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
OkHttpUtil.getInstance().requestAsyncPost(url, file, key, iTRequestResult, clazz, params);
}
/**
* 異步POST請求 單文件上傳 帶tag(關閉頁面則取消請求)
*
* @param url 請求url
* @param activityName 請求activityName
* @param file 待上傳的文件
* @param key 待上傳的key
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
*/
public <T> void requestAsyncPostByTag(String url, String activityName, File file, String key, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
OkHttpUtil.getInstance().requestAsyncPostByTag(url, activityName, file, key, iTRequestResult, clazz, params);
}
/**
* 異步POST請求 多文件上傳
*
* @param url 請求url
* @param files 待上傳的文件s
* @param keys 待上傳文件的keys
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
*/
public <T> void requestAsyncPost(String url, File[] files, String[] keys, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
OkHttpUtil.getInstance().requestAsyncPost(url, files, keys, iTRequestResult, clazz, params);
}
/**
* 異步POST請求 多文件上傳 帶tag(關閉頁面則取消請求)
*
* @param url 請求url
* @param activityName 請求activityName
* @param files 待上傳的文件s
* @param keys 待上傳文件的keys
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
*/
public <T> void requestAsyncPostByTag(String url, String activityName, File[] files, String[] keys, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
OkHttpUtil.getInstance().requestAsyncPostByTag(url, activityName, files, keys, iTRequestResult, clazz, params);
}
/**
* 異步POST請求 單圖片上傳上傳
*
* @param url 請求url
* @param files 待上傳圖片數組
* @param fileName 待上傳圖片名
* @param key 待上傳的key
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
*/
public <T> void requestAsyncPost(String url, byte[] files, String fileName, String key, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
OkHttpUtil.getInstance().requestAsyncPost(url, files, fileName, key, iTRequestResult, clazz, params);
}
/**
* 異步POST請求 單圖片上傳上傳 帶tag(關閉頁面則取消請求)
*
* @param url 請求url
* @param activityName 請求activityName
* @param files 待上傳圖片數組
* @param fileName 待上傳圖片名
* @param key 待上傳的key
* @param iTRequestResult 請求回調
* @param clazz Class<T>
* @param params 請求參數
*/
public <T> void requestAsyncPostByTag(String url, String activityName, byte[] files, String fileName, String key, ITRequestResult<T> iTRequestResult, Class<T> clazz, Param... params) {
OkHttpUtil.getInstance().requestAsyncPostByTag(url, activityName, files, fileName, key, iTRequestResult, clazz, params);
}
/**
* 取消正在請求的url
*
* @param url
*/
public void cancelRequest(String url) {
OkHttpUtil.getInstance().cancelRequest(url);
}
/**
* 取消當前頁面正在請求的請求
*
* @param activity
*/
public void cancelActivityRequest(String activity) {
OkHttpUtil.getInstance().cancelActivityRequest(activity);
}
@Override
public void clearOnApplicationQuit() {
OkHttpUtil.getInstance().destory();
}
}
複製代碼
5)多頁面交互 可能會有多個頁面存在邏輯關係,好比HomeActivity加載圖片成功後,要通知LoginActivity上面的EditText內容清除,固然這個需求是我瞎扯的,然而真正開發中的需求未嘗不是這樣呢。我擦,無心間流露出對產品經理的喜好~~~~。 可能會有人用廣播,用觀察者,應該還有人會定義靜態方法,去直接調用,無論怎麼,我不評價,由於我以前也都用過。。。。 咱們這邊改爲eventbus去作頁面之間的交互,eventbus的好處,相信你們都清楚。 EventBus是一款針對Android優化的發佈/訂閱事件總線。主要功能是替代Intent,Handler,BroadCast在Fragment,Activity,Service,線程之間傳遞消息.優勢是開銷小,代碼更優雅。以及將發送者和接收者解耦。
使用的代碼以下:
HomeActivity.java類
@Override
public void initData() {
Picasso.with(this).load("http://i.imgur.com/DvpvklR.png").resize(DensityUtil.dip2px(this,200), DensityUtil.dip2px(this,200)).centerCrop().into(image);
EventBus.getDefault().post(Event.IMAGE_LOADER_SUCCESS);//發送刷新通知
}
複製代碼
LoginActivity.java類
@Override
public void onEventMainThread(Event event) {
super.onEventMainThread(event);
switch (event){
//接受通知
case IMAGE_LOADER_SUCCESS:
clearEditContent();
break;
}
}
複製代碼
/**
* <事件類型>
*
* @author caoyinfei
* @version [版本號, 2016/6/6]
* @see [相關類/方法]
* @since [V1]
*/
public enum Event {
/**
* 圖片成功
*/
IMAGE_LOADER_SUCCESS,
}
複製代碼
6)其餘公共類的封裝 固然還有不少類的封裝,框架中都有涉及,這邊因爲時間問題不一一介紹了,你們能夠自行研究。
blog.csdn.net/dfskhgalshg… ......
1.Android依賴注入的框架:Dagger、RoboGuice和ButterKnife,依賴注入的框架,見仁見智,有些人很推崇,可是我我的不怎麼喜歡,首先影響了代碼結構,代碼交接成本高,我的小項目能夠嘗試使用,大的公司項目仍是在考慮。
2.該篇文章爲單工程框架,下篇文章會介紹組件化架構
項目地址:關注公衆號,留言:Android項目架構搭建源碼便可獲取。
若有錯誤歡迎指出來,一塊兒學習。