精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

了解Activity啟動過程,從startActivity到ATMS的高效協作

移動開發 Android
在Android的不同版本中,啟動流程可能會有所不同,并且可能涉及到更多的細節和組件。此外,如果啟動的是一個根Activity(例如從Launcher啟動),那么流程中可能還包括應用進程的創建等步驟。

在Android系統中,啟動一個Activity無論是從應用內部啟動Activity,還是通過桌面程序(Launcher)啟動,都需要通過調用startActivity方法來發起啟動請求。

啟動請求的發起

  1. 「應用內部啟動Activity」:

當應用內部需要啟動一個新的Activity時,開發者會調用startActivity方法,并傳遞一個包含目標Activity信息的Intent對象。

這個Intent對象可以指定要啟動的Activity的類名、傳遞的數據、附加的extras等。

調用startActivity后,系統會開始處理啟動請求,并按照Activity啟動流程進行后續操作。

  1. 「Launcher啟動Activity」:
  • Launcher是Android系統的桌面程序,它負責顯示已安裝的應用程序圖標,并提供用戶與應用程序交互的入口。

  • 當用戶從Launcher點擊一個應用程序圖標時,Launcher會創建一個新的Intent,用于啟動該應用程序的主Activity(通常是根Activity)。

  • 然后,Launcher調用startActivity方法,并將該Intent傳遞給系統,以啟動目標Activity。

在兩種情況下,啟動請求的發起都是通過調用startActivity方法實現的。這個方法會觸發一系列的操作,包括將啟動請求傳遞給ActivityTaskManagerService(ATMS),進行線程切換和消息處理,以及最終完成Activity的初始化和顯示。無論是startActivity還是startActivityForResult最終都是調用startActivityForResult。

圖片圖片

public class Activity extends ContextThemeWrapper
        implements LayoutInflater.Factory2,
        Window.Callback, KeyEvent.Callback,
        OnCreateContextMenuListener, ComponentCallbacks2,
        Window.OnWindowDismissedCallback,
        AutofillManager.AutofillClient, ContentCaptureManager.ContentCaptureClient {

    @Override
    public void startActivity(Intent intent, @Nullable Bundle options) {
        //...
        
        if (options != null) {
            startActivityForResult(intent, -1, options);
        } else {
            startActivityForResult(intent, -1);
        }
    }

    public void startActivityForResult(@RequiresPermission Intent intent, int requestCode,
            @Nullable Bundle options) {
        // mParent 是Activity類型,是當前Activity的父類
        if (mParent == null) {
            options = transferSpringboardActivityOptions(options);
            // 調用Instrumentation.execStartActivity啟動activity
            Instrumentation.ActivityResult ar =
                mInstrumentation.execStartActivity(
                    this, mMainThread.getApplicationThread(), mToken, this,
                    intent, requestCode, options);
            //...
            
        } else {
            //...

        }
    }
}

startActivityForResult中調用Instrumentation.execStartActivity方法。Activity中的mInstrumentation是在attach()方法中初始化,由ActivityThread傳入,其作用是通過遠程服務調用啟動activity,連接ActivityThread與activity,處理activity生命周期回調。

// Instrumentation主要用來監控應用程序和系統的交互,比如調用ATMS啟動activity,回調生命周期
public class Instrumentation {

    public ActivityResult execStartActivity(
        Context who, IBinder contextThread, IBinder token, Activity target,
        Intent intent, int requestCode, Bundle options) {
        //...

        try {
            //...
            
            // 通過ATMS遠程調用startActivity
            int result = ActivityTaskManager.getService().startActivity(whoThread,
                    who.getOpPackageName(), who.getAttributionTag(), intent,
                    intent.resolveTypeIfNeeded(who.getContentResolver()), token,
                    target != null ? target.mEmbeddedID : null, requestCode, 0, null, options);
            checkStartActivityResult(result, intent);
        } catch (RemoteException e) {
            throw new RuntimeException("Failure from system", e);
        }
        return null;
    }

    /**
     * 根據result判斷當前能否啟動activity,不能則拋出異常
     */
    public static void checkStartActivityResult(int res, Object intent) {
        if (!ActivityManager.isStartResultFatalError(res)) {
            return;
        }

        switch (res) {
            case ActivityManager.START_INTENT_NOT_RESOLVED:
            case ActivityManager.START_CLASS_NOT_FOUND:
                if (intent instanceof Intent && ((Intent)intent).getComponent() != null)
                    // 沒有在manifest中聲明
                    throw new ActivityNotFoundException(
                            "Unable to find explicit activity class "
                            + ((Intent)intent).getComponent().toShortString()
                            + "; have you declared this activity in your AndroidManifest.xml?");
                throw new ActivityNotFoundException(
                        "No Activity found to handle " + intent);
            case ActivityManager.START_PERMISSION_DENIED:
                throw new SecurityException("Not allowed to start activity "
                        + intent);
            case ActivityManager.START_FORWARD_AND_REQUEST_CONFLICT:
                throw new AndroidRuntimeException(
                        "FORWARD_RESULT_FLAG used while also requesting a result");
            case ActivityManager.START_NOT_ACTIVITY:
                throw new IllegalArgumentException(
                        "PendingIntent is not an activity");
            case ActivityManager.START_NOT_VOICE_COMPATIBLE:
                throw new SecurityException(
                        "Starting under voice control not allowed for: " + intent);
            case ActivityManager.START_VOICE_NOT_ACTIVE_SESSION:
                throw new IllegalStateException(
                        "Session calling startVoiceActivity does not match active session");
            case ActivityManager.START_VOICE_HIDDEN_SESSION:
                throw new IllegalStateException(
                        "Cannot start voice activity on a hidden session");
            case ActivityManager.START_ASSISTANT_NOT_ACTIVE_SESSION:
                throw new IllegalStateException(
                        "Session calling startAssistantActivity does not match active session");
            case ActivityManager.START_ASSISTANT_HIDDEN_SESSION:
                throw new IllegalStateException(
                        "Cannot start assistant activity on a hidden session");
            case ActivityManager.START_CANCELED:
                throw new AndroidRuntimeException("Activity could not be started for "
                        + intent);
            default:
                throw new AndroidRuntimeException("Unknown error code "
                        + res + " when starting " + intent);
        }
    }
}
@SystemService(Context.ACTIVITY_TASK_SERVICE)
public class ActivityTaskManager {
    /**
     * IActivityTaskManager是一個Binder,用于和system_server進程中的ActivityTaskManagerService通信
     */
    public static IActivityTaskManager getService() {
        return IActivityTaskManagerSingleton.get();
    }

