Android初級(jí)開發(fā)筆記 -- 關(guān)于Fragment的回退棧

一直想好好研究一下項(xiàng)目中經(jīng)常用到的Fragment。這篇先記錄一下對(duì)Fragment的回退棧的學(xué)習(xí)委造。
@[toc]

一昏兆、什么是Fragment

Fragment是Android3.0之后引入的可嵌入activity中的碎片化組件妇穴,實(shí)現(xiàn)了界面的最大化利用伟骨。有以下幾個(gè)特點(diǎn):
(1)不可獨(dú)立存在携狭,但是有自己的生命周期逛腿。不過因?yàn)榕cactivity關(guān)聯(lián)在一起单默,生命周期會(huì)受activity影響搁廓。
(2) 可靜態(tài)引入也可動(dòng)態(tài)加載境蜕。推薦使用support-v4中的android.support.v4.app.Fragment而非系統(tǒng)內(nèi)置的android.app.Fragment粱年。因前者能讓Fragment在所有Android系統(tǒng)版本中保持功能一致性台诗。

二拉队、Fragment的生命周期

  • 先po一張經(jīng)典生命周期圖:


    在這里插入圖片描述

    我們可以看到有幾個(gè)關(guān)鍵的回調(diào)方法粱快。

onAttach()
Fragment和Activity建立關(guān)聯(lián)的時(shí)候調(diào)用(獲得activity的傳遞的值)

onCreateView()
為Fragment創(chuàng)建視圖(加載布局)時(shí)調(diào)用(給當(dāng)前的fragment繪制UI布局)

onActivityCreated()
當(dāng)Activity中的onCreate方法執(zhí)行完后調(diào)用(表示activity執(zhí)行oncreate方法完成了的時(shí)候會(huì)調(diào)用此方法)

onDestroyView()
Fragment中的布局被移除時(shí)調(diào)用(表示fragment銷毀相關(guān)聯(lián)的UI布局)

onDetach()
Fragment和Activity解除關(guān)聯(lián)的時(shí)候調(diào)用(脫離activity)

  • 不同情況下的方法回調(diào)

當(dāng)一個(gè)fragment第一次被加載到屏幕上的時(shí)候呐舔,會(huì)依次執(zhí)行:

onAttach()
onCreate()
onCreateView()
onActivityCreated()

接著珊拼,當(dāng)這個(gè)fragment對(duì)用戶可見的時(shí)候澎现,會(huì)依次執(zhí)行:

onStart()
onResume()

這個(gè)時(shí)候剑辫,如果該fragment進(jìn)入了停止?fàn)顟B(tài)(“進(jìn)入后臺(tái)模式”)妹蔽,會(huì)依次執(zhí)行:
onPause()
onStop()

若這個(gè)fragment被銷毀了(或者和ta關(guān)聯(lián)的activity被銷毀了),在執(zhí)行了上面兩個(gè)方法之后緊跟著會(huì)執(zhí)行:
onDestroyView()
onDestroy()
onDetach()

此時(shí)該fragment被銷毀并且與activity解除了關(guān)聯(lián)胳岂。

  • 前面我們說到乳丰,fragment的生命周期受到ta關(guān)聯(lián)的activity的生命周期的影響~
    影響有多大呢产园?看圖說話~


    在這里插入圖片描述

    好了什燕,那么如果該fragment沒有被銷毀呢秋冰?當(dāng)ta又重新回到了運(yùn)行狀態(tài)婶熬,會(huì)依次執(zhí)行:

onCreateView()
onActivityCreated()
onStart()
onResume()
因?yàn)闆]有被銷毀赵颅,所以onCreate()不會(huì)被調(diào)用饺谬。

額那什么情況下fragment沒有被銷毀呢?這就和fragment的回退棧有關(guān)啦~

三族展、實(shí)例講述Fragment回退棧

我們知道Activity是以棧的方式進(jìn)行管理的仪缸,F(xiàn)ragment也有類似的方式恰画。

Fragment的回退棧---是用來保存每一次Fragment事務(wù)發(fā)生的變化 如果你將Fragment任務(wù)添加到回退棧拴还,當(dāng)用戶點(diǎn)擊后退按鈕時(shí)片林,將看到上一次的保存的Fragment拇厢。一旦Fragment完全從后退棧中彈出孝偎,用戶再次點(diǎn)擊后退鍵衣盾,則退出當(dāng)前Activity势决。

