日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

Android fragment源码全解析

發(fā)布時間:2025/3/15 Android 57 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android fragment源码全解析 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

Fragment 相信基本上每個android developer都用過,但是知曉其原理 用的好的還是不多,今天就從源碼的角度上來帶著大家分析一下Fragment的源碼,對fragment有了更深層次的認識以后相信

寫出來的代碼也會越來越好看。

首先,我們來看第一個流程,fragment是怎么加載到界面上的,借著這個流程分析,能讀完絕大多數(shù)fragment的源碼。

一般我們顯示一個fragment的時候 喜歡如下這種做法:

1 blankFragment=new BlankFragment(); 2 fm=getFragmentManager(); 3 FragmentTransaction ft=fm.beginTransaction(); 4 ft.replace(R.id.rootView,blankFragment); 5 ft.commit();

?

這段代碼相信大家都很熟悉了,我們就來一步步跟進去看看 ,2-5 執(zhí)行結(jié)束以后 是怎么把fragment界面顯示到手機屏幕上的。

1 //下面的代碼 來自于activity里面!!!!!!!!!!!!!!! 2 public FragmentManager getFragmentManager() { 3 //到這里能發(fā)現(xiàn)是mFragments返回給我們的FragmentManager 4 return mFragments.getFragmentManager(); 5 } 6 7 //繼續(xù)往下跟 就會發(fā)現(xiàn)mFragments是由FragmentController的createController函數(shù) 構(gòu)造出來的一個對象, 8 //并且這個函數(shù) 需要傳進去一個HostCallBack的對象 9 final FragmentController mFragments = FragmentController.createController(new HostCallbacks()); 10 11 12 13 //下面的代碼就來自于FragmentController 這個類!!!!! 14 private final FragmentHostCallback<?> mHost; 15 16 //從這個函數(shù)就能看出來HostCallbacks 這個類肯定是FragmentHostCallback的子類了 17 public static final FragmentController createController(FragmentHostCallback<?> callbacks) { 18 return new FragmentController(callbacks); 19 } 20 21 private FragmentController(FragmentHostCallback<?> callbacks) { 22 mHost = callbacks; 23 } 24 25 //所以這個getFragmentManager返回的就是FragmentManager這個對象,并且這個對象是mHost的getFragmentManagerImpl函數(shù)返回的。 26 //這里結(jié)合構(gòu)造函數(shù)一看就明白了,這個mHost就是我們在activity代碼里面,第12行那里傳進去的HostCallbacks這個對象來幫助初始化的 27 public FragmentManager getFragmentManager() { 28 return mHost.getFragmentManagerImpl(); 29 } 30 31 32 //下面的代碼在activity里 33 //這個地方一目了然 果然我們這個HostCallbacks 這個類是繼承自FragmentHostCallback的,并且能看出來,我們這里把activity的引用也傳進去了。 34 //所以能馬上得出一個結(jié)論就是一個activity對應著一個HostCallbacks對象 這個對象持有本身這個activity的引用。傳進去以后就代表FragmentController 35 //這個類的成員mHost 也持有了activity的引用 36 class HostCallbacks extends FragmentHostCallback<Activity> { 37 public HostCallbacks() { 38 super(Activity.this /*activity*/); 39 } 40 41 @Override 42 public void onDump(String prefix, FileDescriptor fd, PrintWriter writer, String[] args) { 43 Activity.this.dump(prefix, fd, writer, args); 44 } 45 .................略過余下代碼 46 } 47 48 //下面的代碼來源自 FragmentHostCallback<E> 這個抽象類 49 50 public FragmentHostCallback(Context context, Handler handler, int windowAnimations) { 51 this(null /*activity*/, context, handler, windowAnimations); 52 } 53 54 FragmentHostCallback(Activity activity) { 55 this(activity, activity /*context*/, activity.mHandler, 0 /*windowAnimations*/); 56 } 57 58 //到這里就能看到FragmentHostCallback 持有了acitivty的引用 并且連activity的handler都一并持有! 59 FragmentHostCallback(Activity activity, Context context, Handler handler, 60 int windowAnimations) { 61 mActivity = activity; 62 mContext = context; 63 mHandler = handler; 64 mWindowAnimations = windowAnimations; 65 }

上面 初步分析了getFragmentManager這個方法的由來。那繼續(xù)看這個方法到底是返回的什么?

1 //下面的代碼來源自抽象類FragmentHostCallback 2 FragmentManagerImpl getFragmentManagerImpl() { 3 return mFragmentManager; 4 } 5 //所以就能看出來 我們在activity中調(diào)用的getFragmentManger這個方法最終返回的是FragmentManagerImpl 這個類的對象了 6 final FragmentManagerImpl mFragmentManager = new FragmentManagerImpl();

再進去看看 這個對象的begin方法返回的是什么

1 //源碼來自于抽象類 FragmentManager 2 public FragmentTransaction beginTransaction() { 3 //可以看出來 返回的是BackStackRecord 這個類的對象 4 return new BackStackRecord(this); 5 } 6 //下面的代碼來自于BackStackState這個類 7 //可以看到這個類是一個final類 8 final class BackStackState implements Parcelable { 9 } 10 11 //注意BackStackRecord這個類 和BackStackState 是在同一個文件內(nèi)的 12 //可以看一下BackStackRecord 是FragmentTransaction的子類 并且實現(xiàn)了 13 //BackStackEntry, Runnable這兩個接口 14 final class BackStackRecord extends FragmentTransaction implements 15 FragmentManager.BackStackEntry, Runnable { 16 static final String TAG = FragmentManagerImpl.TAG; 17 18 final FragmentManagerImpl mManager; 19 } 20 21 //下面的這個class就是在BackStackRecord這個類的源碼里面的,這里Op 22 //實際上就是一個雙向鏈表結(jié)構(gòu) 23 static final class Op { 24 Op next; 25 Op prev; 26 int cmd; 27 Fragment fragment; 28 int enterAnim; 29 int exitAnim; 30 int popEnterAnim; 31 int popExitAnim; 32 ArrayList<Fragment> removed; 33 } 34

你看 所以begintranscation返回的最終就是backstackrecord對象了。

我們繼續(xù)看看這個對象的操作