    private static final Singleton<IActivityTaskManager> IActivityTaskManagerSingleton =
            new Singleton<IActivityTaskManager>() {
                @Override
                protected IActivityTaskManager create() {
                    final IBinder b = ServiceManager.getService(Context.ACTIVITY_TASK_SERVICE);
                    return IActivityTaskManager.Stub.asInterface(b);
                }
            };
}

當請求到達ATMS時,ATMS會首先檢查該請求是否合法,包括檢查Intent的有效性、權限等。一旦請求被驗證為有效,ATMS會進一步處理這個請求。

處理過程中,ATMS會根據當前系統的狀態和任務棧的情況來決定如何響應這個啟動請求。例如,它可能會決定創建一個新的Activity實例,或者將已存在的Activity實例帶到前臺。ATMS還會與ActivityManagerService(AMS)進行交互,以協調應用程序組件的生命周期管理。AMS負責跟蹤和管理這些組件的生命周期,確保它們按照預期的方式運行。

Activity的初始化與生命周期管理

當Activity啟動請求到達ActivityTaskManagerService(ATMS)并被驗證為有效后,ATMS會通知相應的應用進程進行Activity的初始化。

// /frameworks/base/services/core/java/com/android/server/wm/ActivityTaskManagerService.java
public class ActivityTaskManagerService extends IActivityTaskManager.Stub {

    public final int startActivity(IApplicationThread caller, String callingPackage,
      String callingFeatureId, Intent intent, String resolvedType, IBinder resultTo,
      String resultWho, int requestCode, int startFlags, ProfilerInfo profilerInfo,
      Bundle bOptions) {
      return startActivityAsUser(caller, callingPackage, callingFeatureId, intent, resolvedType,
              resultTo, resultWho, requestCode, startFlags, profilerInfo, bOptions,
              UserHandle.getCallingUserId());
    }

    private int startActivityAsUser(IApplicationThread caller, String callingPackage,
        @Nullable String callingFeatureId, Intent intent, String resolvedType,
        IBinder resultTo, String resultWho, int requestCode, int startFlags,
        ProfilerInfo profilerInfo, Bundle bOptions, int userId, boolean validateIncomingUser) {
        assertPackageMatchesCallingUid(callingPackage);
        // 判斷調用者進程是否被隔離
        enforceNotIsolatedCaller("startActivityAsUser");
        
        // 檢查調用者權限
        userId = getActivityStartController().checkTargetUser(userId, validateIncomingUser,
                Binder.getCallingPid(), Binder.getCallingUid(), "startActivityAsUser");


        return getActivityStartController().obtainStarter(intent, "startActivityAsUser")
                .setCaller(caller)
                .setCallingPackage(callingPackage)
                .setCallingFeatureId(callingFeatureId)
                .setResolvedType(resolvedType)
                .setResultTo(resultTo)
                .setResultWho(resultWho)
                .setRequestCode(requestCode)
                .setStartFlags(startFlags)
                .setProfilerInfo(profilerInfo)
                .setActivityOptions(bOptions)
                .setUserId(userId)
                .execute();
    }
}

ATMS通過一系列方法最終調到startActivityAsUser方法,先檢查調用者權限,再通過getActivityStartController().obtainStarter創建ActivityStarter類,把參數設置到ActivityStarter.Request類中,最后執行ActivityStarter.execute()方法。

// /frameworks/base/services/core/java/com/android/server/wm/ActivityStarter.java
class ActivityStarter {

    int execute() {
        try {
            //...

            int res;
            synchronized (mService.mGlobalLock) {
                //...
                
                res = executeRequest(mRequest);

                //...
            }
        } finally {
            onExecutionComplete();
        }
    }


