从commit一步步带你走向fragment的生命周期

写在前面

Fragment 是android开发中最常用的组件之一,用了好几年,我都不知道Fragment到底是个什么东西,Activity加载Fragment的原理是怎样的,为什么官方会叫它为碎片?直到前段时间因为工作需要,从头看来一遍Fragment的源代码,然后就有了本文。</br>
本文将从commit开始一步步带你走向Fragmnt的生命周期!!

经典的Frgment加载

从最经典的Activity加载Fragment的流程说起,如下所示,是Activity加载Frgment的例子代码。

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="horizontal"
    android:layout_width="match_parent" android:layout_height="match_parent">
    <FrameLayout android:id="@+id/content" android:layout_weight="1"
            android:layout_width="0px" android:layout_height="match_parent"
            android:background="?android:attr/detailsElementBackground" />

</LinearLayout>
public static class DetailsActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        if (savedInstanceState == null) {
            // During initial setup, plug in the details fragment.
            DetailsFragment details = new DetailsFragment();
            details.setArguments(getIntent().getExtras());
            getFragmentManager().beginTransaction().replace(R.id.content, details).commit();
        }
    }
}

Activity是通过getFragmentManager().beginTransaction()来进行加载Fragment的,我们就从这句话开始吧!</br>
fragment状态的切换时通过beginTransaction()方法来实现的,而beginTransaction()是FragmentManager的抽象方法,而FragmentManager真正的实现是FragamentManager的内部类FragmentManagerImpl,因此在FragmentManagerImpl中查找beginTransaction方法,最终方法发现beginTransaction真正生成的实例是BackStackRecord。</br>

@Override
public FragmentTransaction beginTransaction() {
  return new BackStackRecord(this);
}

由此可以知道replace,add,hide,show等等对Fragment状态的操作的真正实现都是在BackStackRecord的对应方法中实现,再看BackStackRecord的replace、add方法最终都会跳转到doAddOP

private void doAddOp(int containerViewId, Fragment fragment, String tag, int opcmd) {
       fragment.mFragmentManager = mManager;

       if (tag != null) {
           if (fragment.mTag != null && !tag.equals(fragment.mTag)) {
               throw new IllegalStateException("Can't change tag of fragment "
                       + fragment + ": was " + fragment.mTag
                       + " now " + tag);
           }
           fragment.mTag = tag;
       }

       if (containerViewId != 0) {
           if (fragment.mFragmentId != 0 && fragment.mFragmentId != containerViewId) {
               throw new IllegalStateException("Can't change container ID of fragment "
                       + fragment + ": was " + fragment.mFragmentId
                       + " now " + containerViewId);
           }
           fragment.mContainerId = fragment.mFragmentId = containerViewId;
       }

       Op op = new Op();
       op.cmd = opcmd;
       op.fragment = fragment;
       addOp(op);
   }

上面的代码中最核心的部分是最后的四行,可以看到内次replace都会new一个Op,那么OP到底是个什么东西?

static final class Op {
    Op next;
    Op prev;
    int cmd;
    Fragment fragment;
    int enterAnim;
    int exitAnim;
    int popEnterAnim;
    int popExitAnim;
    ArrayList<Fragment> removed;
}

这是OP的源代码,可以看到OP有next和prev的字段,这两个字段的属性都是Op本身,并且它本身还有一些自带的属性,到现在依然不知道这个Op到底是什么,继续看addOp方法

void addOp(Op op) {
   if (mHead == null) {
       mHead = mTail = op;
   } else {
       op.prev = mTail;
       mTail.next = op;
       mTail = op;
   }
   op.enterAnim = mEnterAnim;
   op.exitAnim = mExitAnim;
   op.popEnterAnim = mPopEnterAnim;
   op.popExitAnim = mPopExitAnim;
   mNumOp++;
}

从if-else语句中,再结合Op的结构,可以看出Op是一个双向链表,而addOp方法的作用是将replce生成的Op添加到当前链表中,到现在我们已经知道Op是个双向链表了,如下图所示:

双向链表图

那么Op到底有什么用,从目前看到的源码来看我们还是不知道。继续看BackStackRecord的show、remove等方法。

public FragmentTransaction remove(Fragment fragment) {
    Op op = new Op();
    op.cmd = OP_REMOVE;
    op.fragment = fragment;
    addOp(op);

    return this;
}

public FragmentTransaction hide(Fragment fragment) {
    Op op = new Op();
    op.cmd = OP_HIDE;
    op.fragment = fragment;
    addOp(op);

    return this;
}

public FragmentTransaction show(Fragment fragment) {
    Op op = new Op();
    op.cmd = OP_SHOW;
    op.fragment = fragment;
    addOp(op);

    return this;
}

从上面的代码看出,无论是replcae、add还是remove、hide、show操作,都会new一个Op并加到链表中,而每次添加链表时,Op的cmd和fragment都不一样,因此可以猜测Op链表实质上是操作fragment的命令链表,而执行命令的操作由commit等方法来完成的</br>
继续看BackStackRecord的commit方法,在前面我们猜测fragment状态的控制是由这个方法完成的。