1 //以下代碼來源自backstackrecord 源碼 2 public FragmentTransaction replace(int containerViewId, Fragment fragment) { 3 return replace(containerViewId, fragment, null); 4 } 5 6 //你看這里replace操作 你如果沒有傳進去一個有效的id的話 異常就會在這里出現(xiàn)了 7 public FragmentTransaction replace(int containerViewId, Fragment fragment, String tag) { 8 if (containerViewId == 0) { 9 throw new IllegalArgumentException("Must use non-zero containerViewId"); 10 } 11 12 //最終都是調(diào)用的doaAdddop這個方法來完成操作的 13 doAddOp(containerViewId, fragment, tag, OP_REPLACE); 14 return this; 15 } 16 17 //這個方法說白了 就是拼裝下這個雙向鏈表 18 private void doAddOp(int containerViewId, Fragment fragment, String tag, int opcmd) { 19 fragment.mFragmentManager = mManager; 20 21 if (tag != null) { 22 if (fragment.mTag != null && !tag.equals(fragment.mTag)) { 23 throw new IllegalStateException("Can't change tag of fragment " 24 + fragment + ": was " + fragment.mTag 25 + " now " + tag); 26 } 27 fragment.mTag = tag; 28 } 29 30 if (containerViewId != 0) { 31 if (fragment.mFragmentId != 0 && fragment.mFragmentId != containerViewId) { 32 throw new IllegalStateException("Can't change container ID of fragment " 33 + fragment + ": was " + fragment.mFragmentId 34 + " now " + containerViewId); 35 } 36 fragment.mContainerId = fragment.mFragmentId = containerViewId; 37 } 38 39 Op op = new Op(); 40 op.cmd = opcmd; 41 op.fragment = fragment; 42 addOp(op); 43 } 44 45 //所以我們看到 replace操作或者是add remove這種操作 就是操作雙向鏈表的 除此之外沒有任何有意義的舉動,最終反應到用戶能感知的層面上全都是要走 46 //commit這個函數(shù)的 47 public int commit() { 48 return commitInternal(false); 49 } 50 //構(gòu)造函數(shù)再看一遍 51 52 public BackStackRecord(FragmentManagerImpl manager) { 53 mManager = manager; 54 } 55 int commitInternal(boolean allowStateLoss) { 56 if (mCommitted) { 57 throw new IllegalStateException("commit already called"); 58 } 59 if (FragmentManagerImpl.DEBUG) { 60 Log.v(TAG, "Commit: " + this); 61 LogWriter logw = new LogWriter(Log.VERBOSE, TAG); 62 PrintWriter pw = new FastPrintWriter(logw, false, 1024); 63 dump(" ", null, pw, null); 64 pw.flush(); 65 } 66 mCommitted = true; 67 if (mAddToBackStack) { 68 mIndex = mManager.allocBackStackIndex(this); 69 } else { 70 mIndex = -1; 71 } 72 //這個對象就是 final FragmentManagerImpl mManager; 我們在調(diào)用begin函數(shù)的時候傳進去一個this指針 就是用來初始化他的 73 mManager.enqueueAction(this, allowStateLoss); 74 return mIndex; 75 } 76 77 //所以下面就是FragmentManagerImpl 的源碼了 final class FragmentManagerImpl extends FragmentManager implements LayoutInflater.Factory2 78 79 //這個函數(shù)就很關鍵了,這個mHost 前文介紹過 持有了activity的引用,所以這里你看 就是用activity的handler 去執(zhí)行了mExecCommit 80 //注意是在activity的主線程去執(zhí)行的mExecCommit 這個線程 81 public void enqueueAction(Runnable action, boolean allowStateLoss) { 82 if (!allowStateLoss) { 83 checkStateLoss(); 84 } 85 synchronized (this) { 86 if (mDestroyed || mHost == null) { 87 throw new IllegalStateException("Activity has been destroyed"); 88 } 89 if (mPendingActions == null) { 90 mPendingActions = new ArrayList<Runnable>(); 91 } 92 mPendingActions.add(action); 93 if (mPendingActions.size() == 1) { 94 mHost.getHandler().removeCallbacks(mExecCommit); 95 mHost.getHandler().post(mExecCommit); 96 } 97 } 98 } 99 //這個線程執(zhí)行的execPendingActions 就是這個方法 這個方法也是在FragmentManagerImpl 里的。并不在activity里。所以commit操作就是最終讓activity的主線程去執(zhí)行了FragmentManagerImpl 100 //execPendingActions方法 101 Runnable mExecCommit = new Runnable() { 102 @Override 103 public void run() { 104 execPendingActions(); 105 } 106 }; 107 108 /** 109 * 所以這個方法是只能在主線程里面做的 110 */ 111 public boolean execPendingActions() { 112 if (mExecutingActions) { 113 throw new IllegalStateException("Recursive entry to executePendingTransactions"); 114 } 115 116 if (Looper.myLooper() != mHost.getHandler().getLooper()) { 117 throw new IllegalStateException("Must be called from main thread of process"); 118 } 119 120 boolean didSomething = false; 121 122 while (true) { 123 int numActions; 124 125 synchronized (this) { 126 if (mPendingActions == null || mPendingActions.size() == 0) { 127 break; 128 } 129 130 numActions = mPendingActions.size(); 131 if (mTmpActions == null || mTmpActions.length < numActions) { 132 mTmpActions = new Runnable[numActions]; 133 } 134 mPendingActions.toArray(mTmpActions); 135 mPendingActions.clear(); 136 mHost.getHandler().removeCallbacks(mExecCommit); 137 } 138 139 mExecutingActions = true; 140 for (int i=0; i<numActions; i++) { 141 //你看這里run方法 回過頭去 我們應該還能想起來backstackrecord這個類是繼承了runnable這個接口的,所以最終我們還是要看看backstackrecord 的run方法里面都做了什么 142 mTmpActions[i].run(); 143 mTmpActions[i] = null; 144 } 145 mExecutingActions = false; 146 didSomething = true; 147 } 148 149 if (mHavePendingDeferredStart) { 150 boolean loadersRunning = false; 151 for (int i=0; i<mActive.size(); i++) { 152 Fragment f = mActive.get(i); 153 if (f != null && f.mLoaderManager != null) { 154 loadersRunning |= f.mLoaderManager.hasRunningLoaders(); 155 } 156 } 157 if (!loadersRunning) { 158 mHavePendingDeferredStart = false; 159 startPendingDeferredFragments(); 160 } 161 } 162 return didSomething; 163 }

一直到這里 我們就知道,commit操作 最終執(zhí)行的實際上是我們backstackrecord 這個類里的run方法。

1 //以下代碼就是backstackrecord里面的代碼了 2 //這個run方法 其實就是取op這個雙向鏈表然后分析op.cmd的值 然后根據(jù) 3 //這些不同的值 去調(diào)用FragmentManager里各種轉(zhuǎn)換fragment 4 public void run() { 5 if (FragmentManagerImpl.DEBUG) { 6 Log.v(TAG, "Run: " + this); 7 } 8 9 if (mAddToBackStack) { 10 if (mIndex < 0) { 11 throw new IllegalStateException("addToBackStack() called after commit()"); 12 } 13 } 14 15 bumpBackStackNesting(1); 16 17 SparseArray<Fragment> firstOutFragments = new SparseArray<Fragment>(); 18 SparseArray<Fragment> lastInFragments = new SparseArray<Fragment>(); 19 calculateFragments(firstOutFragments, lastInFragments); 20 beginTransition(firstOutFragments, lastInFragments, false); 21 22 Op op = mHead; 23 while (op != null) { 24 switch (op.cmd) { 25 case OP_ADD: { 26 Fragment f = op.fragment; 27 f.mNextAnim = op.enterAnim; 28 mManager.addFragment(f, false); 29 } 30 break; 31 case OP_REPLACE: { 32 Fragment f = op.fragment; 33 int containerId = f.mContainerId; 34 if (mManager.mAdded != null) { 35 for (int i = 0; i < mManager.mAdded.size(); i++) { 36 Fragment old = mManager.mAdded.get(i); 37 if (FragmentManagerImpl.DEBUG) { 38 Log.v(TAG, 39 "OP_REPLACE: adding=" + f + " old=" + old); 40 } 41 if (old.mContainerId == containerId) { 42 if (old == f) { 43 op.fragment = f = null; 44 } else { 45 if (op.removed == null) { 46 op.removed = new ArrayList<Fragment>(); 47 } 48 op.removed.add(old); 49 old.mNextAnim = op.exitAnim; 50 if (mAddToBackStack) { 51 old.mBackStackNesting += 1; 52 if (FragmentManagerImpl.DEBUG) { 53 Log.v(TAG, "Bump nesting of " 54 + old + " to " + old.mBackStackNesting); 55 } 56 } 57 mManager.removeFragment(old, mTransition, mTransitionStyle); 58 } 59 } 60 } 61 } 62 if (f != null) { 63 f.mNextAnim = op.enterAnim; 64 mManager.addFragment(f, false); 65 } 66 } 67 break; 68 case OP_REMOVE: { 69 Fragment f = op.fragment; 70 f.mNextAnim = op.exitAnim; 71 mManager.removeFragment(f, mTransition, mTransitionStyle); 72 } 73 break; 74 case OP_HIDE: { 75 Fragment f = op.fragment; 76 f.mNextAnim = op.exitAnim; 77 mManager.hideFragment(f, mTransition, mTransitionStyle); 78 } 79 break; 80 case OP_SHOW: { 81 Fragment f = op.fragment; 82 f.mNextAnim = op.enterAnim; 83 mManager.showFragment(f, mTransition, mTransitionStyle); 84 } 85 break; 86 case OP_DETACH: { 87 Fragment f = op.fragment; 88 f.mNextAnim = op.exitAnim; 89 mManager.detachFragment(f, mTransition, mTransitionStyle); 90 } 91 break; 92 case OP_ATTACH: { 93 Fragment f = op.fragment; 94 f.mNextAnim = op.enterAnim; 95 mManager.attachFragment(f, mTransition, mTransitionStyle); 96 } 97 break; 98 default: { 99 throw new IllegalArgumentException("Unknown cmd: " + op.cmd); 100 } 101 } 102 103 op = op.next; 104 } 105 //我們也很容易就能看出來 最終都是走的 mManager.moveToState 這個方法 106 //同時moveToState 也是fragment狀態(tài)分發(fā)最重要的方法了 107 108 mManager.moveToState(mManager.mCurState, mTransition, 109 mTransitionStyle, true); 110 111 if (mAddToBackStack) { 112 mManager.addBackStackState(this); 113 } 114 }