首先我們先認(rèn)識(shí)下這個(gè)方法:FragmentTransaction.addToBackStack(String)【把當(dāng)前事務(wù)的變化情況添加到回退棧果复,一般傳入null即可】

接下來我們用一個(gè)例子來證明一下其起到的作用以及生命周期是否真的有所不同虽抄。

MainActivity的布局文件

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
    xmlns:tools="http://schemas.android.com/tools"  
    android:layout_width="match_parent"  
    android:layout_height="match_parent" >  

    <FrameLayout  
        android:id="@+id/id_content"  
        android:layout_width="match_parent"  
        android:layout_height="match_parent" >  
    </FrameLayout>  

</RelativeLayout>

MainActivity.java文件

public class MainActivity extends Activity {  

    protected void onCreate(Bundle savedInstanceState){  
        super.onCreate(savedInstanceState);  
        requestWindowFeature(Window.FEATURE_NO_TITLE);  
        setContentView(R.layout.activity_main);  

        FragmentManager fm = getFragmentManager();  
        FragmentTransaction tx = fm.beginTransaction();  
        tx.add(R.id.id_content, new FragmentOne(),"ONE");  
        tx.commit();  
    }  
}

FragmentOne.class文件

public class FragmentOne extends Fragment implements OnClickListener {  

    private Button mBtn;  

    @Override  
    public View onCreateView(LayoutInflater inflater, ViewGroup container,  
            Bundle savedInstanceState) {  
        View view = inflater.inflate(R.layout.fragment_one, container, false);  
        mBtn = (Button) view.findViewById(R.id.bn_fragment_one);  
        mBtn.setOnClickListener(this);  
        Log.e("onCreateView", "one");
        return view;  
    }  

    @Override  
    public void onClick(View v) {  
        FragmentTwo fTwo = new FragmentTwo();  
        FragmentManager fm = getFragmentManager();  
        FragmentTransaction tx = fm.beginTransaction();  
        tx.replace(R.id.id_content, fTwo, "TWO");  
        tx.addToBackStack(null); 
        tx.commit();  
    }  
    @Override
    public void onAttach(Context context) {
        super.onAttach(context);
        Log.e("onAttach", "one");
    }

    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Log.e("onCreate", "one");
    }


    @Override
    public void onActivityCreated(@Nullable Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
        Log.e("onActivityCreated", "one");
    }

    @Override
    public void onStart() {
        super.onStart();
        Log.e("onStart", "one");
    }

    @Override
    public void onResume() {
        super.onResume();
        Log.e("onResume", "one");
    }

    @Override
    public void onPause() {
        super.onPause();
        Log.e("onPause", "one");
    }

    @Override
    public void onStop() {
        super.onStop();
        Log.e("onStop", "one");
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        Log.e("onDestroy", "one");
    }

    @Override
    public void onDetach() {
        super.onDetach();
        Log.e("onDetach", "one");
    }

    @Override
    public void onDestroyView() {
        super.onDestroyView();
        Log.e("onDestroyView", "one");
    }

FragmentTwo.class文件

public class FragmentTwo extends Fragment implements OnClickListener {  

    private Button mBtn ;  

    @Override  
    public View onCreateView(LayoutInflater inflater, ViewGroup container,  
            Bundle savedInstanceState) {  
        View view = inflater.inflate(R.layout.fragment_two, container, false);  
        mBtn = (Button) view.findViewById(R.id.bn_fragment_two);  
        mBtn.setOnClickListener(this);  
        Log.e("onCreateView", "two");
        return view ;   
    }  

    @Override  
    public void onClick(View v)  {  
        FragmentThree fThree = new FragmentThree();  
        FragmentManager fm = getFragmentManager();  
        FragmentTransaction tx = fm.beginTransaction();  
        tx.hide(this);  
        tx.add(R.id.id_content , fThree, "THREE");  
        //tx.replace(R.id.id_content, fThree, "THREE");  
        tx.addToBackStack(null);  
        tx.commit();  
    }  

