Fragment 分析上

Fragment状态值 值的大小后面会进行判断

static final int INITIALIZING = 0;     // Not yet created.
static final int CREATED = 1;          // Created.
static final int ACTIVITY_CREATED = 2; // Fully created, not started.
static final int STARTED = 3;          // Created and started, not resumed.
static final int RESUMED = 4;          // Created started and resumed.

我们先看下fragment的生命周期图

生命周期图

activity的生命周期发生改变的时候都会分发不同的事件

比如onDestory

@Override
protected void onDestroy() {
    super.onDestroy();

    if (mViewModelStore != null && !isChangingConfigurations()) {
        mViewModelStore.clear();
    }

    mFragments.dispatchDestroy();
}
/**
 * Dispatch onPause() to fragments.
 */
@Override
protected void onPause() {
    super.onPause();
    mResumed = false;
    if (mHandler.hasMessages(MSG_RESUME_PENDING)) {
        mHandler.removeMessages(MSG_RESUME_PENDING);
        onResumeFragments();
    }
    mFragments.dispatchPause();
}

不管是dispatchDestroy()还是dispatchPause()都会调用dispatchStateChange

public void dispatchStart() {
    mStateSaved = false;
    mStopped = false;
    dispatchStateChange(Fragment.STARTED);
}

public void dispatchResume() {
    mStateSaved = false;
    mStopped = false;
    dispatchStateChange(Fragment.RESUMED);
}

public void dispatchPause() {
    dispatchStateChange(Fragment.STARTED);
}

public void dispatchStop() {
    mStopped = true;
    dispatchStateChange(Fragment.ACTIVITY_CREATED);
}

所以我们查看下dispatchStateChange的方式实现

private void dispatchStateChange(int nextState) {
    try {
        mExecutingActions = true;
        moveToState(nextState, false);
    } finally {
        mExecutingActions = false;
    }
    execPendingActions();
}

我们继续查看moveToState方法

onAttach和onCreate生命周期

上面我们讲了moveToState方法,

fragment初始值为INITIALIZING,所以switch方法会到这里去

case Fragment.INITIALIZING:
    if (newState > Fragment.INITIALIZING) {
        if (DEBUG) Log.v(TAG, "moveto CREATED: " + f);
        if (f.mSavedFragmentState != null) {
            f.mSavedFragmentState.setClassLoader(mHost.getContext()
                    .getClassLoader());
            f.mSavedViewState = f.mSavedFragmentState.getSparseParcelableArray(
                    FragmentManagerImpl.VIEW_STATE_TAG);
            f.mTarget = getFragment(f.mSavedFragmentState,
                    FragmentManagerImpl.TARGET_STATE_TAG);
            if (f.mTarget != null) {
                f.mTargetRequestCode = f.mSavedFragmentState.getInt(
                        FragmentManagerImpl.TARGET_REQUEST_CODE_STATE_TAG, 0);
            }
            if (f.mSavedUserVisibleHint != null) {
                f.mUserVisibleHint = f.mSavedUserVisibleHint;
                f.mSavedUserVisibleHint = null;
            } else {
                f.mUserVisibleHint = f.mSavedFragmentState.getBoolean(
                        FragmentManagerImpl.USER_VISIBLE_HINT_TAG, true);
            }
            if (!f.mUserVisibleHint) {
                f.mDeferStart = true;
                if (newState > Fragment.ACTIVITY_CREATED) {
                    newState = Fragment.ACTIVITY_CREATED;
                }
            }
        }

        f.mHost = mHost;
        f.mParentFragment = mParent;
        f.mFragmentManager = mParent != null
                ? mParent.mChildFragmentManager : mHost.getFragmentManagerImpl();

        // If we have a target fragment, push it along to at least CREATED
        // so that this one can rely on it as an initialized dependency.
        if (f.mTarget != null) {
            if (mActive.get(f.mTarget.mIndex) != f.mTarget) {
                throw new IllegalStateException("Fragment " + f
                        + " declared target fragment " + f.mTarget
                        + " that does not belong to this FragmentManager!");
            }
            if (f.mTarget.mState < Fragment.CREATED) {
                moveToState(f.mTarget, Fragment.CREATED, 0, 0, true);
            }
        }

        dispatchOnFragmentPreAttached(f, mHost.getContext(), false);
        f.mCalled = false;
        f.onAttach(mHost.getContext());
        if (!f.mCalled) {
            throw new SuperNotCalledException("Fragment " + f
                    + " did not call through to super.onAttach()");
        }
        if (f.mParentFragment == null) {
            mHost.onAttachFragment(f);
        } else {
            f.mParentFragment.onAttachFragment(f);
        }
        dispatchOnFragmentAttached(f, mHost.getContext(), false);

        if (!f.mIsCreated) {
            dispatchOnFragmentPreCreated(f, f.mSavedFragmentState, false);
            f.performCreate(f.mSavedFragmentState);
            dispatchOnFragmentCreated(f, f.mSavedFragmentState, false);
        } else {
            f.restoreChildFragmentState(f.mSavedFragmentState);
            f.mState = Fragment.CREATED;
        }
        f.mRetaining = false;
    }

我只复制了一部分,

f.onAttach(mHost.getContext());

f.performCreate(f.mSavedFragmentState); 这个方法调用的时候

void performCreate(Bundle savedInstanceState) {
    if (mChildFragmentManager != null) {
        mChildFragmentManager.noteStateNotSaved();
    }
    mState = CREATED;
    mCalled = false;
    onCreate(savedInstanceState);
    mIsCreated = true;
    if (!mCalled) {
        throw new SuperNotCalledException("Fragment " + this
                + " did not call through to super.onCreate()");
    }
    mLifecycleRegistry.handleLifecycleEvent(Lifecycle.Event.ON_CREATE);
}

我们可以看到状态变成了create下次进来的时候switch case就会发生改变

onCreatView和onViewCreated生命周期

同理create情况下会发生

f.performCreateView(f.performGetLayoutInflater(
        f.mSavedFragmentState), container, f.mSavedFragmentState);
 f.onViewCreated(f.mView, f.mSavedFragmentState);
 f.performActivityCreated(f.mSavedFragmentState);

onStart和onResume生命周期

performActivityCreated的时候 状态变成了ACTIVITY_CREATED

switch case继续发生改变

case Fragment.ACTIVITY_CREATED:
    if (newState > Fragment.ACTIVITY_CREATED) {
        if (DEBUG) Log.v(TAG, "moveto STARTED: " + f);
        f.performStart();
        dispatchOnFragmentStarted(f, false);
    }
    // fall through
case Fragment.STARTED:
    if (newState > Fragment.STARTED) {
        if (DEBUG) Log.v(TAG, "moveto RESUMED: " + f);
        f.performResume();
        dispatchOnFragmentResumed(f, false);
        f.mSavedFragmentState = null;
        f.mSavedViewState = null;
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,607评论 6 507
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,239评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,960评论 0 355
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,750评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,764评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,604评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,347评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,253评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,702评论 1 315
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,893评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,015评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,734评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,352评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,934评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,052评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,216评论 3 371
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,969评论 2 355