到這里應該就差不多了,最終的線索就是 只要搞明白moveToState這個函數(shù)就可以了。

1 //下面的代碼來自于fragmentmanager 2 //我們首先來看一下movetostate這個函數(shù)總共有一個 3 void moveToState(Fragment f) 4 void moveToState(int newState, boolean always) 5 void moveToState(int newState, int transit, int transitStyle, boolean always) 6 void moveToState(Fragment f, int newState, int transit, int transitionStyle, 7 boolean keepActive) 8 9 //可以看到movetoState總共4種。 10 //在詳細介紹movetostate函數(shù)之前,我們先去看看這個函數(shù)的參數(shù)之一new state是什么 11 12 //下面代碼來自于fragment 13 //其實new state 就是代表新的狀態(tài),總共他的值有7種 就全在這里了 預先都是定義好的 14 static final int INVALID_STATE = -1; // Invalid state used as a null value. 15 static final int INITIALIZING = 0; // Not yet created. 16 static final int CREATED = 1; // Created. 17 static final int ACTIVITY_CREATED = 2; // The activity has finished its creation. 這個狀態(tài)其實很好理解, 18 //就是fragement在oncreate函數(shù)結(jié)束的時候會調(diào)用dispatchActivityCreated 就是通知fragment 跟你綁定的宿主activity已經(jīng)走完onCreate了 19 static final int STOPPED = 3; // Fully created, not started. 20 static final int STARTED = 4; // Created and started, not resumed. 21 static final int RESUMED = 5; // Created started and resumed. 22 23 24 //下面我們可以模擬一個流程 幫助大家理解這個狀態(tài)到底是干嘛的 有什么用。 25 //比如 我們先看看 fragmentactivity的源碼, 26 //首先我們假設 我們想看看activity 發(fā)生onResumne事件的時候 對fragment有什么影響 27 protected void onResume() { 28 super.onResume(); 29 mHandler.sendEmptyMessage(MSG_RESUME_PENDING); 30 mResumed = true; 31 mFragments.execPendingActions(); 32 } 33 //繼續(xù)追蹤代碼 發(fā)現(xiàn)最后是調(diào)用的onResumeFragments 這個方法 34 final Handler mHandler = new Handler() { 35 @Override 36 public void handleMessage(Message msg) { 37 switch (msg.what) { 38 case MSG_REALLY_STOPPED: 39 if (mStopped) { 40 doReallyStop(false); 41 } 42 break; 43 case MSG_RESUME_PENDING: 44 onResumeFragments(); 45 mFragments.execPendingActions(); 46 break; 47 default: 48 super.handleMessage(msg); 49 } 50 } 51 52 }; 53 //原來當activity走onresume流程的時候 最終都是走到這里 54 55 protected void onResumeFragments() { 56 mFragments.dispatchResume(); 57 } 58 //前面已經(jīng)分析過mFragements就是FragmentController的對象 59 //所以下面的代碼 來自于FragmentController 60 public void dispatchResume() { 61 //前面的源碼也分析過了mHost.mFragmentManager 就是 final FragmentManagerImpl mFragmentManager = new FragmentManagerImpl(); 62 mHost.mFragmentManager.dispatchResume(); 63 } 64 65 //下面的代碼來自fragmentmanager 66 //一直追蹤到這里就能明白 activity的聲明周期 與fragment聲明周期關聯(lián)的時候 就是通過moveToState 這個函數(shù)來完成的 67 public void dispatchResume() { 68 mStateSaved = false; 69 moveToState(Fragment.RESUMED, false); 70 } 71 72 //movetostate這個函數(shù)前面已經(jīng)說過總共有4種 不一樣的聲明 但是最終起效果的只有這一個 73 //這個函數(shù)非常的長 我就簡單挑幾個注意的點進行注釋 代碼我就不全部復制粘貼進來了。太長了 74 //有興趣的同學可以自己跟進去看看 其實邏輯挺簡單的 75 void moveToState(Fragment f, int newState, int transit, int transitionStyle, 76 boolean keepActive) { 77 ...... 78 if (f.mState < newState) { 79 // For fragments that are created from a layout, when restoring from 80 // state we don't want to allow them to be created until they are 81 // being reloaded from the layout. 82 if (f.mFromLayout && !f.mInLayout) { 83 return; 84 } 85 if (f.mAnimatingAway != null) { 86 // The fragment is currently being animated... but! Now we 87 // want to move our state back up. Give up on waiting for the 88 // animation, move to whatever the final state should be once 89 // the animation is done, and then we can proceed from there. 90 f.mAnimatingAway = null; 91 moveToState(f, f.mStateAfterAnimating, 0, 0, true); 92 } 93 switch (f.mState) { 94 case Fragment.INITIALIZING: 95 ......................................... 96 } 97 } 98 f.mHost = mHost; 99 f.mParentFragment = mParent; 100 f.mFragmentManager = mParent != null 101 ? mParent.mChildFragmentManager : mHost.getFragmentManagerImpl(); 102 f.mCalled = false; 103 //這個地方相信很多人一看就明白了,這行代碼就說明了在onAttach的時候 就能使用和fragment關聯(lián)的activity了,這也是為什么 104 //fragment與activity通信時,我們喜歡定義接口來完成,并且在onAttach的時候綁定接口 的原因 105 f.onAttach(mHost.getContext()); 106 if (!f.mCalled) { 107 throw new SuperNotCalledException("Fragment " + f 108 + " did not call through to super.onAttach()"); 109 } 110 if (f.mParentFragment == null) { 111 mHost.onAttachFragment(f); 112 } 113 114 if (!f.mRetaining) { 115 f.performCreate(f.mSavedFragmentState); 116 } 117 f.mRetaining = false; 118 if (f.mFromLayout) { 119 // For fragments that are part of the content view 120 // layout, we need to instantiate the view immediately 121 // and the inflater will take care of adding it. 122 f.mView = f.performCreateView(f.getLayoutInflater( 123 f.mSavedFragmentState), null, f.mSavedFragmentState); 124 if (f.mView != null) { 125 f.mInnerView = f.mView; 126 if (Build.VERSION.SDK_INT >= 11) { 127 ViewCompat.setSaveFromParentEnabled(f.mView, false); 128 } else { 129 f.mView = NoSaveStateFrameLayout.wrap(f.mView); 130 } 131 if (f.mHidden) f.mView.setVisibility(View.GONE); 132 f.onViewCreated(f.mView, f.mSavedFragmentState); 133 } else { 134 f.mInnerView = null; 135 } 136 } 137 case Fragment.CREATED: 138 ...................... 139 }

一直分析到這里,相信大家就對fragment的源碼基礎知識有一個不錯的理解了,在這里 就簡單總結(jié)一下 上面的分析:

1.FragmentActivity 是具有支持fragment功能的最底層的activity。其他什么AppCompatActivity都是他的子類!

2.FragmentActivity主要負責就是生命周期的轉(zhuǎn)發(fā),比如onCreate onResume onDestroy等等,這就是為什么activity和fragment狀態(tài)能統(tǒng)一的原因了!

當然了,分發(fā)的原因就是因為fragmentactivity源碼里面持有一個fragmentController的實例!

3.其實將白了,fragmentController就是因為他自己有一個fragmenthostcallback,然后這個hostback還持有了fragmentmanger 所以這個controller 能分發(fā)activity的事件!

4.fragementhostcallback持有了activity的很多資源,context handler 是最主要的2個。fragmentmanger就是因為拿到了activty的這2個資源,所以才能和activty互相通信的!

5.fragmentmangerimple就是fragmentmanger的具體實現(xiàn)類。movetostate方法就是在這個里面實現(xiàn)的?

6.FragmentTransition 也是個抽象類,他主要就是提供對外的接口函數(shù)的 add replace move 這種。BackStackRecord 就是它的具體實現(xiàn)類。還額外實現(xiàn)了runnable接口。

所以BackStackRecord 里面會有個run方法 這個run方法就是根據(jù)不同的操作(所謂操作就是OP.CMD的那個值) 來分發(fā)不同的事件,從而調(diào)用fragmentmanger的各種轉(zhuǎn)換fragment生命周期的方法!

?

