首先我們來分析為什么需要懶加載?
一個APP中可能經(jīng)常有這種布局:

通用的做法是用ViewPager嵌套Fragment,那么onCreate這個界面后,會同時加載當(dāng)前當(dāng)前fragment左右兩邊的布局,當(dāng)然也有辦法控制左邊兩邊布局的數(shù)量:
/**
* @param limit How many pages will be kept offscreen in an idle state.
*/
public void setOffscreenPageLimit(int limit) {
源碼的注釋什么意思呢,比如viewPager.setOffscreenPageLimit(2),那么會同時創(chuàng)建左右兩邊的2個fragment,如果左邊沒有則不創(chuàng)建。換句話說,滑動到第N個,會保證 N-2,N-1,N ,N+1,N+2都已加載,如果.....N-4,N-3,N+3,N+4.....已創(chuàng)建,則會被銷毀。
那么問題就來了,如果剛進(jìn)入主界面就加載3個fragment,會增加手機(jī)性能的損耗,可能用戶只想看第一個。又如果用戶對第五個fragment感興趣,那完蛋了,可能34567同時加載了,這肯定不可以接受的。以前我是這么解決,setOffscreenPageLimit(1),但是也有問題,如果用戶經(jīng)常在這幾個fragment中來回切換,那么viewpager會及時的銷毀,可能就會丟失一些用戶的操作記錄,fragment重復(fù)創(chuàng)建銷毀也會增加內(nèi)存損耗,一個優(yōu)質(zhì)的APP應(yīng)該考慮這些問題。
解決思路就是懶加載,只有用戶真正的滑動這一頁才進(jìn)行數(shù)據(jù)的加載操作,并且可以通過設(shè)置setOffscreenPageLimit的大小來控制Fragment緩存的大小,得以保存數(shù)據(jù)卻不浪費資源加載數(shù)據(jù)。
直接貼代碼,思路的啥的都寫在注釋里了:(CommonFragment是我寫的一個基類,本文就不貼了,后面會放上github的倉庫地址,可自行查看)
package com.a26c.android.frame.base;
import android.os.Bundle;
import android.view.View;
/**
* Created by guilinlin on 16/7/20 11:20.
* email 973635949@qq.com
*
* @desc
*/
public abstract class CommonLazyLoadFragment extends CommonFragment {
/**
* 標(biāo)記已加載完成,保證懶加載只能加載一次
*/
private boolean hasLoaded = false;
/**
* 標(biāo)記Fragment是否已經(jīng)onCreate
*/
private boolean isCreated = false;
/**
* 界面對于用戶是否可見
*/
private boolean isVisibleToUser = false;
private View view;
/**
* 這個方法是實現(xiàn)了基類的抽象方法,你可以把它當(dāng)成和OnCreate()方法等價
*/
@Override
public void init(View view, Bundle savedInstanceState) {
isCreated = true;//注:關(guān)鍵步驟
this.view = view;
lazyLoad(this.view, savedInstanceState);
}
/**
* 監(jiān)聽界面是否展示給用戶,實現(xiàn)懶加載
* 這個方法也是網(wǎng)上的一些方法用的最多的一個,我的思路也是這個,不過把整體思路完善了一下
*/
@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
this.isVisibleToUser = isVisibleToUser;//注:關(guān)鍵步驟
super.setUserVisibleHint(isVisibleToUser);
lazyLoad(view, null);
}
/**
* 懶加載方法,獲取數(shù)據(jù)什么的放到這邊來使用,在切換到這個界面時才進(jìn)行網(wǎng)絡(luò)請求
*/
private void lazyLoad(View view, Bundle savedInstanceState) {
//如果該界面不對用戶顯示、已經(jīng)加載、fragment還沒有創(chuàng)建,
//三種情況任意一種,不獲取數(shù)據(jù)
if (!isVisibleToUser || hasLoaded || !isCreated) {
return;
}
lazyInit(view, savedInstanceState);
hasLoaded = true;//注:關(guān)鍵步驟,確保數(shù)據(jù)只加載一次
}
/**
* 子類必須實現(xiàn)的方法,這個方法里面的操作都是需要懶加載的
*/
public abstract void lazyInit(View view, Bundle savedInstanceState);
@Override
public void onDestroyView() {
super.onDestroyView();
isCreated = false;
hasLoaded = false;
}
}
舉個栗子??:
public class TestLazyFragment extends CommonLazyLoadFragment {
/**
* 基類的抽象方法,設(shè)置Fragment引用的布局id
*/
@Override
public int getLayoutId() {
return R.layout.layout_lazy;
}
@Override
public void init(View view, Bundle savedInstanceState) {
super.init(view, savedInstanceState);
Log.i("tag", "這個方法里面走的是正常的流程");
}
@Override
public void lazyInit(View view, Bundle savedInstanceState) {
Log.i("tag", "當(dāng)且僅當(dāng)Fragment第一次顯示時,加載數(shù)據(jù)");
}
@Override
public void setEvent(View view) {
}
}