    private int executeRequest(Request request) {
        // 判斷啟動的理由不為空
        if (TextUtils.isEmpty(request.reason)) {
            throw new IllegalArgumentException("Need to specify a reason.");
        }

        // 獲取調用的進程
        WindowProcessController callerApp = null;
        if (caller != null) {
          callerApp = mService.getProcessController(caller);
          if (callerApp != null) {
              // 獲取調用進程的pid和uid并賦值
              callingPid = callerApp.getPid();
              callingUid = callerApp.mInfo.uid;
          } else {
              err = ActivityManager.START_PERMISSION_DENIED;
          }
        }

        final int userId = aInfo != null && aInfo.applicationInfo != null
              ? UserHandle.getUserId(aInfo.applicationInfo.uid) : 0;

        ActivityRecord sourceRecord = null;
        ActivityRecord resultRecord = null;
        if (resultTo != null) {
          // 獲取調用者所在的ActivityRecord
          sourceRecord = mRootWindowContainer.isInAnyStack(resultTo);
          if (sourceRecord != null) {
              if (requestCode >= 0 && !sourceRecord.finishing) {
                  //requestCode = -1 則不進入
                  resultRecord = sourceRecord;
              }
          }
        }

        final int launchFlags = intent.getFlags();
        if ((launchFlags & Intent.FLAG_ACTIVITY_FORWARD_RESULT) != 0 && sourceRecord != null) {
           // activity執行結果的返回由源Activity轉換到新Activity, 不需要返回結果則不會進入該分支
        }

        if (err == ActivityManager.START_SUCCESS && intent.getComponent() == null) {
           // 從Intent中無法找到相應的Component
           err = ActivityManager.START_INTENT_NOT_RESOLVED;
        }

        if (err == ActivityManager.START_SUCCESS && aInfo == null) {
           // 從Intent中無法找到相應的ActivityInfo
           err = ActivityManager.START_CLASS_NOT_FOUND;
        }

        
        //執行后resultStack = null
        final ActivityStack resultStack = resultRecord == null
              ? null : resultRecord.getRootTask();

        //權限檢查
        if (mService.mController != null) {
            try {
                Intent watchIntent = intent.cloneFilter();
                abort |= !mService.mController.activityStarting(watchIntent,
                      aInfo.applicationInfo.packageName);
            } catch (RemoteException e) {
              mService.mController = null;
            }
        }

        if (abort) {
            //權限檢查不滿足,才進入該分支則直接返回;
            return START_ABORTED;
        

        if (aInfo != null) {
            if (mService.getPackageManagerInternalLocked().isPermissionsReviewRequired(
                  aInfo.packageName, userId)) {
             
                // 向PKMS獲取啟動Activity的ResolveInfo
                rInfo = mSupervisor.resolveIntent(intent, resolvedType, userId, 0,
                      computeResolveFilterUid(
                              callingUid, realCallingUid, request.filterCallingUid));
                // 向PKMS獲取啟動Activity的ActivityInfo
                aInfo = mSupervisor.resolveActivity(intent, rInfo, startFlags,
                      null /*profilerInfo*/);

            }
        }


        // 創建即將要啟動的Activity的描述類ActivityRecord
        final ActivityRecord r = new ActivityRecord(mService, callerApp, callingPid, callingUid,
              callingPackage, callingFeatureId, intent, resolvedType, aInfo,
              mService.getGlobalConfiguration(), resultRecord, resultWho, requestCode,
              request.componentSpecified, voiceSession != null, mSupervisor, checkedOptions,
              sourceRecord);
        mLastStartActivityRecord = r;


        // 調用 startActivityUnchecked
        mLastStartActivityResult = startActivityUnchecked(r, sourceRecord, voiceSession,
                request.voiceInteractor, startFlags, true /* doResume */, checkedOptions, inTask,
                restrictedBgActivity, intentGrants);

        if (request.outActivity != null) {
            request.outActivity[0] = mLastStartActivityRecord;
        }

        return mLastStartActivityResult;
    }
}

在ActivityStarter中調用executeRequest方法,先做一系列檢查,包括進程檢查、intent檢查、權限檢查、向PKMS獲取啟動Activity的ActivityInfo等信息,調用startActivityUnchecked方法開始對要啟動的activity進行任務棧管理。

private int startActivityUnchecked(final ActivityRecord r, ActivityRecord sourceRecord,
            IVoiceInteractionSession voiceSession, IVoiceInteractor voiceInteractor,
            int startFlags, boolean doResume, ActivityOptions options, Task inTask,
            boolean restrictedBgActivity, NeededUriGrants intentGrants) {
    

    try {

        result = startActivityInner(r, sourceRecord, voiceSession, voiceInteractor,
                startFlags, doResume, options, inTask, restrictedBgActivity, intentGrants);
    } finally {
        
        //...
    }

    //...

    return result;
}

ActivityRecord mStartActivity;

private ActivityStack mSourceStack;
private ActivityStack mTargetStack;
private Task mTargetTask;


// 主要處理棧管理相關的邏輯
int startActivityInner(final ActivityRecord r, ActivityRecord sourceRecord,
                     IVoiceInteractionSession voiceSession, IVoiceInteractor voiceInteractor,
                     int startFlags, boolean doResume, ActivityOptions options, Task inTask,
                     boolean restrictedBgActivity, NeededUriGrants intentGrants) {
    // 初始化啟動Activity的各種配置,在初始化前會重置各種配置再進行配置,
    // 這些配置包括:ActivityRecord、Intent、Task和LaunchFlags(啟動的FLAG)等等
    setInitialState(r, options, inTask, doResume, startFlags, sourceRecord, voiceSession,
          voiceInteractor, restrictedBgActivity);

    // 給不同的啟動模式計算出mLaunchFlags
    computeLaunchingTaskFlags();

    // 主要作用是設置ActivityStack
    computeSourceStack();

    // 將mLaunchFlags設置給Intent
    mIntent.setFlags(mLaunchFlags);

    // 確定是否應將新活動插入現有任務。如果不是,則返回null,
    // 或者返回帶有應將新活動添加到其中的任務的ActivityRecord。
    final Task reusedTask = getReusableTask();


    // 如果reusedTask為null,則計算是否存在可以使用的任務棧
    final Task targetTask = reusedTask != null ? reusedTask : computeTargetTask();
    final boolean newTask = targetTask == null; // 啟動Activity是否需要新創建棧
    mTargetTask = targetTask;

    computeLaunchParams(r, sourceRecord, targetTask);

    // 檢查是否允許在給定任務或新任務上啟動活動。
    int startResult = isAllowedToStart(r, newTask, targetTask);
    if (startResult != START_SUCCESS) {
        return startResult;
    }


    final ActivityStack topStack = mRootWindowContainer.getTopDisplayFocusedStack();
    if (topStack != null) {
        // 檢查正在啟動的活動是否與當前位于頂部的活動相同,并且應該只啟動一次
        startResult = deliverToCurrentTopIfNeeded(topStack, intentGrants);
        if (startResult != START_SUCCESS) {
            return startResult;
        }
    }

    if (mTargetStack == null) {
        // 復用或者創建堆棧
        mTargetStack = getLaunchStack(mStartActivity, mLaunchFlags, targetTask, mOptions);
    }
    if (newTask) {
        // 新建一個task
        final Task taskToAffiliate = (mLaunchTaskBehind && mSourceRecord != null)
              ? mSourceRecord.getTask() : null;
        setNewTask(taskToAffiliate);
        if (mService.getLockTaskController().isLockTaskModeViolation(
              mStartActivity.getTask())) {
            Slog.e(TAG, "Attempted Lock Task Mode violation mStartActivity=" + mStartActivity);
            return START_RETURN_LOCK_TASK_MODE_VIOLATION;
        }
    } else if (mAddingToTask) {
        // 復用之前的task
        addOrReparentStartingActivity(targetTask, "adding to task");
    }

    

    // 檢查是否需要觸發過渡動畫和開始窗口
    mTargetStack.startActivityLocked(mStartActivity,
          topStack != null ? topStack.getTopNonFinishingActivity() : null, newTask,
          mKeepCurTransition, mOptions);


    if (mDoResume) {

        // 調用RootWindowContainer的resumeFocusedStacksTopActivities方法
        mRootWindowContainer.resumeFocusedStacksTopActivities(
              mTargetStack, mStartActivity, mOptions);
    }


    return START_SUCCESS;
}


private void setInitialState(ActivityRecord r, ActivityOptions options, Task inTask,
                           boolean doResume, int startFlags, ActivityRecord sourceRecord,
                           IVoiceInteractionSession voiceSession, IVoiceInteractor voiceInteractor,
                           boolean restrictedBgActivity) {

    reset(false /* clearRequest */); 
    mStartActivity = r; 
    mIntent = r.intent; 
    mSourceRecord = sourceRecord; 
    mLaunchMode = r.launchMode; 
    // 啟動Flags
    mLaunchFlags = adjustLaunchFlagsToDocumentMode(
          r, LAUNCH_SINGLE_INSTANCE == mLaunchMode,
          LAUNCH_SINGLE_TASK == mLaunchMode, mIntent.getFlags());
    mInTask = inTask;
    // ...
}


private void computeLaunchingTaskFlags() {
    if (mInTask == null) {
        if (mSourceRecord == null) {
            if ((mLaunchFlags & FLAG_ACTIVITY_NEW_TASK) == 0 && mInTask == null) {
                mLaunchFlags |= FLAG_ACTIVITY_NEW_TASK;
            }
        } else if (mSourceRecord.launchMode == LAUNCH_SINGLE_INSTANCE) {
            mLaunchFlags |= FLAG_ACTIVITY_NEW_TASK;
        } else if (isLaunchModeOneOf(LAUNCH_SINGLE_INSTANCE, LAUNCH_SINGLE_TASK)) {
            mLaunchFlags |= FLAG_ACTIVITY_NEW_TASK;
        }
    }
}

// 設置ActivityStack
private void computeSourceStack() {
    if (mSourceRecord == null) {
        mSourceStack = null;
        return;
    }
    if (!mSourceRecord.finishing) {
        mSourceStack = mSourceRecord.getRootTask();
        return;
    }

    if ((mLaunchFlags & FLAG_ACTIVITY_NEW_TASK) == 0) {
        mLaunchFlags |= FLAG_ACTIVITY_NEW_TASK;
        mNewTaskInfo = mSourceRecord.info;

        final Task sourceTask = mSourceRecord.getTask();
        mNewTaskIntent = sourceTask != null ? sourceTask.intent : null;
    }
    mSourceRecord = null;
    mSourceStack = null;
}

private Task getReusableTask() {
    // If a target task is specified, try to reuse that one
    if (mOptions != null && mOptions.getLaunchTaskId() != INVALID_TASK_ID) {
        Task launchTask = mRootWindowContainer.anyTaskForId(mOptions.getLaunchTaskId());
        if (launchTask != null) {
            return launchTask;
        }
        return null;
    }

    //標志位,如果為true,說明要放入已經存在的棧,
    // 可以看出,如果是設置了FLAG_ACTIVITY_NEW_TASK 而沒有設置 FLAG_ACTIVITY_MULTIPLE_TASK,
    // 或者設置了singleTask以及singleInstance
    boolean putIntoExistingTask = ((mLaunchFlags & FLAG_ACTIVITY_NEW_TASK) != 0 &&
            (mLaunchFlags & FLAG_ACTIVITY_MULTIPLE_TASK) == 0)
            || isLaunchModeOneOf(LAUNCH_SINGLE_INSTANCE, LAUNCH_SINGLE_TASK);
    // 重新檢驗
    putIntoExistingTask &= mInTask == null && mStartActivity.resultTo == null;
    ActivityRecord intentActivity = null;
    if (putIntoExistingTask) {
        if (LAUNCH_SINGLE_INSTANCE == mLaunchMode) {
            //如果是 singleInstance,那么就找看看之前存在的該實例,找不到就為null
            intentActivity = mRootWindowContainer.findActivity(mIntent, mStartActivity.info,
                    mStartActivity.isActivityTypeHome());
        } else if ((mLaunchFlags & FLAG_ACTIVITY_LAUNCH_ADJACENT) != 0) {
            // For the launch adjacent case we only want to put the activity in an existing
            // task if the activity already exists in the history.
            intentActivity = mRootWindowContainer.findActivity(mIntent, mStartActivity.info,
                    !(LAUNCH_SINGLE_TASK == mLaunchMode));
        } else {
            // Otherwise find the best task to put the activity in.
            intentActivity =
                    mRootWindowContainer.findTask(mStartActivity, mPreferredTaskDisplayArea);
        }
    }

    if (intentActivity != null
            && (mStartActivity.isActivityTypeHome() || intentActivity.isActivityTypeHome())
            && intentActivity.getDisplayArea() != mPreferredTaskDisplayArea) {
        // Do not reuse home activity on other display areas.
        intentActivity = null;
    }

    return intentActivity != null ? intentActivity.getTask() : null;
}

// 計算啟動的Activity的棧
private Task computeTargetTask() {
    if (mStartActivity.resultTo == null && mInTask == null && !mAddingToTask
            && (mLaunchFlags & FLAG_ACTIVITY_NEW_TASK) != 0) {
        // 返回null,應該新創建一個Task,而不是使用現有的Task
        return null;
    } else if (mSourceRecord != null) {
        // 使用源Activity的task
        return mSourceRecord.getTask();
    } else if (mInTask != null) {
        // 使用啟動時傳遞的task
        return mInTask;
    } else {
        // 理論上的可能,不可能走到這里
        final ActivityStack stack = getLaunchStack(mStartActivity, mLaunchFlags,
                null /* task */, mOptions);
        final ActivityRecord top = stack.getTopNonFinishingActivity();
        if (top != null) {
            return top.getTask();
        } else {
            // Remove the stack if no activity in the stack.
            stack.removeIfPossible();
        }
    }
    return null;
}

在startActivityInner方法中,根據啟動模式和flag等條件判斷要啟動的activity的ActivityRecord是加入現有的Task棧中或創建新的Task棧。在為Activity準備好Task棧后,調用RootWindowContainer.resumeFocusedStacksTopActivities方法。

class RootWindowContainer extends WindowContainer<DisplayContent>
      implements DisplayManager.DisplayListener {

    boolean resumeFocusedStacksTopActivities(
            ActivityStack targetStack, ActivityRecord target, ActivityOptions targetOptions) {

        //...
        boolean result = false;
        if (targetStack != null && (targetStack.isTopStackInDisplayArea()
                || getTopDisplayFocusedStack() == targetStack)) {
            // 調用ActivityStack.resumeTopActivityUncheckedLocked
            result = targetStack.resumeTopActivityUncheckedLocked(target, targetOptions);
        }
        //...
        return result;
    }
}

class ActivityStack extends Task {