最后在說一下 fragment的 緩存和恢復機制吧。

1 //保存fragment狀態(tài)的 主要是靠FragmentState 這個類來完成的 2 final class FragmentState implements Parcelable 3 //可以看一下這個類的構(gòu)造函數(shù) 4 public FragmentState(Fragment frag) { 5 mClassName = frag.getClass().getName(); 6 mIndex = frag.mIndex; 7 mFromLayout = frag.mFromLayout; 8 mFragmentId = frag.mFragmentId; 9 mContainerId = frag.mContainerId; 10 mTag = frag.mTag; 11 mRetainInstance = frag.mRetainInstance; 12 mDetached = frag.mDetached; 13 mArguments = frag.mArguments; 14 } 15 16 //再看一下這個類: 這里保存了3個數(shù)組 并且這3個數(shù)組元素都實現(xiàn)了Parcelable 接口 17 //這意味著他們都可以被序列化 18 final class FragmentManagerState implements Parcelable { 19 FragmentState[] mActive; 20 int[] mAdded; 21 BackStackState[] mBackStack; 22 23 public FragmentManagerState() { 24 } 25 26 public FragmentManagerState(Parcel in) { 27 mActive = in.createTypedArray(FragmentState.CREATOR); 28 mAdded = in.createIntArray(); 29 mBackStack = in.createTypedArray(BackStackState.CREATOR); 30 } 31 32 public int describeContents() { 33 return 0; 34 } 35 36 public void writeToParcel(Parcel dest, int flags) { 37 dest.writeTypedArray(mActive, flags); 38 dest.writeIntArray(mAdded); 39 dest.writeTypedArray(mBackStack, flags); 40 } 41 42 public static final Parcelable.Creator<FragmentManagerState> CREATOR 43 = new Parcelable.Creator<FragmentManagerState>() { 44 public FragmentManagerState createFromParcel(Parcel in) { 45 return new FragmentManagerState(in); 46 } 47 48 public FragmentManagerState[] newArray(int size) { 49 return new FragmentManagerState[size]; 50 } 51 }; 52 } 53 54 //上面那個類的3個屬性 實際上對應保存著是fragemntmanager里的 三個成員 55 ArrayList<Fragment> mActive;//他還保存了mBackStack所有相關的fragment 所以mAdder是mActive的子集 56 ArrayList<Fragment> mAdded; 57 ArrayList<BackStackRecord> mBackStack;//這個就是保存調(diào)用了addToBackStack方法的FragementTransaction,你看就是這個東西記錄了 58 //你commit的操作 所以當你調(diào)用了addToBackStack 以后再按返回鍵 就可以回到上一個fragment了

然后我們看一下 當我們的activity onstop以后 會給fragment帶來什么?

1 //下面代碼來自于fragmentactivity 2 @Override 3 protected void onStop() { 4 super.onStop(); 5 6 mStopped = true; 7 mHandler.sendEmptyMessage(MSG_REALLY_STOPPED); 8 9 mFragments.dispatchStop(); 10 } 11 12 //來自于fragmentcontroller 13 public void dispatchStop() { 14 mHost.mFragmentManager.dispatchStop(); 15 } 16 17 //來自于fragemntmanager 18 public void dispatchStop() { 19 // See saveAllState() for the explanation of this. We do this for 20 // all platform versions, to keep our behavior more consistent between 21 // them. 22 mStateSaved = true; 23 //你看這里就是轉(zhuǎn)換了一下狀態(tài) 24 moveToState(Fragment.STOPPED, false); 25 } 26 27 //所以對應的你也能猜到了 當activity onresume的時候 這里也無非就是把fragement的狀態(tài) 從stopped 變成resumed了。 fragement是實例并沒有銷毀 還在 28 public void dispatchResume() { 29 mStateSaved = false; 30 moveToState(Fragment.RESUMED, false); 31 }

我們再考慮一下另外一個場景:

比如說 我們旋轉(zhuǎn)了屏幕。并且

setRetainInstance 為true的時候

看看fragment是怎么處理的(為false的情況 就是fragment和activity一樣了 activity怎么做fragment就怎么做 沒什么好講的必要。。)

?

1 //下面代碼來自于fragmentmanager 2 //如果Fragment設置了fragment.setRetainInstance(true) 最終ams 會一步步調(diào)用到這個函數(shù)的 3 //所以你看 這里就是返回了mActive 數(shù)組的拷貝呀! 4 ArrayList<Fragment> retainNonConfig() { 5 ArrayList<Fragment> fragments = null; 6 if (mActive != null) { 7 for (int i=0; i<mActive.size(); i++) { 8 Fragment f = mActive.get(i); 9 if (f != null && f.mRetainInstance) { 10 if (fragments == null) { 11 fragments = new ArrayList<Fragment>(); 12 } 13 fragments.add(f); 14 f.mRetaining = true; 15 f.mTargetIndex = f.mTarget != null ? f.mTarget.mIndex : -1; 16 if (DEBUG) Log.v(TAG, "retainNonConfig: keeping retained " + f); 17 } 18 } 19 } 20 return fragments; 21 } 22 23 //上面說了保存fragment實例 下面肯定要說如何存儲fragemnt的實例的 24 //下面代碼來自于activity 25 26 NonConfigurationInstances retainNonConfigurationInstances() { 27 Object activity = onRetainNonConfigurationInstance(); 28 HashMap<String, Object> children = onRetainNonConfigurationChildInstances(); 29 List<Fragment> fragments = mFragments.retainNonConfig(); 30 ArrayMap<String, LoaderManager> loaders = mFragments.retainLoaderNonConfig(); 31 if (activity == null && children == null && fragments == null && loaders == null 32 && mVoiceInteractor == null) { 33 return null; 34 } 35 //這里nci你看就知道了 看下類的源碼你看他保存的東西 并沒有做什么序列化反序列化的操作, 36 //所以他可以保存任何東西!當然了,這個nci 是最終保存在activitythread對象里的, 37 //activitytheread對象里有個鍵值對叫mActivies。他有個數(shù)據(jù)結(jié)構(gòu)叫activityclientrecord 38 //有興趣的人可以去看下activitytheread的源碼 這里不深入展開了。 39 NonConfigurationInstances nci = new NonConfigurationInstances(); 40 //注意 nci.activity這個地方 可不是activity,他是activity源碼中onRetainNonConfigurationInstance方法返回的對象咯,看63行就知道了 41 nci.activity = activity; 42 nci.children = children; 43 nci.fragments = fragments; 44 nci.loaders = loaders; 45 if (mVoiceInteractor != null) { 46 mVoiceInteractor.retainInstance(); 47 nci.voiceInteractor = mVoiceInteractor; 48 } 49 return nci; 50 }. 51 52 //所以看到這里就應該明白,如果你的setRetainInstance設置了true的話,當activity重新recreate的時候,雖然activity生成了一個全新的,fragmentmanger也是一個全新的, 53 //但是你的fragment實際上還是舊的,生命周期會有一些不同的,不會有oncreate和ondestroy了。他會走85行那里的restoreAllState方法了 54 static final class NonConfigurationInstances { 55 Object activity; 56 HashMap<String, Object> children; 57 List<Fragment> fragments; 58 ArrayMap<String, LoaderManager> loaders; 59 VoiceInteractor voiceInteractor; 60 } 61 62 63 //下面這個方法在fragemntactitivy源碼里 64 public final Object onRetainNonConfigurationInstance() { 65 if (mStopped) { 66 doReallyStop(true); 67 } 68 69 Object custom = onRetainCustomNonConfigurationInstance(); 70 71 List<Fragment> fragments = mFragments.retainNonConfig(); 72 SimpleArrayMap<String, LoaderManager> loaders = mFragments.retainLoaderNonConfig(); 73 74 if (fragments == null && loaders == null && custom == null) { 75 return null; 76 } 77 78 NonConfigurationInstances nci = new NonConfigurationInstances(); 79 nci.custom = custom; 80 nci.fragments = fragments; 81 nci.loaders = loaders; 82 return nci; 83 } 84 85 //以下代碼來自于fragmentmanger restoreAllState這個方法就是恢復保存的fragment實例的 86 void restoreAllState(Parcelable state, List<Fragment> nonConfig) { 87 // If there is no saved state at all, then there can not be 88 // any nonConfig fragments either, so that is that. 89 if (state == null) return; 90 FragmentManagerState fms = (FragmentManagerState)state; 91 if (fms.mActive == null) return; 92 93 // First re-attach any non-config instances we are retaining back 94 // to their saved state, so we don't try to instantiate them again. 95 if (nonConfig != null) { 96 for (int i=0; i<nonConfig.size(); i++) { 97 Fragment f = nonConfig.get(i); 98 if (DEBUG) Log.v(TAG, "restoreAllState: re-attaching retained " + f); 99 FragmentState fs = fms.mActive[f.mIndex]; 100 fs.mInstance = f; 101 f.mSavedViewState = null; 102 f.mBackStackNesting = 0; 103 f.mInLayout = false; 104 f.mAdded = false; 105 f.mTarget = null; 106 if (fs.mSavedFragmentState != null) { 107 fs.mSavedFragmentState.setClassLoader(mHost.getContext().getClassLoader()); 108 f.mSavedViewState = fs.mSavedFragmentState.getSparseParcelableArray( 109 FragmentManagerImpl.VIEW_STATE_TAG); 110 f.mSavedFragmentState = fs.mSavedFragmentState; 111 } 112 } 113 } 114 115 ................................. 116 }