public int commit() {
    return commitInternal(false);
}

public int commitAllowingStateLoss() {
    return commitInternal(true);
}

int commitInternal(boolean allowStateLoss) {
    if (mCommitted) throw new IllegalStateException("commit already called");
    if (FragmentManagerImpl.DEBUG) {
        Log.v(TAG, "Commit: " + this);
        LogWriter logw = new LogWriter(TAG);
        PrintWriter pw = new PrintWriter(logw);
        dump("  ", null, pw, null);
    }
    mCommitted = true;
    if (mAddToBackStack) {
        mIndex = mManager.allocBackStackIndex(this);
    } else {
        mIndex = -1;
    }
    mManager.enqueueAction(this, allowStateLoss);
    return mIndex;
}

上面是commit的源代码,核心代码是倒数第二行,看到命令的执行最终还是有mManager控制,但是真的是这样吗?mManager到底是在哪个地方初始化的,还记得FragmentManagerImpl的beginTransaction方法吗?</br>
跳转到FragmentManagerImpl中,继续看enqueueAction

/**
 * Adds an action to the queue of pending actions.
 *
 * @param action the action to add
 * @param allowStateLoss whether to allow loss of state information
 * @throws IllegalStateException if the activity has been destroyed
 */
public void enqueueAction(Runnable action, boolean allowStateLoss) {
    if (!allowStateLoss) {
        checkStateLoss();
    }
    synchronized (this) {
        if (mDestroyed || mHost == null) {
            throw new IllegalStateException("Activity has been destroyed");
        }
        if (mPendingActions == null) {
            mPendingActions = new ArrayList<Runnable>();
        }
        mPendingActions.add(action);
        if (mPendingActions.size() == 1) {
            mHost.getHandler().removeCallbacks(mExecCommit);
            mHost.getHandler().post(mExecCommit);
        }
    }
}

从这个方法可以看出,真正执行执行Op命令的是mExecCommit这个线程,继续看mExecCommit线程。

Runnable mExecCommit = new Runnable() {
    @Override
    public void run() {
        execPendingActions();
    }
};
/**
* Only call from main thread!
*/
public boolean execPendingActions() {
   if (mExecutingActions) {
       throw new IllegalStateException("Recursive entry to executePendingTransactions");
   }

   if (Looper.myLooper() != mHost.getHandler().getLooper()) {
       throw new IllegalStateException("Must be called from main thread of process");
   }

   boolean didSomething = false;

   while (true) {
       int numActions;

       synchronized (this) {
           if (mPendingActions == null || mPendingActions.size() == 0) {
               break;
           }

           numActions = mPendingActions.size();
           if (mTmpActions == null || mTmpActions.length < numActions) {
               mTmpActions = new Runnable[numActions];
           }
           mPendingActions.toArray(mTmpActions);
           mPendingActions.clear();
           mHost.getHandler().removeCallbacks(mExecCommit);
       }

       mExecutingActions = true;
       for (int i=0; i<numActions; i++) {
           mTmpActions[i].run();
           mTmpActions[i] = null;
       }
       mExecutingActions = false;
       didSomething = true;
   }

   if (mHavePendingDeferredStart) {
       boolean loadersRunning = false;
       for (int i=0; i<mActive.size(); i++) {
           Fragment f = mActive.get(i);
           if (f != null && f.mLoaderManager != null) {
               loadersRunning |= f.mLoaderManager.hasRunningLoaders();
           }
       }
       if (!loadersRunning) {
           mHavePendingDeferredStart = false;
           startPendingDeferredFragments();
       }
   }
   return didSomething;
}

这是mExecCommit线程真正执行的方法,但是还是看不到任何关于Fragment被操作的痕迹,也看不到任何的Op命令,继续看execPendingActions方法,在这个方法中,有个代码片段吸引了我的注意

for (int i=0; i<numActions; i++) {
     mTmpActions[i].run();
     mTmpActions[i] = null;
 }
