mindLine
本節(jié)課用到的樣例在我的github歡迎關(guān)注??
我們?yōu)槭裁匆肍ragment
簡單的講帚湘,F(xiàn)ragment有一些類似于activity的控件垄分,它主要有以下幾個優(yōu)點:
-
便于各種屏幕的適應(yīng)鹅髓,可以為不同屏幕大小的設(shè)備創(chuàng)建動態(tài)靈活的UI(小屏幕手機到大屏幕平板電腦),極大提高了UI的靈活性今穿。
- 利于代碼復(fù)用,每個frgment都是獨立的怎诫,可以被多個activity公用瘾晃。
1.Fragment的生命周期
onAttach()
作用:fragment已經(jīng)關(guān)聯(lián)到activity,
這個是 回調(diào)函數(shù)
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
Log.i("onAttach_Fragment");
}
這個時候 activity已經(jīng)傳進來了
獲得activity的傳遞的值
就可以進行 與activity的通信里
當然也可以使用getActivity(),前提是這個fragment已經(jīng)和宿主的activity關(guān)聯(lián)刽虹,并且沒有脫離
他只調(diào)用一次酗捌。
onCreate()
系統(tǒng)創(chuàng)建fragment的時候回調(diào)他呢诬,在他里面實例化一些變量
這些個變量主要是:當你 暫停 停止的時候 你想保持的數(shù)據(jù)
如果我們要為fragment啟動一個后臺線程涌哲,可以考慮將代碼放于此處。
參數(shù)是:Bundle savedInstance, 用于保存 Fragment 參數(shù), Fragement 也可以 重寫 onSaveInstanceState(BundleoutState) 方法, 保存Fragement狀態(tài);
可以用于 文件保護
他只調(diào)用一次尚镰。
onCreateView()
第一次使用的時候 fragment會在這上面畫一個layout出來阀圾,
為了可以畫控件 要返回一個 布局的view,也可以返回null
當系統(tǒng)用到fragment的時候 fragment就要返回他的view狗唉,越快越好初烘,所以盡量在這里不要做耗時操作,比如從數(shù)據(jù)庫加載大量數(shù)據(jù)顯示listview分俯,當然線程還是可以的肾筐。
給當前的fragment繪制ui布局,可以使用線程更新UI
說白了就是加載fragment的布局的缸剪。
這里一般都先判斷是否為null
if(text==null){
Bundle args=getArguments();
text=args.getString("text");
}
if (view == null) {
view = inflater.inflate(R.layout.hello, null);
}
這樣進行各判斷省得每次都要加載吗铐,減少資源消耗
onActivityCreated()
當Activity中的onCreate方法執(zhí)行完后調(diào)用。
注意了:
從這句官方的話可以看出:當執(zhí)行onActivityCreated()的時候 activity的onCreate才剛完成杏节。所以在onActivityCreated()調(diào)用之前 activity的onCreate可能還沒有完成唬渗,所以不能再onCreateView()中進行 與activity有交互的UI操作,UI交互操作可以砸onActivityCreated()里面進行奋渔。
所以呢:這個方法主要是初始化那些你需要你的父Activity或者Fragment的UI已經(jīng)被完整初始化才能初始化的元素镊逝。
如果在onCreateView里面初始化空間 會慢很多,比如listview等
onStart()
和activity一致 啟動, Fragement 啟動時回調(diào), 此時Fragement可見;
onResume()
和activity一致 在activity中運行是可見的
激活, Fragement 進入前臺, 可獲取焦點時激活;
onPause()
和activity一致 其他的activity獲得焦點嫉鲸,這個仍然可見
第一次調(diào)用的時候撑蒜,指的是 用戶 離開這個fragment(并不是被銷毀)
通常用于 用戶的提交(可能用戶離開后不會回來了)
onStop()
和activity一致
fragment不可見的, 可能情況:activity被stopped了 OR fragment被移除但被加入到回退棧中
一個stopped的fragment仍然是活著的如果長時間不用也會被移除
onDestroyView()
Fragment中的布局被移除時調(diào)用玄渗。
表示fragemnt銷毀相關(guān)聯(lián)的UI布局
清除所有跟視圖相關(guān)的資源
以前以為這里沒什么用處其實 大有文章可做座菠,
相信大家都用過ViewPager+Fragment,由于ViewPager的緩存機制捻爷,每次都會加載3頁辈灼。
例如:有四個 fragment 當滑動到第四頁的時候 第一頁執(zhí)行onDestroyView(),但沒有執(zhí)行onDestroy。他依然和activity關(guān)聯(lián)也榄。當在滑動到第一頁的時候又執(zhí)行了 onCreateView()巡莹。 生命周期可以自己試一下司志。
那么問題來了。會出現(xiàn)重復(fù)加載view的局面降宅,所以這么做(下面是先人的代碼)
@Override
public void onDestroyView() {
Log.i("onDestroyView_Fragment");
if(view!=null){
((ViewGroup)view.getParent()).removeView(view);
}
super.onDestroyView();
}
onDestroy()
銷毀fragment對象
跟activity類似了骂远。
onDetach()
Fragment和Activity解除關(guān)聯(lián)的時候調(diào)用。
脫離activity
Fragment實現(xiàn)方式
靜態(tài)方法
以repair為例
動態(tài)方法
以新人項目為例
Fragment和Activity之間的區(qū)別
- 生命周期不同
- 得到context的方法不同 Fragment需要調(diào)用getActivity() Activity只需要直接Activity.this就可以了.
- 在使用butterknife 的時候 在Activity中通過 Retrofit.bind(this);
fragment通過Retrofit.bind(this,view);
Fragment與 Activity之間的通信
以repair為例
Fragment -> Activity
- 在Fragment中創(chuàng)建Interface
/**
* Fragment與Activity通信的接口
* 用于傳遞輸入的電話號碼
*/
public interface InputMobileListener {
//傳入電話號碼給Activity
void inputPhoneNumberNextAction(String mobile);
}
這里在Fragment中定義了一個接口腰根,在接口中寫了一個方法以保存手機號的字符串為參數(shù)激才,
- 在Fragment所依賴的Activity中實現(xiàn)接口并重寫方法
public class ForgetPwdActivity implements InputMobileFragment.InputMobileListener{
······
@Override
public void inputPhoneNumberNextAction(final String mobile) {
//替換Fragment
replaceFragment(R.id.frame_layout, checkCodeFragment, null);
//請求后臺為該號碼發(fā)送驗證碼
sendVerifyCode(mobile);
}
}
在Activity中重寫該方法,在方法體中接收到手機號并進行一系列操作额嘿,此處是發(fā)送驗證碼
- 在Fragment中重寫回調(diào)方法
Fragment與Activity一樣瘸恼,在創(chuàng)建和銷毀時都會回調(diào)一系列回調(diào)方法,而onAttach方法是在Fragment創(chuàng)建之初回調(diào)的册养,所以我們重寫onAttach方法便可以保證Fragment全程皆能調(diào)用接口中的方法东帅。
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
//抓取異常,在父Activity未實現(xiàn)接口時拋出球拦。
try {
//Java的多態(tài)特性之一靠闭,父類引用子類對象
InputMobileListener inputMobileListener = (InputMobileListener) activity;
} catch (ClassCastException e) {
throw new ClassCastException(activity.toString() + getString(R.string.implementSPN));
}
}
- 在Fragment中通過接口引用Activity對象調(diào)用接口方法
public void onButtonClicked(){
//獲取手機號輸入框中的手機號
String mobile = mobileEdit.getText().toString().trim();
//此處應(yīng)對獲取的手機號做正則匹配,無關(guān)代碼便不列出了
//調(diào)用重寫后的接口方法
inputMobileListener = (InputMobileListener) activity;
}
Activity -> Fragment
public static class MainActivity extends Activity
implements HeadlinesFragment.OnHeadlineSelectedListener{
...
public void onArticleSelected(int position) {
// 用戶選中HeadlinesFragment中的頭標題后
// 做一些必要的業(yè)務(wù)操作
ArticleFragment articleFrag = (ArticleFragment)
getSupportFragmentManager().findFragmentById(R.id.article_fragment);
if (articleFrag != null) {
// 如果 article frag 不為空坎炼,那么我們在同時顯示兩個fragmnet的布局中...
// 調(diào)用ArticleFragment中的方法去更新它的內(nèi)容
articleFrag.updateArticleView(position);
} else {
// 否則愧膀,我們就是在僅包含一個fragment的布局中并需要交換fragment...
// 創(chuàng)建fragment并給他一個跟選中的文章有關(guān)的參數(shù)
ArticleFragment newFragment = new ArticleFragment();
Bundle args = new Bundle();
args.putInt(ArticleFragment.ARG_POSITION, position);
newFragment.setArguments(args);
FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
// 用這個fragment替換任何在fragment_container中的東西
// 并添加事務(wù)到back stack中以便用戶可以回退到之前的狀態(tài)
transaction.replace(R.id.fragment_container, newFragment);
transaction.addToBackStack(null);
// 提交事務(wù)
transaction.commit();
}
}
}
Fragment與Fragment之間的通信
就是通過Fragment1 -> Activity -> Fragment2的方式進行的
FragmentManager
manager的概念
用法
- Activity需要繼承FragmentActivity(直接繼承 AppCompatActivity 也可以)
- 給PagerAdapter傳入FragmentManager
WaterfallPagerAdapter pagerAdapter = new WaterfallPagerAdapter(getSupportFragmentManager());
viewPager 和 tabLayout
ViewPager
java
xml
FragmentPagerAdapter
數(shù)據(jù)域
- FragmentList
- titleList
方法
- add()
- getItem(int)重載
- getCount()重載
- getPageTitile()重載