最后再考慮一種場景,假設我們的宿主activity 在后臺掛起的時候,因為內(nèi)存不足 被系統(tǒng)殺掉了。fragment會發(fā)生什么?

其實也很簡單啊,源碼就不貼了,大家自己看,我說下簡單的流程:

1.首先要明確 activity的onSaveInstanceState的方法,是在onPause以后 onStop以前調(diào)用的。

2.activty放到后臺的時候會調(diào)用onstop方法,但是onSaveInstanceState是在這之前被調(diào)用的

3.所以實際上FragmentManager保存的那3個數(shù)組mActive、mAdded、mBackStack都被提前保存到FragmentManagerState里面了

4.等到activity重新回到前臺 走oncreate的時候,會獲得savedInstanceState這個實例,通過他去創(chuàng)建新的FragmentManager實例和新的fragment對象。

5.此時不管fragment是否setRetainInstance(true),Fragment實例都會重新被創(chuàng)建,原因如下:

retainNonConfig是在Activity在onDestroy被保存的,有人會說,你上面被系統(tǒng)回收了不是也要最終走ondestroy嗎,但是要注意的是:

只有被relaunch的activity在destroy時才會在ActivityThread代碼中被調(diào)用retainNonConfig去通知Activity返回需要保存實例,其他的destroy不會。

所謂relaunch是指 比如我們手動調(diào)用了activity的recreate方法,或者更改了系統(tǒng)語言 屏幕方向等造成的activity重新創(chuàng)建。而系統(tǒng)資源不足回收造成的activity重新創(chuàng)建

是不屬于relaunch這一行為的。


原文地址:https://developer.android.com/guide/components/fragments.html

總結(jié)

以上是生活随笔為你收集整理的Android fragment源码全解析的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