    @Override
    public void onAttach(Context context) {
        super.onAttach(context);
        Log.e("onAttach", "two");
    }

    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Log.e("onCreate", "two");
    }

    @Override
    public void onActivityCreated(@Nullable Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
        Log.e("onActivityCreated", "two");
    }

    @Override
    public void onStart() {
        super.onStart();
        Log.e("onStart", "two");
    }

    @Override
    public void onResume() {
        super.onResume();
        Log.e("onResume", "two");
    }

    @Override
    public void onPause() {
        super.onPause();
        Log.e("onPause", "two");
    }

    @Override
    public void onStop() {
        super.onStop();
        Log.e("onStop", "two");
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        Log.e("onDestroy", "two");
    }

    @Override
    public void onDetach() {
        super.onDetach();
        Log.e("onDetach", "two");
    }

    @Override
    public void onDestroyView() {
        super.onDestroyView();
        Log.e("onDestroyView", "two");
    }

FragmentThree.class文件

public class FragmentThree extends Fragment implements OnClickListener {  

    private Button mBtn;  

    @Override  
    public View onCreateView(LayoutInflater inflater, ViewGroup container,  
            Bundle savedInstanceState) {  
        View view = inflater.inflate(R.layout.fragment_three, container, false);  
        mBtn = (Button) view.findViewById(R.id.bn_fragment_three);  
        mBtn.setOnClickListener(this);  
        Log.e("onCreateView", "three");
        return view;  
    }  

    @Override  
    public void onClick(View v) {  
        Toast.makeText(getActivity(), " i am a btn in Fragment three",  
                Toast.LENGTH_SHORT).show();  
    }  
    
    @Override
    public void onAttach(Context context) {
        super.onAttach(context);
        Log.e("onAttach", "three");
    }

    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Log.e("onCreate", "three");
    }

    @Override
    public void onActivityCreated(@Nullable Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
        Log.e("onActivityCreated", "three");
    }

    @Override
    public void onStart() {
        super.onStart();
        Log.e("onStart", "three");
    }

    @Override
    public void onResume() {
        super.onResume();
        Log.e("onResume", "three");
    }

    @Override
    public void onPause() {
        super.onPause();
        Log.e("onPause", "three");
    }

    @Override
    public void onStop() {
        super.onStop();
        Log.e("onStop", "three");
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        Log.e("onDestroy", "three");
    }

    @Override
    public void onDetach() {
        super.onDetach();
        Log.e("onDetach", "three");
    }

    @Override
    public void onDestroyView() {
        super.onDestroyView();
        Log.e("onDestroyView", "three");
    }

R.layout.fragment_one文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <EditText
        android:id="@+id/et_fragment_one"
        android:layout_width="match_parent"
        android:text="myself"
        android:layout_height="wrap_content" />

    <Button
        android:id="@+id/bn_fragment_one"
        android:layout_width="wrap_content"
        android:text="Button in one"
        android:layout_height="wrap_content" />
</LinearLayout>

這個(gè)過程中所調(diào)用的回調(diào)方法湖员,Log打印如下:


在這里插入圖片描述

在從FragmentOne跳轉(zhuǎn)到FragmentTwo的時(shí)候娘摔,代碼如下:

 @Override
    public void onClick(View v) {
        FragmentTwo fTwo = new FragmentTwo();
        FragmentManager fm = getFragmentManager();
        FragmentTransaction tx = fm.beginTransaction();
        tx.replace(R.id.id_content, fTwo, "TWO");
        tx.addToBackStack(null);
        tx.commit();
    }

此時(shí)FragmentOne被替換【replace】晰筛,添加到回退棧中【addToBackStack】读第,onPause()怜瞒,onStop(),onDestroyView()被調(diào)用,但是onDestroy(),onDetach()就沒被調(diào)用惠窄,該Fragment實(shí)例被保存了下來漾橙。Log:


在這里插入圖片描述

在從FragmentTwo跳轉(zhuǎn)到FragmentThree的時(shí)候


在這里插入圖片描述

第一次返回,由于FragmentThree并沒有被保存到回退棧中脾歇,所以會(huì)調(diào)用到onDestroy(),onDetach()方法
在這里插入圖片描述

第二次返回,從FragmentTwo返回到FragmentOne藕各,由于FragmentOne實(shí)例仍在焦除,所以沒有調(diào)用onCreate()


在這里插入圖片描述

第三次返回膘魄,回到桌面
在這里插入圖片描述