    boolean resumeTopActivityUncheckedLocked(ActivityRecord prev, ActivityOptions options) {
      if (mInResumeTopActivity) {
          // Don't even start recurscheduleTransactionsing.
          return false;
      }

      boolean result = false;
      try {
          mInResumeTopActivity = true;
          // 繼續調用resumeTopActivityInnerLocked
          result = resumeTopActivityInnerLocked(prev, options);

          final ActivityRecord next = topRunningActivity(true /* focusableOnly */);
          if (next == null || !next.canTurnScreenOn()) {
              checkReadyForSleep();
          }
      } finally {
          mInResumeTopActivity = false;
      }

      return result;
  }

  private boolean resumeTopActivityInnerLocked(ActivityRecord prev, ActivityOptions options) {

      // Find the next top-most activity to resume in this stack that is not finishing and is
      // focusable. If it is not focusable, we will fall into the case below to resume the
      // top activity in the next focusable task.
      // 在當前Task棧中找到最上層正在運行的activity,如果這個activity沒有獲取焦點,那這個activity將會被重新啟動
      ActivityRecord next = topRunningActivity(true /* focusableOnly */);
      final boolean hasRunningActivity = next != null;

      if (next.attachedToProcess()) {
          ...
      } else {
          ...
          // 調用StackSupervisor.startSpecificActivity
          mStackSupervisor.startSpecificActivity(next, true, true);
      }
      return true;
  }
}

ActivityStack用于單個活動棧的管理,最終調到ActivityStackSupervisor.startSpecificActivity()。

public class ActivityStackSupervisor implements RecentTasks.Callbacks {