人人爽久久久噜噜噜电影 | 国产xxxxx在线观看 | 最近2019好看的中文字幕免费 | 亚洲最新视频在线 | 久久久免费精品 | 国产视频黄 | 国产小视频在线免费观看 | 国产91精品在线观看 | www日日夜夜| 成人性生爱a∨ | 亚洲精品国产综合99久久夜夜嗨 | 在线 成人 | 国产黄色片久久久 | www激情久久| 97成人在线 | 日本久久电影 | 91黄色视屏| 青春草免费视频 | 美女网站视频色 | 久久精品国产成人 | 国产精品男女啪啪 | 99精品视频免费看 | 在线观看日本韩国电影 | 特级毛片爽www免费版 | 国产精品美女久久久久久 | 日韩高清在线一区二区 | 人人藻人人澡人人爽 | 日本在线观看视频一区 | 一区二区三区日韩视频在线观看 | 成年人在线 | 日日干综合 | 91重口视频| 久草网站 | 综合av在线| 视频一区在线播放 | 亚洲九九九在线观看 | 不卡av电影在线 | 国产黄网在线 | 久久久人人人 | 欧美另类v | 中文字幕永久 | 国产视频网站在线观看 | 日韩大片在线 | 国产明星视频三级a三级点| 美女视频黄频大全免费 | 国产伦理久久精品久久久久_ | 亚洲网久久 | 夜色成人网 | 99久久这里有精品 | 美女精品在线 | 在线免费观看视频a | 最新av网址在线 | 精品一二 | 一区二区三区四区五区在线 | 久草在线视频看看 | 亚洲另类在线视频 | av千婊在线免费观看 | 国产午夜精品一区二区三区在线观看 | 久久精品视频观看 | 亚洲不卡123 | 欧美污在线观看 | 成年人视频在线免费播放 | 国产又黄又猛又粗 | 91网在线看| 亚洲综合色播 | 国产精品成人免费 | 97成人资源| 99精品国产福利在线观看免费 | 91精品国产99久久久久久红楼 | 久久成人高清 | 黄色毛片视频免费观看中文 | 韩日av一区二区 | 免费97视频| 国产精品一区二区在线免费观看 | 国产成人在线观看免费 | 丁香5月婷婷久久 | 热精品 | 黄色网址在线播放 | 亚洲 欧美 日韩 综合 | 亚洲精品成人 | 亚洲网久久| 亚洲精品在线一区二区 | 久久看看| 日韩 在线a | 国产免费久久久久 | 午夜精品一区二区三区在线 | 免费看黄色大全 | 狠狠躁夜夜躁人人爽超碰91 | 五月婷婷在线观看 | 在线日韩中文 | 色在线免费视频 | 欧美一级特黄aaaaaa大片在线观看 | 国产 字幕 制服 中文 在线 | 国产在线精品区 | 日韩一二区在线 | 天天天天天天操 | 最近中文字幕大全 | 婷婷久久亚洲 | 亚洲精品男人天堂 | 久久手机精品视频 | 国产精品不卡在线观看 | 久久免费电影网 | 狠狠干成人综合网 | 久久精品日产第一区二区三区乱码 | 激情视频二区 | 182午夜在线观看 | 免费91在线 | 在线观看视频黄 | 91成人看片 | www.色com| 99 国产精品 | 国产福利在线不卡 | 伊人久久国产精品 | 亚洲黄色片一级 | bbbbb女女女女女bbbbb国产 | 中文字幕一区二区三区在线播放 | www.香蕉视频 | 高清视频一区 | 日韩高清成人在线 | 69国产精品视频免费观看 | 一区二区电影在线观看 | 免费视频99| 蜜臀aⅴ国产精品久久久国产 | av中文字幕在线免费观看 | 在线免费黄色 | 91久久久国产精品 | 日韩免费福利 | 午夜视频在线观看网站 | 欧美在线久久 | 久久久久网站 | 婷婷激情综合网 | 成人av在线影院 | 在线视频精品 | 欧美综合久久 | aaa日本高清在线播放免费观看 | 色婷婷综合在线 | 欧美一级久久久久 | 成人精品一区二区三区中文字幕 | 国产精品综合av一区二区国产馆 | 亚洲国产无| 色综合天天色综合 | 天天色天天骑天天射 | 狠狠地日 | 97香蕉久久超级碰碰高清版 | 香蕉在线视频播放网站 | 欧美日本啪啪无遮挡网站 | 黄免费在线观看 | 成人久久免费 | 欧美成人xxxx | 亚洲一区二区三区毛片 | 国产成人一区二区三区在线观看 | 久久深夜福利免费观看 | 久久精品99北条麻妃 | 三级毛片视频 | 日韩三级在线观看 | 婷婷激情欧美 | 久久成人精品电影 | 一区二区三区在线观看中文字幕 | 久久无码精品一区二区三区 | 国产91免费在线 | 91视频免费观看 | 香蕉影视app | av一区在线播放 | 国产亚洲视频在线 | 韩国av一区二区 | 手机av永久免费 | 久久综合五月天婷婷伊人 | 亚洲在线视频观看 | 天天干天天射天天插 | 日韩一区在线免费观看 | 久久精品久久99精品久久 | 色噜噜狠狠色综合中国 | 深爱开心激情网 | 亚洲欧美日韩在线看 | 亚洲日本中文字幕在线观看 | 在线观看成人 | 狠狠干夜夜操 | 91久久电影| 免费大片黄在线 | 久久er99热精品一区二区三区 | 黄色在线观看www | 天天鲁一鲁摸一摸爽一爽 | 精壮的侍卫呻吟h | 国产精品私人影院 | 午夜国产在线 | 免费看成人a| 亚洲精品美女久久久久网站 | 日批视频国产 | 日韩在线免费观看视频 | 色综合久久综合中文综合网 | 黄色网在线免费观看 | 成人在线观看网址 | 精品在线观看一区二区三区 | 韩国在线视频一区 | 日本精品一区二区 | 欧美一区二视频在线免费观看 | 最近中文字幕高清字幕在线视频 | 日韩美在线| 黄色三级网站 | 精品乱码一区二区三四区 | 精品国产电影一区二区 | 久久精品首页 | 婷婷色资源 | 国产精品美女在线观看 | 亚洲精品美女久久久久网站 | 亚洲第一区在线观看 | 国产精品成人免费 | 天天插天天射 | 久久99精品久久久久婷婷 | 天天干天天操av | 五月天狠狠操 | 欧美午夜性生活 | 亚洲va欧美va人人爽春色影视 | 国产电影一区二区三区四区 | 精品久久精品久久 | 中文字幕永久在线 | 日韩亚洲国产中文字幕 | 午夜三级大片 | 久草视频中文在线 | 久久综合九色欧美综合狠狠 | 久久高清国产 | 日日躁夜夜躁aaaaxxxx | 国产一区免费视频 | av中文字幕日韩 | 中午字幕在线观看 | 久久视频精品在线 | 色夜视频 | 99久久99久久精品免费 | 九九免费观看全部免费视频 | 在线精品播放 | 91精品一 | aⅴ视频在线 | 色综合激情网 | 亚洲狠狠丁香婷婷综合久久久 | 亚洲人人射 | 午夜久久久久久久久久久 | 国产aaa免费视频 | 国产三级av在线 | 国产精品丝袜久久久久久久不卡 | 久久99久久99免费视频 | 国产精品一区免费看8c0m | 日本中文字幕电影在线免费观看 | 成人动漫视频在线 | 久久伦理电影网 | 99产精品成人啪免费网站 | 日狠狠 | 日韩一区二区三区免费电影 | 欧洲黄色片 | 成人观看| 狠狠ri| 99精品久久久久久久 | 日韩动漫免费观看高清完整版在线观看 | 麻豆首页| 日本大片免费观看在线 | 日韩色中色 | av一级片网站| 免费高清在线观看电视网站 | 在线观看一区二区视频 | 国产成人精品亚洲 | 免费视频xnxx com| 亚洲成人av一区二区 | 免费a v在线| 欧美视屏一区二区 | 免费在线播放 | www.天天射| 久久久久久久久久福利 | 狠狠地操 | 麻豆系列在线观看 | 91在线视频观看免费 | 在线看中文字幕 | 亚洲综合涩 | 欧美aa一级 | 国产精品久久久久久久久久久久午 | 韩日电影在线 | 91视频三区| 久久久久欠精品国产毛片国产毛生 | 岛国av在线 | 精品美女国产在线 | 国产精品麻豆果冻传媒在线播放 | 丰满少妇一级片 | 久久免费a | 免费在线观看av不卡 | 午夜精品视频免费在线观看 | 国产成人久久久久 | 日韩艹 | 97碰碰视频| 久久不射电影网 | 国产亚洲精品久久久久久无几年桃 | 97视频久久久| 日韩成人欧美 | 在线免费高清一区二区三区 | 五月天综合激情网 | 亚洲艳情 | 亚洲影视资源 | 国产麻豆成人传媒免费观看 | 精品字幕在线 | 亚洲黄色在线播放 | 日韩中文在线电影 | 国产成人精品一区二区三区免费 | 日韩簧片在线观看 | 国产护士hd高朝护士1 | 激情欧美国产 | 色久av| 成人动漫一区二区三区 | 国产字幕在线看 | 久久99久久99精品 | 国产手机视频在线观看 | 久久综合九色欧美综合狠狠 | 麻豆视频在线免费看 | 婷婷丁香激情五月 | 亚洲精品国精品久久99热 | 91亚洲精品久久久蜜桃网站 | 99这里只有久久精品视频 | 国产精品99久久久久久人免费 | 在线视频欧美日韩 | 西西444www高清大胆 | www.香蕉视频在线观看 | 亚洲电影图片小说 | 久久超级碰视频 | 久久激情视频 久久 | 久久久久国产成人精品亚洲午夜 | 国产亚洲情侣一区二区无 | 亚洲美女在线一区 | 亚洲精选久久 | 另类五月激情 | 久久久久夜色 | 国产精品久一 | 91丨九色丨国产在线观看 | 久久性生活片 | 日韩av手机在线看 | 奇米影视999| 99精品国产一区二区 | 91最新在线| 亚洲国产中文字幕在线视频综合 | 五月天六月色 | 狠狠躁18三区二区一区ai明星 | 精品国产伦一区二区三区观看说明 | 99热九九这里只有精品10 | 色综合久久久久综合 | 亚洲日本va午夜在线影院 | 一区二区三区福利 | 久久久久久中文字幕 | 色偷偷网站视频 | a天堂一码二码专区 | 五月婷婷视频在线观看 | 九九热在线精品视频 | 97网在线观看 | 成人av在线直播 | 天天爽人人爽夜夜爽 | 久草精品电影 | 国产精品色婷婷视频 | 午夜影院一级 | 亚洲女人av| 国产精品久久久久影视 | 久久艹免费 | 久久免费激情视频 | 国产午夜一级毛片 | 99精品福利视频 | 国产激情久久久 | 成人毛片在线观看视频 | 欧美一区二区三区免费观看 | 国产精品2019| 亚洲女同ⅹxx女同tv | 久久精品1区 | 亚洲高清激情 | 日日操天天爽 | 一二三久久久 | 日韩精品视频一二三 | 精品视频中文字幕 | 欧美另类高清 videos | 黄色小说网站在线 | 久久久久久欧美二区电影网 | 中文字幕免费高清在线观看 | 成人在线播放免费观看 | 亚洲天天在线日亚洲洲精 | 国产精品美女免费看 | 黄色美女免费网站 | 久久久久久久久久久久影院 | 狠狠色丁香婷婷综合久小说久 | 久久成年人网站 | 91av在线免费播放 | 天天色天天 | 91在线永久 | 一区二区三区韩国免费中文网站 | 日本久久高清视频 | 人人爱人人添 | 美女精品在线 | 日本中文一区二区 | 99热在线网站 | 欧美在线一二 | 亚洲精品久久久蜜桃直播 | 在线观看黄污 | 中文字幕丝袜美腿 | 国产精品丝袜 | 一区二区中文字幕在线播放 | 国产精品区二区三区日本 | www..com毛片 | 伊人网综合在线观看 | 波多野结衣电影一区二区三区 | 最近免费观看的电影完整版 | 激情开心色 | 成人av在线资源 | 日韩久久久久久久久久久久 | 色偷偷88888欧美精品久久久 | 伊人久久影视 | 久久久久久蜜桃一区二区 | 亚洲精品久久久久58 | 日日操日日插 | 欧美精品一区二区免费 | 丁香六月久久综合狠狠色 | 9色在线视频| 手机版av在线 | 深夜福利视频一区二区 | 99精品视频在线观看 | 国产在线观看一区 | 亚洲有 在线 | 去看片 | 色狠狠综合 | 日韩精品视频网站 | 欧美天天综合网 | 黄色成人免费电影 | 欧美福利在线播放 | 69精品视频在线观看 | 国产做aⅴ在线视频播放 | 欧美精品免费在线观看 | 91女神的呻吟细腰翘臀美女 | 久久久精品国产一区二区电影四季 | 在线成人小视频 | 国内精品亚洲 | 91精品视频免费观看 | 色就干| 久久99国产综合精品免费 | 天天操夜夜操 | 96久久欧美麻豆网站 | 亚洲夜夜爽 | 在线涩涩 | 色播激情五月 | 91免费高清观看 | 日本 在线 视频 中文 有码 | 综合网五月天 | 日日碰狠狠躁久久躁综合网 | 五月天欧美精品 | 久久成年人 | 六月色婷婷 | www.亚洲精品在线 | 国产激情电影综合在线看 | 国产亚洲永久域名 | 超碰97国产在线 | 国产精品第一页在线 | 九九久久国产 | 久久高清av| 国产高清av免费在线观看 | 西西人体4444www高清视频 | 国产精品免费观看国产网曝瓜 | 又黄又爽又无遮挡的视频 | 国产在线观看免 | 国产视频一二三 | 亚洲国产成人在线观看 | 国产在线精品福利 | 91亚洲精品久久久久图片蜜桃 | 婷婷综合视频 | 国产精品99视频 | av资源免费在线观看 | 久久久久久久久久久免费 | 国产黄色片网站 | 激情综合网婷婷 | 夜添久久精品亚洲国产精品 | 欧美日韩免费看 | 97在线视 | 日韩成人不卡 | 日韩av在线免费看 | 特级毛片aaa | 国产精品免费久久久久久 | 99日韩精品 | 天天曰天天 | 超碰国产人人 | 夜添久久精品亚洲国产精品 | 毛片美女网站 | 香蕉视频导航 | 日韩免费三区 | 96精品在线 | av网址在线播放 | 国产日产精品久久久久快鸭 | 亚洲天堂自拍视频 | 久久久天堂 | 久久精品欧美一区二区三区麻豆 | 国产亚洲成人精品 | 国产美女免费观看 | 麻豆视频成人 | 五月天综合网站 | 99草视频 | 99在线观看视频网站 | 国产精品久久久一区二区 | 亚洲成色777777在线观看影院 | 久久久久久久久久久成人 | 色综合久久综合中文综合网 | 美女网站视频色 | 99久久这里有精品 | 在线观看久| 99精品亚洲| 青青色影院 | 天堂网中文在线 | 亚洲激情精品 | 国产a免费 | 国产精品国产自产拍高清av | 日韩免费播放 | 97超碰中文字幕 | 极品中文字幕 | 激情视频区 | 亚洲在线视频免费观看 | 久操视频在线免费看 | 精品在线播放视频 | 韩国三级av在线 | 超碰com| 成人超碰97 | 日黄网站| 久久综合九色九九 | 激情五月在线观看 | 免费男女羞羞的视频网站中文字幕 | 99久久这里有精品 | 国产精品18久久久久久久 | 开心综合网 | 久久99精品久久久久蜜臀 | 国产精品99久久久精品免费观看 | 亚洲国产999 | 一区 二区 精品 | 国产在线观 | 伊人色综合网 | 成人全视频免费观看在线看 | 美女网站在线 | 免费网站黄 | 婷婷丁香激情综合 | 国产成免费视频 | 色吊丝在线永久观看最新版本 | 成人在线中文字幕 | 99国产成+人+综合+亚洲 欧美 | 99热99re6国产在线播放 | 97免费视频在线 | 毛片3 | 不卡av电影在线观看 | 日韩久久电影 | 欧美成人h版 | 麻豆传媒在线免费看 | 国产精品自产拍在线观看中文 | av在线网站免费观看 | 日韩精品一区二区三区在线播放 | 狠狠操操操 | 黄色国产精品 | 日日狠狠 | 久久精品久久久久久久 | av日韩不卡 | 不卡中文字幕在线 | 国产精品久久二区 | 日日操天天操狠狠操 | 亚洲资源在线网 | 久草久视频 | 国产日韩欧美网站 | 亚洲人人网 | 综合黄色网 | 国产成人在线一区 | 久久久久一区二区三区 | 久久精品免费播放 | 成人精品国产 | 五月综合久久 | 丁香六月久久综合狠狠色 | 99久久精品免费看国产麻豆 | 欧美日本在线视频 | 在线视频日韩欧美 | 在线影院中文字幕 | 色多多污污在线观看 | 成人h在线播放 | 午夜一级免费电影 | 国产在线自 | 91香蕉视频污在线 | 在线成人一区二区 | 色综合久久综合网 | 日韩午夜电影 | 亚洲综合视频在线 | 亚洲精品动漫成人3d无尽在线 | 精品在线视频一区二区三区 | 国内精品久久久久久中文字幕 | 国产亚洲午夜高清国产拍精品 | 一区在线播放 | 亚洲精品一区二区久 | 成人免费 在线播放 | a级国产乱理论片在线观看 特级毛片在线观看 | 在线 国产一区 | 日本深夜福利视频 | 久久论理| 日韩理论在线观看 | 国产老妇av| 日韩中文字幕在线观看 | 激情五月播播久久久精品 | 国产一级视屏 | 超碰av在线播放 | 久久久99精品免费观看乱色 | 日韩免费观看一区二区 | 国产一区二区电影在线观看 | 黄色综合| 天天插综合网 | 一区二区精品久久 | 91人人爽久久涩噜噜噜 | 五月婷婷色丁香 | 久久成人免费视频 | 久久综合影院 | 国产亲近乱来精品 | 久久精品超碰 | 精品国产自在精品国产精野外直播 | 91精品国产综合久久福利不卡 | 久久精品中文字幕 | 久久www免费人成看片高清 | 午夜av色 | 天天爱天天插 | 精品视频在线免费观看 | 欧美与欧洲交xxxx免费观看 | 国产精品久久久久久久久婷婷 | 911久久 | 久久精品视频在线免费观看 | 超碰九九 | 手机成人在线电影 | 日韩免费看视频 | 久久 在线| 日韩欧美视频在线播放 | 草久久久久 | 久久毛片网 | 国产专区精品 | 国产色婷婷精品综合在线手机播放 | 欧美污污网站 | 日三级在线 | 亚洲精品在 | 国产精品久久久久av福利动漫 | 中文字幕在线免费观看 | www免费在线观看 | 久久久久久久久久久久久久av | 日日夜精品 | 亚洲婷婷在线 | 亚洲.www | 一区二区激情 | 国产激情免费 | 婷婷激情五月 | 国产小视频在线看 | 97视频精品| 色在线中文字幕 | 69久久夜色精品国产69 | 成人午夜精品久久久久久久3d | 国产午夜一区 | 五月天伊人网 | 欧美一二三视频 | 在线国产欧美 | 91精品爽啪蜜夜国产在线播放 | 91污视频在线 | 欧美日韩成人一区 | 久久国产精品小视频 | 欧美国产日韩在线观看 | 国产精品高清在线 | 丁香久久激情 | 亚洲精品午夜久久久久久久久久久 | 亚洲区视频在线观看 | 日本激情视频中文字幕 | av无限看 | 亚洲精品视频在 | 在线免费中文字幕 | 亚洲综合导航 | 日韩高清黄色 | 91久久精品一区二区三区 | 99精品视频网站 | av电影不卡在线 | 在线色亚洲 | 久久精品一区二区三区四区 | 最新成人av | 日本精品久久久一区二区三区 | 亚洲精品tv久久久久久久久久 | 91精品一区二区三区蜜臀 | 国产美女视频免费 | 成人一级影视 | 亚洲丝袜一区 | 精品国产一区二区三区噜噜噜 | 久久久久久高潮国产精品视 | 色香蕉在线视频 | 中文不卡视频 | 日韩精品电影在线播放 | 国产精品24小时在线观看 | 久久黄色网址 | 97夜夜澡人人双人人人喊 | 国产一区二区三区免费视频 | 超碰在线人人草 | 成人免费网站在线观看 | 婷婷在线五月 | 99人久久精品视频最新地址 | 国产视频精品免费播放 | 曰本三级在线 | 亚洲国产日韩在线 | 四虎影院在线观看av | 热久久视久久精品18亚洲精品 | 美女一二三区 | 国产精品免费一区二区三区 | 久久综合狠狠综合 | 一区二区三区免费在线观看视频 | 成年人在线看片 | 97精品国产97久久久久久春色 | 亚洲国产成人久久综合 | 男女免费av | 久久成电影| 国产黄色一级片在线 | 狠狠躁夜夜躁人人爽视频 | 91久久电影| 福利视频一区二区 | 超碰在线天天 | 少妇bbbb搡bbbb搡bbbb | av黄色在线播放 | 亚洲精品免费在线播放 | 精品播放 | 国产视频一区精品 | 中文字幕韩在线第一页 | 美女黄网站视频免费 | 久草在线观看 | 午夜精品视频免费在线观看 | 黄色小说在线免费观看 | 手机av电影在线 | 国产午夜精品理论片在线 | 亚洲精品视频二区 | 婷婷久久综合九色综合 | 午夜手机看片 | 中文字幕在线观看你懂的 | 亚洲欧美视屏 | 中文字幕有码在线观看 | 国色天香在线 | 国产精品欧美激情在线观看 | 中国一区二区视频 | 欧美极品一区二区三区 | 国产精品一区二区三区电影 | 免费在线观看一区二区三区 | 国产成人三级在线观看 | 在线高清av | 日韩色爱| 丁香综合 | 九九九视频精品 | 久久久免费看片 | 免费的国产精品 | 色搞搞 | 国产中文字幕91 | 成人毛片a | 射综合网 | av一级免费 | 中文字幕观看在线 | 成人黄色小说在线观看 | 国产盗摄精品一区二区 | 丁香六月激情 | 黄色小说18 | 人人躁 | 四虎天堂 | 9ⅰ精品久久久久久久久中文字幕 | 久久精品国产一区二区三 | 激情五月婷婷综合 | 日韩有码网站 | 97国产在线播放 | 高清美女视频 | 一区中文字幕在线观看 | 91精品国产乱码在线观看 | 久久观看最新视频 | 香蕉视频国产在线观看 | 欧美日韩一二三四区 | 色多视频在线观看 | 97精品国产97久久久久久 | 在线免费av观看 | 国产精品久久久久久久久久久免费看 | 免费看片成人 | 色综合国产 | 日韩欧美高清在线 | 久久免费国产视频 | 激情综合中文娱乐网 | 午夜久久 | 青春草免费视频 | 国产香蕉视频在线播放 | 欧美日韩久久久 | 亚洲精品久久久久中文字幕二区 | 国产在线一线 | 欧美综合色在线图区 | 国产精品色 | 日韩免费成人av | 天天色天天操综合网 | 视频二区在线视频 | 91私密保健| 日韩成人精品一区二区三区 | 日韩中文字幕国产 | 精品女同一区二区三区在线观看 | 色婷婷伊人 | 日本xxxx.com | 欧美先锋影音 | 96久久欧美麻豆网站 | 最近中文字幕高清字幕免费mv | av3级在线 | 亚洲最大激情中文字幕 | 亚洲国产精品一区二区久久,亚洲午夜 | 久草在线久 | 开心激情婷婷 | 国产成人在线观看免费 | 欧美一级淫片videoshd | av在线h| 最新日韩精品 | 国产成人av电影在线观看 | 成人av手机在线 | 久久久久久久久久久久久国产精品 | 欧美黑吊大战白妞欧美 | 天天干亚洲| 国产韩国日本高清视频 | 中文字幕视频免费观看 | 免费 在线 中文 日本 | 在线观看电影av | 久久黄网站 | 亚洲综合精品视频 | 亚洲国产精品va在线 | 视频二区在线 | 欧美最猛性xxxxx免费 | 日韩精品一区二区三区水蜜桃 | 久久久影院一区二区三区 | 色综合 久久精品 | 亚洲va欧美va国产va黑人 | 超碰在线公开免费 | 免费黄色a网站 | a v在线视频 | 91手机电影 | 国内精品久久久久久中文字幕 | 国产午夜在线 | 久久久久国产精品一区二区 | 麻豆免费视频网站 | 丁香婷五月| 中文区中文字幕免费看 | 精品久久国产一区 | 又爽又黄又刺激的视频 | 免费久久片 | 国色天香在线 | 亚洲成人免费 | 欧美精品一区二区在线观看 | 国产福利不卡视频 | 亚洲成人精品在线 | 精品一区二区免费视频 | 午夜精品久久一牛影视 | 在线免费黄色毛片 | 五月婷婷电影网 | 一区二区三区免费播放 | 狠狠干狠狠久久 | 天天拍天天色 | 一区二区三区免费在线观看 | 字幕网资源站中文字幕 | 18国产精品白浆在线观看免费 | 国产伦理一区二区三区 | a黄色片在线观看 | 精品久久1 | 国产精品成人一区二区三区 | 91中文字幕永久在线 | 国产91综合一区在线观看 | 精品超碰| 国产四虎在线 | 婷婷色中文网 | 亚洲激情在线视频 | 天天摸夜夜添 | 天堂网中文在线 | 午夜av在线播放 | av成人免费在线观看 | 亚洲另类xxxx | 精品国产一区二区三区蜜臀 | 狠狠干夜夜爱 | 91大神一区二区三区 | 国内一区二区视频 | 国产午夜精品理论片在线 | 欧美日韩国产在线精品 | 91精品视频免费在线观看 | 97超碰人人| 国产精品久久久久久久久久久久 | 国产免费一区二区三区最新 | 狠狠色伊人亚洲综合网站色 | 免费热情视频 | 久久综合影音 | 在线观看资源 | 欧美精品一区在线发布 | 中文字幕在线免费观看 | 国产成人精品久久亚洲高清不卡 | 久久精品影片 | 欧美在线视频二区 | 一区二区不卡在线观看 | 91av美女| 精品国产欧美 | 黄色免费网站大全 | 深夜福利视频在线观看 | 人人舔人人插 | 97看片网 | 天天综合视频在线观看 | av中文在线观看 | 国产精品毛片久久久久久久 | 色在线免费 | 亚洲综合成人婷婷小说 | 免费黄色网址大全 | 免费在线观看一区 | 91超在线 | 国产成人精品一区二区在线观看 | 精品一二 | 在线观看av中文字幕 | 精品久久综合 | 黄网站www | 在线不卡的av | 久久久久久久久网站 | 97超碰网 | 99久久99久久免费精品蜜臀 | 国产色拍拍拍拍在线精品 | 精品亚洲视频在线 | av在线免费在线 | 国产精品免费看 | 久久国产欧美日韩 | 亚洲成人二区 | 中文字幕区 | 这里只有精品视频在线 | 成人永久视频 | 国产日产欧美在线观看 | 久草视频在线免费看 | 日日操日日干 | 黄色在线免费观看网址 | 国产视频97| 最新av在线网站 | 福利视频 | 国产高清小视频 | 久久久网址 | 欧美色噜噜 | www.国产毛片 | 国产精品精品国产色婷婷 | 亚洲国产精品人久久电影 | 亚洲在线激情 | 日韩欧美在线观看一区 | 久久久久久久久久久免费视频 | 综合网婷婷 | 欧美一区二区三区免费看 | 日韩欧美高清不卡 | 日韩视频一区二区 | 久久xxxx | 国产一区高清在线 | 久久久精品小视频 | 五月婷婷丁香在线观看 | 免费成人av在线看 | 日韩高清在线一区二区三区 | 在线观看一区二区精品 | 97人人人人 | 日韩欧美国产成人 | 精品在线观看一区二区三区 | 亚洲最大激情中文字幕 | 国产精品一区二区美女视频免费看 | 久久爱导航 | 国产99在线 | 91香蕉视频720p | 天天骚夜夜操 | 黄色网www | 午夜影院三级 | 久久黄色美女 | 在线免费91 | 99久久精品国产毛片 | 奇人奇案qvod | 婷婷丁香色 | 丰满少妇在线观看资源站 | 国产精品毛片网 | 中文字幕在线播放日韩 | 黄视频色网站 | 中文字幕久久精品 | 日日色综合 | 六月色| 日韩高清毛片 | 9久久精品 | 精品国产乱码久久久久 | 亚洲黄色免费在线看 | 精品在线视频一区二区三区 | 中文字幕一区二区三区久久蜜桃 | 亚洲人久久久 | 日本精品在线看 | 天天艹天天干天天 | 欧美久久久久久 | 四虎成人精品永久免费av | 亚洲国产精品一区二区久久hs | 在线观看中文字幕dvd播放 | 69视频在线播放 | 综合色中色 | 综合色婷婷 | 激情视频在线高清看 | 91在线看视频免费 | 最近更新好看的中文字幕 | 天天操夜夜操天天射 | 蜜臀aⅴ国产精品久久久国产 | 精品人妖videos欧美人妖 | 人成免费网站 | 蜜桃视频日韩 | 午夜av在线电影 | 久久久男人的天堂 | 欧美精品二区 | 成人免费在线播放视频 | 9992tv成人免费看片 | 激情网色 | 国产成人一区二区在线观看 | 国产一区二区在线观看视频 | 丁香婷婷深情五月亚洲 | 日本公妇色中文字幕 | 成人三级视频 |