有沒有發(fā)現(xiàn)黔帕,在從FragmentTwo跳轉(zhuǎn)到FragmentThree的以及重新回到FragmentTwo的時(shí)候蹈丸,并沒有調(diào)用到FragmentTwo任何回調(diào)方法呐芥?是的沒有錯(cuò)就是這樣思瘟。
因?yàn)閺腇ragmentTwo到FragmentThree的代碼是這樣寫的:
@Override
    public void onClick(View v)  {
        FragmentThree fThree = new FragmentThree();
        FragmentManager fm = getFragmentManager();
        FragmentTransaction tx = fm.beginTransaction();
        tx.hide(this);
        tx.add(R.id.id_content , fThree, "THREE");
        //tx.replace(R.id.id_content, fThree, "THREE");
        tx.addToBackStack(null);
        tx.commit();
    }

調(diào)用的是hide()滨攻,而非replace()

區(qū)別

hide()光绕,show():調(diào)用了Fragment創(chuàng)建及到前臺(tái)的幾個(gè)回調(diào)方法后畜份,該Fragment在后臺(tái)或者重新回到前臺(tái)的時(shí)候爆雹,不會(huì)調(diào)用到相關(guān)生命周期回調(diào)方法,所以視圖不會(huì)重繪

replace():相當(dāng)于remove和add的合體慧起,視圖重繪

總結(jié)

1蚓挤、replace屈尼,加回退棧 --- Fragment不銷毀,但是切換時(shí)會(huì)銷毀視圖和重新創(chuàng)建視圖

2、replace, 不加回退棧 --- Fragment銷毀

3演熟、hide芒粹、show --- Fragment不銷毀化漆,也不銷毀視圖钦奋。隱藏和顯示不走生命周期

四疙赠、結(jié)語

未完待續(xù)~畢竟Fragment還有很多要學(xué)的東西

然后還是那句老話圃阳,希望各位看官不吝賜教~蟹蟹啦

內(nèi)推信息

  • 我們正在招募小伙伴捍岳,有興趣的小伙伴可以把簡(jiǎn)歷發(fā)到 app@talkmoney.cn睬隶,備注:來自簡(jiǎn)書社區(qū)
  • 詳情可以戳這里--> 廣州蘆葦信息科技
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末苏潜,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子砖顷,更是在濱河造成了極大的恐慌滤蝠,老刑警劉巖授嘀,帶你破解...
    沈念sama閱讀 206,968評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件览闰,死亡現(xiàn)場(chǎng)離奇詭異巷折,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)油吭,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來推穷,“玉大人,你說我怎么就攤上這事蟹腾×爰眩” “怎么了?”我有些...
    開封第一講書人閱讀 153,220評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵述寡,是天一觀的道長鲫凶。 經(jīng)常有香客問我螟炫,道長昼钻,這世上最難降的妖魔是什么然评? 我笑而不...
    開封第一講書人閱讀 55,416評(píng)論 1 279
  • 正文 為了忘掉前任碗淌,我火速辦了婚禮亿眠,結(jié)果婚禮上磅废,老公的妹妹穿的比我還像新娘。我一直安慰自己爹耗,他們只是感情好谜喊,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,425評(píng)論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著诵次,像睡著了一般。 火紅的嫁衣襯著肌膚如雪铸本。 梳的紋絲不亂的頭發(fā)上箱玷,一...
    開封第一講書人閱讀 49,144評(píng)論 1 285
  • 那天锡足,我揣著相機(jī)與錄音舶得,去河邊找鬼沐批。 笑死九孩,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的颖对。 我是一名探鬼主播捻撑,決...
    沈念sama閱讀 38,432評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼缤底!你這毒婦竟也來了顾患?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,088評(píng)論 0 261
  • 序言:老撾萬榮一對(duì)情侶失蹤个唧,失蹤者是張志新(化名)和其女友劉穎江解,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體徙歼,經(jīng)...
    沈念sama閱讀 43,586評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡犁河,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,028評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了桨螺。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,137評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡哄褒,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出链嘀,到底是詐尸還是另有隱情窃肠,我是刑警寧澤碧囊,帶...
    沈念sama閱讀 33,783評(píng)論 4 324
  • 正文 年R本政府宣布熄驼,位于F島的核電站,受9級(jí)特大地震影響祭芦,放射性物質(zhì)發(fā)生泄漏轴或。R本人自食惡果不足惜避矢,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,343評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望碍庵。 院中可真熱鬧,春花似錦、人聲如沸得问。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽叉信。三九已至覆享,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間跪另,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評(píng)論 1 262
  • 我被黑心中介騙來泰國打工迹卢, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,595評(píng)論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像业栅,于是被迫代替她去往敵國和親帮孔。 傳聞我的和親對(duì)象是個(gè)殘疾皇子焕檬,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,901評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容