    // 檢查啟動Activity所在進程是否有啟動,沒有則先啟動進程
    void startSpecificActivity(ActivityRecord r, boolean andResume, boolean checkConfig) {
        // 根據processName和Uid查找啟動Activity的所在進程
        final WindowProcessController wpc =
                mService.getProcessController(r.processName, r.info.applicationInfo.uid);

        boolean knownToBeDead = false;

        if (wpc != null && wpc.hasThread()) {
            // 進程已經存在,則直接啟動Activity
            try {
                // 啟動Activity ,并返回
                realStartActivityLocked(r, wpc, andResume, checkConfig);
                return;
            } catch (RemoteException e) {
               ...
            }
            knownToBeDead = true;
        }

        // 所在進程沒創建則調用ATMS.startProcessAsync創建進程并啟動Activity
        mService.startProcessAsync(r, knownToBeDead, isTop, isTop ? "top-activity" : "activity");
    }

    // 啟動Activity的進程存在,則執行此方法 
    boolean realStartActivityLocked(ActivityRecord r, WindowProcessController proc,
                                    boolean andResume, boolean checkConfig) throws RemoteException {

        ...

        // 創建活動啟動事務
        // proc.getThread()是一個IApplicationThread對象,可以通過ClientTransaction.getClient()獲取
        final ClientTransaction clientTransaction = ClientTransaction.obtain(
                proc.getThread(), r.appToken);

        // 為事務設置Callback,為LaunchActivityItem,在客戶端時會被調用
        clientTransaction.addCallback(LaunchActivityItem.obtain(new Intent(r.intent),
                System.identityHashCode(r), r.info,
                // TODO: Have this take the merged configuration instead of separate global
                // and override configs.
                mergedConfiguration.getGlobalConfiguration(),
                mergedConfiguration.getOverrideConfiguration(), r.compat,
                r.launchedFromPackage, task.voiceInteractor, proc.getReportedProcState(),
                r.getSavedState(), r.getPersistentSavedState(), results, newIntents,
                dc.isNextTransitionForward(), proc.createProfilerInfoIfNeeded(),
                r.assistToken, r.createFixedRotationAdjustmentsIfNeeded()));

        // 設置所需的最終狀態
        final ActivityLifecycleItem lifecycleItem;
        if (andResume) {
            lifecycleItem = ResumeActivityItem.obtain(dc.isNextTransitionForward());
        } else {
            lifecycleItem = PauseActivityItem.obtain();
        }
        clientTransaction.setLifecycleStateRequest(lifecycleItem);

        // 執行事件,調用ClientLifecycleManager.scheduleTransaction
        mService.getLifecycleManager().scheduleTransaction(clientTransaction);

        ...
        return true;
    }
}

在ActivityStackSupervisor中,先檢查要啟動activity的進程是否存在,存在則調用realStartActivityLocked方法,通過ClientTransaction事務回調ApplicationThread.scheduleTransaction方法;進程不存在則創建進程。

Activity的顯示

// 主要是處理AMS端的請求
private class ApplicationThread extends IApplicationThread.Stub {
    @Override
    public void scheduleTransaction(ClientTransaction transaction) throws RemoteException {
        ActivityThread.this.scheduleTransaction(transaction);
    }
}

// ActivityThread的父類
public abstract class ClientTransactionHandler {

    void scheduleTransaction(ClientTransaction transaction) {
        transaction.preExecute(this);
        // 發送EXECUTE_TRANSACTION消息
        sendMessage(ActivityThread.H.EXECUTE_TRANSACTION, transaction);
    }
}

// 它管理應用程序進程中主線程的執行,根據活動管理器的請求,在其上調度和執行活動、廣播和其他操作。
public final class ActivityThread extends ClientTransactionHandler {

    class H extends Handler {
        public void handleMessage(Message msg) {
            switch (msg.what) {
                case EXECUTE_TRANSACTION:
                    final ClientTransaction transaction = (ClientTransaction) msg.obj;
                    // 調用TransactionExecutor.execute去處理ATMS階段傳過來的ClientTransaction
                    mTransactionExecutor.execute(transaction);
                    if (isSystem()) {
                        // Client transactions inside system process are recycled on the client side
                        // instead of ClientLifecycleManager to avoid being cleared before this
                        // message is handled.
                        transaction.recycle();
                    }
                    break;
            }
        }
    }
}

public class TransactionExecutor {
    public void execute(ClientTransaction transaction) {
        //...
        // 調用傳過來的ClientTransaction事務的Callback
        executeCallbacks(transaction);

        executeLifecycleState(transaction);
    }