``
在这我看到了run方法!!!而mTmpActions是mPendingActions的数组化,在`enqueueAction`中,mPendingActions是加载的是BackStackRecord!!!!!**而BackStackRecord是实现Runnable接口的!!**,现在一切都清楚了,真正执行Op命令的还是在BackStackRecord中。</br>
继续看BackStackRecord的run方法。
```java
while (op != null) {
    ...
    switch (op.cmd) {
        ...
        case OP_ADD: {
            Fragment f = op.fragment;
            f.mNextAnim = enterAnim;
            mManager.addFragment(f, false);
        } break;
        case OP_HIDE: {
            Fragment f = op.fragment;
            f.mNextAnim = exitAnim;
            mManager.hideFragment(f, transition, transitionStyle);
        } break;
        case OP_SHOW: {
            Fragment f = op.fragment;
            f.mNextAnim = enterAnim;
            mManager.showFragment(f, transition, transitionStyle);
        } break;
        ...
    }
    op = op.next;
}

上面是run的部分代码,到这里我们终于看到了希望见到的Op命令!!在run方法中,while会遍历并执行Op链表中的所有命令,执行命令的过程最终还是通过mManager来执行,拿addFragment来示例,

public void addFragment(Fragment fragment, boolean moveToStateNow) {
    ...
    if (moveToStateNow) {
        moveToState(fragment);
    }
}

从上面的代码看到真正控制Fragment状态的是moveToState方法,多次跳转后最终跳转到

void moveToState(Fragment f, int newState, int transit, int transitionStyle, boolean keepActive) {
    ...
    if (f.mState < newState) {
        ...
        switch (f.mState) {
        case Fragment.INITIALIZING:
            ...
            f.onAttach(mHost.getContext());
            ...
        case Fragment.CREATED:
            ...
            // onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) return 的View将在这里被指向
            f.mView = f.performCreateView(f.getLayoutInflater(f.mSavedFragmentState), container, f.mSavedFragmentState);
            //onActivityCreated(savedInstanceState); 在这里被调用
            f.performActivityCreated(f.mSavedFragmentState);
            // Fragment的View将在这被加载进Activity中
            container.addView(f.mView);
            ...
        case Fragment.ACTIVITY_CREATED:
        case Fragment.STOPPED:
            ...
            // onStart() 在这里被调用
            f.performStart();
            ...
        case Fragment.STARTED:
            ...
            // onResume() 在这里被调用
            f.performResume();
            ...
        }
    }else if(f.mState > newState){
        switch (f.mState) {
            case Fragment.RESUMED:
                if (newState < Fragment.RESUMED) {
                    if (DEBUG) Log.v(TAG, "movefrom RESUMED: " + f);
                    // onPause()在这被调用
                    f.performPause();
                    f.mResumed = false;
                }
            case Fragment.STARTED:
                if (newState < Fragment.STARTED) {
                    if (DEBUG) Log.v(TAG, "movefrom STARTED: " + f);
                    // onStop()在这被调用
                    f.performStop();
                }
                ...
            case Fragment.CREATED:
                if (newState < Fragment.CREATED) {
                    if (f.mAnimatingAway != null) {
                        f.mStateAfterAnimating = newState;
                        newState = Fragment.CREATED;
                    } else {
                        if (DEBUG) Log.v(TAG, "movefrom CREATED: " + f);
                        if (!f.mRetaining) {
                            // onDestroy 在这被回调
                            f.performDestroy();
                        }
                    }
                }
    }
}

由于篇幅所限,上面的只是代码片段,从moveToState方法中,我们见到了Fragment的整个生命周期!!!</br>
在第一个switch的case Fragment.CREATED:中Fragment 的 view 的被加载,不知道你是否还记得文章最开始的那个xml中id为content的FrameLayout,在这个case下,fragment的view正是被加载到这个Layout中!!!!!!</br>
这里有个有意思的地方,两个switch都是没有break语句的,当我们第一次add是,f.mState的默认状态为INITIALIZING,该状态下,fragemnt就要走完onAttach--onResume的所有流程,通过上面的代码,最终可以知道,fragment的全部生命周期都由mState这个字段决定。</br>

到了现在getFragmentManager().beginTransaction().replace(R.id.content, details).commit();这句话我们算是完全清楚了它的工作流程。

总结

在加载Fragament中,beginTransaction()创建了一个BackStackRecord对象,该对象实现了Runnable接口,replace方法将replace命令加载到BackStackRecord的Op链表中,当开发者调用commit方法时,commit方法将以事件的形式生成Op命令并将Op传递给FragmentManagerImpl,FragmentManagerImpl在Activity的handler中启动mExecCommit线程,mExecCommit线程执行BackStackRecord线程,在BackStackRecord的run方法里面,遍历所有Op链表,依次执行Op链表中所有的命令,run的switch根据Op命令,动态执行调用FragmentManagerImpl的replace、add、show等方法,在FragmentManagerImpl中的replace、add、show等方法最终都会调用moveToState方法,而整个Fragment的生命周期都在这个moveToState方法中。在这方法中,在create中,fragment最终将被加载找eginTransaction().replace(R.id.content, details),replace第一个参数所指向的Layout中!!加载View完成后,moveToState将根据fragment的mState继续执行Fragment的生命周期。</br>

到现在整个Fragment的生成加载,生命周期我们全部了解完成了,得出的最后一个结论是Fragment本质上是嵌入在Activity中一个ViewGroup的View,但是谷歌给这个View赋予了生命周期,看到Fragment后终于明白Square为什么要建议放弃Fragment了!!!</br>
虽然感觉Square公司有点偏激,但是作为一般开发者,在能用自定义View的情况下还是尽量不要用Fragment,因为Fragment实在太复杂了,一旦出现奇怪的问题,根本找不到哪个地方出的错,说多了都是泪!!!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,928评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,192评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,468评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,186评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,295评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,374评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,403评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,186评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,610评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,906评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,075评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,755评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,393评论 3 320
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,079评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,313评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,934评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,963评论 2 351

推荐阅读更多精彩内容