    public void executeCallbacks(ClientTransaction transaction) {
        final List<ClientTransactionItem> callbacks = transaction.getCallbacks();
        ...
        final int size = callbacks.size();
        for (int i = 0; i < size; ++i) {
            final ClientTransactionItem item = callbacks.get(i);
            ...
            // 調用LaunchActivityItem.execute
            item.execute(mTransactionHandler, token, mPendingActions);
            item.postExecute(mTransactionHandler, token, mPendingActions);
            ...
        }
    }
}

public class LaunchActivityItem extends ClientTransactionItem {
    public void execute(ClientTransactionHandler client, IBinder token,
                        PendingTransactionActions pendingActions) {
        ActivityClientRecord r = new ActivityClientRecord(token, mIntent, mIdent, mInfo,
                mOverrideConfig, mCompatInfo, mReferrer, mVoiceInteractor, mState, mPersistentState,
                mPendingResults, mPendingNewIntents, mIsForward,
                mProfilerInfo, client, mAssistToken, mFixedRotationAdjustments);
        // 調用ActivityThread.handleLaunchActivity
        client.handleLaunchActivity(r, pendingActions, null /* customIntent */);
    }
}

ApplicationThread最后調用在ATMS階段設置的ClientTransaction的CallBack的execute方法,即LaunchActivityItem.execute方法,此時創建一個ActivityClientRecord對象,然后通過ActivityThread.handleLaunchActivity開啟真正的Actvity啟動。

public final class ActivityThread extends ClientTransactionHandler {
    // ActivityThread啟動Activity的過程
    @Override
    public Activity handleLaunchActivity(ActivityClientRecord r,
                                         PendingTransactionActions pendingActions, Intent customIntent) {
        // ...
        WindowManagerGlobal.initialize();
        // 啟動Activity
        final Activity a = performLaunchActivity(r, customIntent);
    
        if (a != null) {
           ...
        } else {
            // 啟動失敗,調用ATMS停止Activity啟動
            try {
                ActivityTaskManager.getService()
                        .finishActivity(r.token, Activity.RESULT_CANCELED, null,
                                Activity.DONT_FINISH_TASK_WITH_ACTIVITY);
            } catch (RemoteException ex) {
                throw ex.rethrowFromSystemServer();
            }
        }
    
    return a;
}

private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) {
    // 獲取ActivityInfo類
    ActivityInfo aInfo = r.activityInfo;
    if (r.packageInfo == null) {
        // 獲取APK文件的描述類LoadedApk
        r.packageInfo = getPackageInfo(aInfo.applicationInfo, r.compatInfo,
                Context.CONTEXT_INCLUDE_CODE);
    }
    // 啟動的Activity的ComponentName類
    ComponentName component = r.intent.getComponent();

    // 創建要啟動Activity的上下文環境
    ContextImpl appContext = createBaseContextForActivity(r);
    Activity activity = null;
    try {
        java.lang.ClassLoader cl = appContext.getClassLoader();
        // 用類加載器來創建該Activity的實例
        activity = mInstrumentation.newActivity(
                cl, component.getClassName(), r.intent);
        // ...
    } catch (Exception e) {
        // ...
    }

    try {
        // 創建Application
        Application app = r.packageInfo.makeApplication(false, mInstrumentation);

        if (activity != null) {
            // 初始化Activity
            activity.attach(appContext, this, getInstrumentation(), r.token,
                    r.ident, app, r.intent, r.activityInfo, title, r.parent,
                    r.embeddedID, r.lastNonConfigurationInstances, config,
                    r.referrer, r.voiceInteractor, window, r.configCallback,
                    r.assistToken);

            ...
            // 調用Instrumentation的callActivityOnCreate方法來啟動Activity
            if (r.isPersistable()) {
                mInstrumentation.callActivityOnCreate(activity, r.state, r.persistentState);
            } else {
                mInstrumentation.callActivityOnCreate(activity, r.state);
            }
          ...
        }
      ...

    } catch (SuperNotCalledException e) {
        throw e;

    } catch (Exception e) {
        ...
    }

    return activity;
}


public class Instrumentation {

    public void callActivityOnCreate(Activity activity, Bundle icicle) {
        prePerformCreate(activity);
        // 調用Activity的performCreate
        activity.performCreate(icicle);
        postPerformCreate(activity);
    }
}

public class Activity extends ContextThemeWrapper
      implements LayoutInflater.Factory2,
      Window.Callback, KeyEvent.Callback,
      OnCreateContextMenuListener, ComponentCallbacks2,
      Window.OnWindowDismissedCallback,
      AutofillManager.AutofillClient, ContentCaptureManager.ContentCaptureClient {

    final void performCreate(Bundle icicle) {
        performCreate(icicle, null);
    }

    @UnsupportedAppUsage
    final void performCreate(Bundle icicle, PersistableBundle persistentState) {
        ...
        // 調用onCreate方法
        if (persistentState != null) {
            onCreate(icicle, persistentState);
        } else {
            onCreate(icicle);
        }
        ...
    }

}

當初始化完成后,Activity的界面會被繪制并顯示到屏幕上。此時,用戶可以與Activity進行交互。

總結

  1. 「啟動請求的發起」:

無論是從應用內部啟動Activity,還是通過桌面程序(Launcher)啟動,都需要通過調用startActivity方法來發起啟動請求。

這個請求包含了要啟動的Activity的信息,通常通過Intent對象來傳遞。

  1. 「請求到達ActivityTaskManagerService(ATMS)」:
  • 當啟動請求被發起后,它會首先到達ActivityTaskManagerService(ATMS)。

  • ATMS是負責管理Activity生命周期和任務棧的系統服務。

  1. 「線程切換與消息處理」:

  • 在ATMS處理啟動請求的過程中,可能涉及到線程切換和消息處理。

  • 例如,將請求從應用線程切換到系統服務線程,或者通過消息隊列來處理請求。

  1. 「Activity的初始化與生命周期管理」:

  • 一旦ATMS決定了要啟動哪個Activity,它會通知相應的應用進程進行Activity的初始化。

  • 這包括創建Activity的實例、加載布局、初始化組件等。

  • 同時,Activity的生命周期方法(如onCreate、onStart、onResume等)也會被調用,以確保Activity的正確初始化和狀態管理。

  1. 「Activity的顯示」:

  • 當初始化完成后,Activity的界面會被繪制并顯示到屏幕上。

  • 此時,用戶可以與Activity進行交互。

在Android的不同版本中,啟動流程可能會有所不同,并且可能涉及到更多的細節和組件。此外,如果啟動的是一個根Activity(例如從Launcher啟動),那么流程中可能還包括應用進程的創建等步驟。Activity的啟動流程是一個涉及多個組件和服務的復雜過程,它確保了Android應用程序能夠正確地創建、初始化和顯示Activity,從而為用戶提供流暢和一致的體驗。

責任編輯:武曉燕 來源: 沐雨花飛蝶
相關推薦

2023-03-24 14:52:27

AbilityPage應用

2014-06-23 10:31:09

Android啟動過程

2011-09-05 17:35:18

MTK啟動過程RTOS

2011-07-28 10:34:38

Cocoa 程序 啟動

2021-07-02 06:34:53

Go語言sysmon

2011-06-28 13:27:13

ARM Linux

2009-12-03 10:00:46

Linux系統啟動

2010-07-05 17:38:39

IIS 7.0 FTP

2012-08-16 09:07:57

Erlang

2018-03-13 13:00:03

Linux運維啟動分析

2012-02-20 14:47:08

JavaPlay

2023-12-09 08:58:30

AndroidActivity啟動模式

2024-09-11 09:25:03

Tomcat組件PREP

2010-05-06 14:05:15

Unix系統

2021-09-28 15:03:06

Linux內核arm

2020-03-19 08:59:15

SpringMVC啟動過程

2019-05-27 14:43:49

Tomcat架構部署

2018-10-18 14:06:15

Linux系統過程

2020-04-20 21:30:51

Tomcat部署架構

2009-08-11 09:03:45

Windows 7系統啟動
點贊
收藏

51CTO技術棧公眾號

超碰在线免费97| 国产二区视频在线| 波多野结衣视频观看| 成人激情诱惑| 日韩欧美一区二区三区在线| 久久这里只有精品8| 五月婷婷久久久| 老司机精品视频在线| 九九热精品视频在线播放| 人妻体内射精一区二区三区| 欧美free嫩15| 亚洲综合视频在线| 日韩久久精品一区二区三区| 精品国产乱码一区二区三| 一区二区国产精品| 色悠悠久久久久| theav精尽人亡av| 国产不卡精品| 色94色欧美sute亚洲线路一久| 成人性做爰片免费视频| 亚洲欧洲国产综合| 国产精品一区久久久久| 欧美性视频在线| 久久免费看少妇高潮v片特黄| 网曝91综合精品门事件在线| 日韩欧美一级精品久久| 日本久久精品一区二区| 波多野结衣在线播放| 国产精品久久久久久久久动漫| 免费日韩电影在线观看| av资源免费看| 免费成人美女在线观看.| 午夜精品在线观看| 国产精品免费人成网站酒店| 狠狠操综合网| 国产视频在线一区二区| 亚洲少妇中文字幕| 91精品福利观看| 欧美日韩一卡二卡| 国产一区二区视频免费在线观看| sm久久捆绑调教精品一区| 亚洲人成亚洲人成在线观看图片| 久久久一本精品99久久精品66| 99热这里只有精品3| 毛片一区二区三区| 国产精品福利在线观看| 欧美黑人一区二区| 日韩视频精品在线观看| 久久99国产综合精品女同 | 国产欧美啪啪| 日韩欧美二区三区| 黄色a级三级三级三级| 激情小说亚洲| 欧美三级三级三级爽爽爽| 少妇人妻互换不带套| 日韩伦理福利| 日韩欧美在线观看视频| 日韩av片在线看| 一区二区三区短视频| 天天综合网 天天综合色| 亚洲 欧美 综合 另类 中字| 青草青在线视频| 亚洲主播在线播放| 福利视频一区二区三区四区| brazzers在线观看| 黄色精品一区二区| 欧美亚洲另类色图| 三级成人在线| 欧美三级电影网| 一个色综合久久| 日本久久伊人| 亚洲国产精品99久久| av无码一区二区三区| 欧美亚视频在线中文字幕免费| 亚洲精品国产精品国自产观看浪潮 | 亚洲国产成人在线播放| 东京热av一区| 伊人久久综合影院| 中文字幕亚洲在线| 草视频在线观看| 亚洲国产一区二区三区高清| 欧美有码在线视频| 成人毛片一区二区三区| 久久99久久久欧美国产| 成人免费视频网站| 欧美高清成人| 中文字幕在线观看不卡视频| 国产高清www| 日韩久久一区二区三区| 91精品国产手机| 国产一级免费片| 精品久久久久久久| 久久影视电视剧免费网站| 国产无码精品一区二区| 日韩在线播放一区二区| 成人午夜激情免费视频| 欧美自拍偷拍一区二区| 亚洲国产精品成人久久综合一区| 免费观看中文字幕| 欧美aa在线| 欧美久久久久免费| 国产又黄又粗又猛又爽的视频| 日韩成人三级| 26uuu另类亚洲欧美日本一| 黄色片网站在线免费观看| 激情偷乱视频一区二区三区| 精品一区二区国产| 麻豆传媒在线免费| 欧美日韩免费看| 中文字幕第10页| 国产一区二区三区四区| 久久久久久久久久久人体| 中文字幕av网站| 成人免费毛片片v| 亚洲一区二区在线观| 女人高潮被爽到呻吟在线观看 | 开心激情综合网| 中文字幕av资源一区| 又粗又黑又大的吊av| 国产精品亚洲欧美一级在线| 亚洲日韩第一页| 久草视频在线资源| 久久66热偷产精品| 欧美精品在线一区| 波多野结衣在线播放| 91精品国产综合久久精品性色| 中文字幕第20页| 国产精品久久久免费| 亚洲自拍偷拍区| 97超碰国产一区二区三区| 日韩人在线观看| 国产极品一区二区| 国色天香一区二区| 成人黄色午夜影院| 77777影视视频在线观看| 色综合网色综合| 中文字幕无人区二| 一区二区三区在线电影| 国产精品一区二区三区毛片淫片| 免费在线观看一级毛片| 黄网动漫久久久| 美女又爽又黄免费| 国色天香一区二区| 国产精品日韩高清| 男男gaygays亚洲| 日韩精品中文字幕一区| 午夜精品福利在线视频| 国内久久婷婷综合| 伊人久久青草| 国产精品美女久久久久| 日韩一区二区精品视频| 亚洲综合精品国产一区二区三区 | 精品一二三四在线| 亚洲欧洲国产日韩精品| 丁香久久综合| 日韩视频免费在线观看| 国产精品乱码久久久| 综合中文字幕亚洲| 久久久久xxxx| 欧美激情一区| 精品国产一区二区三区四区精华| 99色在线观看| 亚洲欧洲一区二区三区在线观看| aaaaaa毛片| 国产欧美一区二区三区在线老狼| 自拍偷拍 国产| 999精品色在线播放| 亚洲va男人天堂| 在线中文字幕第一页| 欧美不卡一区二区| 日韩免费在线视频观看| 久久久久久久久蜜桃| 国产wwwxx| 99精品视频在线观看免费播放| 亚洲aⅴ男人的天堂在线观看| 中文字幕在线三区| 亚洲国产精品国自产拍av秋霞| av资源免费观看| 国产精品午夜免费| 在线一区二区不卡| 99视频一区| 日韩久久不卡| 在线播放一区二区精品视频| 8x拔播拔播x8国产精品| av在线日韩国产精品| 欧美一区二区三区性视频| 国产精品日日夜夜| 久久精品欧美一区二区三区麻豆| 不用播放器的免费av| 红桃视频国产一区| 日本一区二区三区精品视频| www.久久久久爱免| 欧美一级bbbbb性bbbb喷潮片| 青青影院在线观看| 亚洲国产成人精品女人久久久 | 麻豆视频在线免费看| a级高清视频欧美日韩| 男人添女人下面免费视频| 中文字幕激情视频| 国一区二区在线观看| 欧美另类一区| 国产日韩欧美中文在线| 5566成人精品视频免费| 午夜免费福利在线观看| 亚洲福利视频网| 在线亚洲欧美日韩| 精品久久久久久久中文字幕| 青青操在线视频观看| 91看片淫黄大片一级在线观看| 狠狠干狠狠操视频| 久久九九国产| 日本午夜激情视频| 欧美3p在线观看| 久久人人九九| 日韩精品一区二区三区中文字幕 | 97久久精品一区二区三区的观看方式| 午夜精品一区二区三区在线视| 麻豆传媒在线免费看| 亚洲色图综合网| 日韩在线视频观看免费| 91麻豆精品国产91久久久久久久久| 91玉足脚交嫩脚丫在线播放| 亚洲一二三四区| 欧美日韩色视频| 欧美激情中文不卡| 极品人妻一区二区三区| 成人免费高清在线观看| 久久精品亚洲天堂| 精品免费久久久| 久久久99精品免费观看不卡| av天堂一区二区| 国产在线日韩欧美| 91 在线视频观看| 久久最新视频| 乱子伦视频在线看| 国产欧美在线| 国产午夜大地久久| 亚洲经典视频在线观看| 免费在线精品视频| 99精品国产一区二区三区| 午夜欧美性电影| 欧美日韩在线二区| 日韩欧美亚洲区| 国产区精品区| 欧美一区国产一区| 中日韩免视频上线全都免费| 精品国产免费久久久久久尖叫| 六月丁香久久丫| 国产精品有限公司| 国产伦精品一区二区三区免费优势| 爱情岛论坛亚洲入口| 成人av色网站| 国产精品揄拍500视频| av亚洲一区| 国产精品一区二区3区| 国产a亚洲精品| 国产在线拍偷自揄拍精品| 麻豆久久久久| 亚洲一区亚洲二区| 久久伊人影院| 国产精品一区二区在线观看| 久久午夜影院| 欧美日韩高清免费| 成人免费av| 一区二区精品在线观看| 亚洲天天影视网| 日韩成人手机在线| 国产欧美成人| 杨幂毛片午夜性生毛片| 免费视频一区二区| 涩多多在线观看| 成人综合在线观看| 99re久久精品国产| 久久精品亚洲一区二区三区浴池| 99久久99久久精品免费| 亚洲手机成人高清视频| 欧美成人精品欧美一级| 天天做天天摸天天爽国产一区| 7799精品视频天天看| 欧美三级在线视频| 国内精品久久久久久久久久| 亚洲成人网在线| 国产福利小视频在线| 日韩中文字幕视频| 欧美寡妇性猛交xxx免费| 欧美中文字幕视频| 日本欧美在线| 狠狠久久综合婷婷不卡| 欧美一级本道电影免费专区| 影音先锋成人资源网站| 亚洲在线国产日韩欧美| 午夜两性免费视频| 成人精品高清在线| 婷婷色一区二区三区| 一区二区三区在线观看国产| 羞羞影院体验区| 91.com在线观看| 你懂的视频在线观看| 久久综合五月天| 亚洲精品88| 成人免费网站在线观看| 久久亚洲黄色| 亚洲av综合色区| 蜜桃久久久久久| 久久久久9999| 亚洲黄一区二区三区| 探花国产精品一区二区| 欧美不卡一区二区三区| 岛国大片在线观看| 久久久久亚洲精品| 欧美网站免费| 久久亚洲综合网| 国产一区美女| 一二三级黄色片| 国产亚洲一区字幕| 精品少妇爆乳无码av无码专区| 精品视频资源站| 欧洲成人av| 久久免费少妇高潮久久精品99| 国产精品一区二区免费福利视频| 精品免费二区三区三区高中清不卡| 99热国内精品| 成人亚洲视频在线观看| 99久久久久久| www.99re7.com| 日韩西西人体444www| 自拍视频在线| 国产成人精品久久二区二区| av男人一区| 国产片侵犯亲女视频播放| 久久国产夜色精品鲁鲁99| 性久久久久久久久久久| 亚洲人精品午夜| 国产精品无码AV| 中文字幕精品www乱入免费视频| 交100部在线观看| 91丝袜脚交足在线播放| 天天综合网网欲色| 亚洲视频在线a| 久久综合九色欧美综合狠狠| 日本三级网站在线观看| 日韩欧美一级精品久久| 四虎av在线| 3d动漫精品啪啪一区二区三区免费| 第九色区aⅴ天堂久久香| 波多野结衣家庭教师视频| 99视频精品全部免费在线| 一区二区三区免费高清视频| 日韩视频在线你懂得| 手机av在线播放| 91蜜桃网站免费观看| 亚洲第一天堂| 日本中文字幕有码| 亚洲午夜在线电影| 亚洲欧美黄色片| 性欧美在线看片a免费观看| 第四色中文综合网| 久艹视频在线免费观看| 不卡av免费在线观看| 99热只有这里有精品| 精品视频中文字幕| 亚洲第一会所| 一道精品一区二区三区| 蜜桃av一区二区三区电影| www成人啪啪18软件| 欧美日韩情趣电影| 免费在线观看av| 亚洲综合小说区| 激情综合久久| 国产精品无码电影| 在线精品国精品国产尤物884a| av在线天堂| 91亚洲国产成人久久精品网站 | 在线成人激情黄色| 久久av影院| 成人在线播放网址| 久久久九九九九| 国产又粗又猛又黄| 欧美黑人性视频| 黑人操亚洲人| 午夜天堂在线视频| 亚洲成人福利片| 国产三级在线| 成人精品久久久| 亚洲国产国产亚洲一二三| 久久无码人妻精品一区二区三区| 欧美亚洲一区二区三区四区| 麻豆视频在线| 久久久久天天天天| 精品一区二区三区欧美| 国产精品美女毛片真酒店| 国产亚洲精品综合一区91| 日本在线一区二区三区| 日韩视频免费播放| 中文字幕第一页久久| 亚洲精品视频91| 国产精品久久久久7777婷婷| 亚洲视频久久| av在线播放网址| 在线成人小视频|