Activity启动流程

我们来看一下Activity启动流程,当 A 调用 startActivity(intent); 启动 B 的时候,系统是如何帮我们打开一个新的Activity的?
下面我们来从源码分析(Api : 25):

1、首先我们来到Activity类中

    @Override
    public void startActivity(Intent intent) {
        this.startActivity(intent, null);
    }

    @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) {
        if (mParent == null) {
            options = transferSpringboardActivityOptions(options);
            Instrumentation.ActivityResult ar =
                // 关键是调用这句代码
                mInstrumentation.execStartActivity(
                    this, mMainThread.getApplicationThread(), mToken, this,
                    intent, requestCode, options);
            // ...
        } else {
            // ...
        }
    }

执行 mInstrumentation.execStartActivity方法,我们找到 mInstrumentation 为 Instrumentation这个类
2、我们现在来到Instrumentation类中

    public ActivityResult execStartActivity(
            Context who, IBinder contextThread, IBinder token, Activity target,
            Intent intent, int requestCode, Bundle options) {
        // ...
        
        try {
            intent.migrateExtraStreamToClipData();
            intent.prepareToLeaveProcess(who);
            // 关键调用了这句代码
            int result = ActivityManagerNative.getDefault()
                .startActivity(whoThread, who.getBasePackageName(), 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;
    }

这里 ActivityManagerNative.getDefault() 返回的是一个 IActivityManager 接口,并且 ActivityManagerNative是实现 IActivityManager 接口的
3、我们来到 ActivityManagerNative类中

    public int startActivity(IApplicationThread caller, String callingPackage, Intent intent,
            String resolvedType, IBinder resultTo, String resultWho, int requestCode,
            int startFlags, ProfilerInfo profilerInfo, Bundle options) throws RemoteException {
        // 典型的IPC机制,这里保存了一些数据
        Parcel data = Parcel.obtain();
        Parcel reply = Parcel.obtain();
        data.writeInterfaceToken(IActivityManager.descriptor);
        data.writeStrongBinder(caller != null ? caller.asBinder() : null);
        data.writeString(callingPackage);
        intent.writeToParcel(data, 0);
        data.writeString(resolvedType);
        data.writeStrongBinder(resultTo);
        data.writeString(resultWho);
        data.writeInt(requestCode);
        data.writeInt(startFlags);
        if (profilerInfo != null) {
            data.writeInt(1);
            profilerInfo.writeToParcel(data, Parcelable.PARCELABLE_WRITE_RETURN_VALUE);
        } else {
            data.writeInt(0);
        }
        if (options != null) {
            data.writeInt(1);
            options.writeToParcel(data, 0);
        } else {
            data.writeInt(0);
        }
        // IPC机制,发送一个消息,我们直接在本类中来搜 START_ACTIVITY_TRANSACTION,搜索服务端的代码
        mRemote.transact(START_ACTIVITY_TRANSACTION, data, reply, 0);
        reply.readException();
        int result = reply.readInt();
        reply.recycle();
        data.recycle();
        return result;
    }

    @Override
    public boolean onTransact(int code, Parcel data, Parcel reply, int flags)
            throws RemoteException {
        switch (code) {
        // 这里就是我们搜索到服务端代码入口
        case START_ACTIVITY_TRANSACTION:
        {
            // 接收客户端传来的消息
            data.enforceInterface(IActivityManager.descriptor);
            IBinder b = data.readStrongBinder();
            IApplicationThread app = ApplicationThreadNative.asInterface(b);
            String callingPackage = data.readString();
            Intent intent = Intent.CREATOR.createFromParcel(data);
            String resolvedType = data.readString();
            IBinder resultTo = data.readStrongBinder();
            String resultWho = data.readString();
            int requestCode = data.readInt();
            int startFlags = data.readInt();
            ProfilerInfo profilerInfo = data.readInt() != 0
                    ? ProfilerInfo.CREATOR.createFromParcel(data) : null;
            Bundle options = data.readInt() != 0
                    ? Bundle.CREATOR.createFromParcel(data) : null;
            // 启动Activity,ActivityManagerNative是一个抽象类的,所以这里调用ActivityManagerService中的startActivity
            int result = startActivity(app, callingPackage, intent, resolvedType,
                    resultTo, resultWho, requestCode, startFlags, profilerInfo, options);
            reply.writeNoException();
            reply.writeInt(result);
            return true;
        }
        // ...
    }

4、我们来到 ActivityManagerService 类中

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


    @Override
    public final int startActivityAsUser(IApplicationThread caller, String callingPackage,
            Intent intent, String resolvedType, IBinder resultTo, String resultWho, int requestCode,
            int startFlags, ProfilerInfo profilerInfo, Bundle bOptions, int userId) {
        // 这里调用ActivityStarter类中的startActivityMayWait
        return mActivityStarter.startActivityMayWait(caller, -1, callingPackage, intent,
                resolvedType, null, null, resultTo, resultWho, requestCode, startFlags,
                profilerInfo, null, null, bOptions, false, userId, null, null);
    }

5、我们来到 ActivityStarter 类中

    final int startActivityMayWait(IApplicationThread caller, int callingUid,
            String callingPackage, Intent intent, String resolvedType,
            IVoiceInteractionSession voiceSession, IVoiceInteractor voiceInteractor,
            IBinder resultTo, String resultWho, int requestCode, int startFlags,
            ProfilerInfo profilerInfo, IActivityManager.WaitResult outResult, Configuration config,
            Bundle bOptions, boolean ignoreTargetSecurity, int userId,
            IActivityContainer iContainer, TaskRecord inTask) {
        // ...
        int res = startActivityLocked(caller, intent, ephemeralIntent, resolvedType,
                    aInfo, rInfo, voiceSession, voiceInteractor,
                    resultTo, resultWho, requestCode, callingPid,
                    callingUid, callingPackage, realCallingPid, realCallingUid, startFlags,
                    options, ignoreTargetSecurity, componentSpecified, outRecord, container,
                    inTask);
        // ...
        return res;
    }


    final int startActivityLocked(IApplicationThread caller, Intent intent, Intent ephemeralIntent,
            String resolvedType, ActivityInfo aInfo, ResolveInfo rInfo,
            IVoiceInteractionSession voiceSession, IVoiceInteractor voiceInteractor,
            IBinder resultTo, String resultWho, int requestCode, int callingPid, int callingUid,
            String callingPackage, int realCallingPid, int realCallingUid, int startFlags,
            ActivityOptions options, boolean ignoreTargetSecurity, boolean componentSpecified,
            ActivityRecord[] outActivity, ActivityStackSupervisor.ActivityContainer container,
            TaskRecord inTask) {
        // ...
        int err = startActivityUnchecked(r, sourceRecord, voiceSession, voiceInteractor, startFlags,
                    true, options, inTask);
        return err;
    }


    private int startActivityUnchecked(final ActivityRecord r, ActivityRecord sourceRecord,
            IVoiceInteractionSession voiceSession, IVoiceInteractor voiceInteractor,
            int startFlags, boolean doResume, ActivityOptions options, TaskRecord inTask) {
        // ...
        
        // mSupervisor: ActivityStackSupervisor 
        mSupervisor.resumeFocusedStackTopActivityLocked(mTargetStack, mStartActivity,
                        mOptions);
        
        // ...
        return START_SUCCESS;
    }

6、我们来到 ActivityStackSupervisor 类中

    boolean resumeFocusedStackTopActivityLocked(
            ActivityStack targetStack, ActivityRecord target, ActivityOptions targetOptions) {
        if (targetStack != null && isFocusedStack(targetStack)) {
            // targetStack : ActivityStack
            return targetStack.resumeTopActivityUncheckedLocked(target, targetOptions);
        }
        // ...
        return false;
    }

7、我们来到 ActivityStack 类中

    boolean resumeTopActivityUncheckedLocked(ActivityRecord prev, ActivityOptions options) {
        // ...
        boolean result = false;
        try {
            // ...
            // 关键调用这句代码
            result = resumeTopActivityInnerLocked(prev, options);
        } finally {
            mStackSupervisor.inResumeTopActivity = false;
        }
        return result;
    }

    private boolean resumeTopActivityInnerLocked(ActivityRecord prev, ActivityOptions options) {
        // ...
        
        if (mResumedActivity != null) {
            // 此处就会调用暂停方法
            pausing |= startPausingLocked(userLeaving, false, next, dontWaitForPause);
        }
        
        //mStackSupervisor : ActivityStackSupervisor,这里就会调用新建方法
        mStackSupervisor.startSpecificActivityLocked(next, true, true);
        
        return true;
    }

    final boolean startPausingLocked(boolean userLeaving, boolean uiSleeping,
            ActivityRecord resuming, boolean dontWait) {
         // ...
         // prev.app.thread为 ActivityThread 
         prev.app.thread.schedulePauseActivity(prev.appToken, prev.finishing,
                        userLeaving, prev.configChangeFlags, dontWait);
         // ...
         return true;
    }

8、我们先来看暂停的方法,我们来到 ActivityThread 类中

    public final void schedulePauseActivity(IBinder token, boolean finished,
                boolean userLeaving, int configChanges, boolean dontReport) {
            int seq = getLifecycleSeq();
            // 发送了一个消息,消息key: PAUSE_ACTIVITY,我们来搜这个key
            sendMessage(
                    finished ? H.PAUSE_ACTIVITY_FINISHING : H.PAUSE_ACTIVITY,
                    token,
                    (userLeaving ? USER_LEAVING : 0) | (dontReport ? DONT_REPORT : 0),
                    configChanges,
                    seq);
        }

        case PAUSE_ACTIVITY: {
                    // 关键调用方法
                    handlePauseActivity((IBinder) args.arg1, false,
                            (args.argi1 & USER_LEAVING) != 0, args.argi2,
                            (args.argi1 & DONT_REPORT) != 0, args.argi3);
                    maybeSnapshot();
                    Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
                } break;

     private void handlePauseActivity(IBinder token, boolean finished,
            boolean userLeaving, int configChanges, boolean dontReport, int seq) {
         
         // ...
         performPauseActivity(token, finished, r.isPreHoneycomb(), "handlePauseActivity");
         
         // ...
    }

    final Bundle performPauseActivity(ActivityClientRecord r, boolean finished,
            boolean saveState, String reason) {
        // ...
        performPauseActivityIfNeeded(r, reason);
        // ...
        return !r.activity.mFinished && saveState ? r.state : null;
    }

    private void performPauseActivityIfNeeded(ActivityClientRecord r, String reason) {
        // ...
        // mInstrumentation: Instrumentation
        mInstrumentation.callActivityOnPause(r.activity);
        // ...
    }

9、我们来到Instrumentation类中:

    public void callActivityOnPause(Activity activity) {
        activity.performPause();
    }

10、我们来到Activity类中

    final void performPause() {
        // ...
        // 终于看到了一个我们比较熟悉的方法,到这里调用了A的暂停方法
        onPause();
        // ...
        mResumed = false; 
    }

11、还记得我们在第7步中还有一个方法没走嘛? 现在我们回到 第7步中的ActivityStack 的resumeTopActivityInnerLocked方法中,在暂停方法下面还有一个新建方法:mStackSupervisor.startSpecificActivityLocked(next, true, true);
我们来到ActivityStackSupervisor类中

    void startSpecificActivityLocked(ActivityRecord r,
            boolean andResume, boolean checkConfig) {
        // ... 
        // 关键代码
        realStartActivityLocked(r, app, andResume, checkConfig);
        // ... 
    }

    final boolean realStartActivityLocked(ActivityRecord r, ProcessRecord app,
            boolean andResume, boolean checkConfig) throws RemoteException {
        // ... 
        // 关键代码,app.thread:ActivityThread 
        app.thread.scheduleLaunchActivity(new Intent(r.intent), r.appToken,
                    System.identityHashCode(r), r.info, new Configuration(mService.mConfiguration),
                    new Configuration(task.mOverrideConfig), r.compat, r.launchedFromPackage,
                    task.voiceInteractor, app.repProcState, r.icicle, r.persistentState, results,
                    newIntents, !andResume, mService.isNextTransitionForward(), profilerInfo);
        // ... 
        return true;
    }

12、我们再次来到 ActivityThread 类中

        @Override
        public final void scheduleLaunchActivity(Intent intent, IBinder token, int ident,
                ActivityInfo info, Configuration curConfig, Configuration overrideConfig,
                CompatibilityInfo compatInfo, String referrer, IVoiceInteractor voiceInteractor,
                int procState, Bundle state, PersistableBundle persistentState,
                List<ResultInfo> pendingResults, List<ReferrerIntent> pendingNewIntents,
                boolean notResumed, boolean isForward, ProfilerInfo profilerInfo) {

            
            // ...
            // 关键代码,发送一个消息,消息key为 LAUNCH_ACTIVITY,我们来搜 LAUNCH_ACTIVITY
            sendMessage(H.LAUNCH_ACTIVITY, r);
        }

        case LAUNCH_ACTIVITY: {
                    Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "activityStart");
                    final ActivityClientRecord r = (ActivityClientRecord) msg.obj;

                    r.packageInfo = getPackageInfoNoCheck(
                            r.activityInfo.applicationInfo, r.compatInfo);
                    // 关键代码
                    handleLaunchActivity(r, null, "LAUNCH_ACTIVITY");
                    Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
                } break;

    private void handleLaunchActivity(ActivityClientRecord r, Intent customIntent, String reason) {
        // ...
        
        // 关键代码
        Activity a = performLaunchActivity(r, customIntent);
        
        if (a != null) {
            // ...
            // 关键代码
            handleResumeActivity(r.token, false, r.isForward,
                    !r.activity.mFinished && !r.startsNotResumed, r.lastProcessedSeq, reason);
        }
        
        // 关键代码
        Looper.myQueue().addIdleHandler(new Idler());
    }

    private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) {
        // System.out.println("##### [" + System.currentTimeMillis() + "] ActivityThread.performLaunchActivity(" + r + ")");
        
        // mInstrumentation: Instrumentation
        mInstrumentation.callActivityOnCreate(activity, r.state);

        return activity;
    }

13、我们再次来到Instrumentation类中:

    public void callActivityOnCreate(Activity activity, Bundle icicle,
            PersistableBundle persistentState) {
        prePerformCreate(activity);
        activity.performCreate(icicle, persistentState);
        postPerformCreate(activity);
    }

14、我们再次来到Activity类中

    final void performCreate(Bundle icicle) {
        restoreHasCurrentPermissionRequest(icicle);
        // 终于看到了我们的onCreate方法
        onCreate(icicle);
        mActivityTransitionState.readState(icicle);
        performCreateCommon();
    }

15、我们再次回到12 的 ActivityThread的handleLaunchActivity方法中,继续执行handleResumeActivity方法

    final void handleResumeActivity(IBinder token,
            boolean clearHide, boolean isForward, boolean reallyResume, int seq, String reason) {
        r = performResumeActivity(token, clearHide, reason);
    }

    public final ActivityClientRecord performResumeActivity(IBinder token,
            boolean clearHide, String reason) {
        // ...
        r.activity.performResume();
        return r;
    }

16、我们再次来到Activity类中

    final void performResume() {
        // ...
        performRestart();
        // ...
        // mInstrumentation: Instrumentation
        mInstrumentation.callActivityOnResume(this);
    }

    final void performRestart() {
        performStart();
    }

    final void performStart() {
        // mInstrumentation: Instrumentation
        mInstrumentation.callActivityOnStart(this);
    }

17、我们再次来到Instrumentation类中:

    public void callActivityOnStart(Activity activity) {
        // 在这里看到了我们的onStart方法
        activity.onStart();
    }

    public void callActivityOnResume(Activity activity) {
        // 在这里看到了我们的onResume方法
        activity.onResume();
    }

18、我们再次回到12 的 ActivityThread的handleLaunchActivity方法中,继续执行Looper.myQueue().addIdleHandler(new Idler());方法

    private class Idler implements MessageQueue.IdleHandler {
        @Override
        public final boolean queueIdle() {
            // ...
            // 这里的am就是ActivityManagerService 
            IActivityManager am = ActivityManagerNative.getDefault();
            am.activityIdle(a.token, a.createdConfig, stopProfiling);
            // ...
        }
    }

19、我们再次来到ActivityManagerService 类中:

    @Override
    public final void activityIdle(IBinder token, Configuration config, boolean stopProfiling) {
        // ...
        // 这里的mStackSupervisor:ActivityStackSupervisor
        ActivityRecord r = mStackSupervisor.activityIdleInternalLocked(token, false, config);
        // ...
    }

20、我们再次来到 ActivityStackSupervisor 类中:

    final ActivityRecord activityIdleInternalLocked(final IBinder token, boolean fromTimeout,
            Configuration config) {
        // ...
        // 这里的stack: ActivityStack
        stack.stopActivityLocked(r);
        // ...
        return r;
    }

21、我们来到 ActivityStack 类中

    final void stopActivityLocked(ActivityRecord r) {
        r.app.thread.scheduleStopActivity(r.appToken, r.visible, r.configChangeFlags);
    }

22、我们再次来到ActivityThread类中

    public final void scheduleStopActivity(IBinder token, boolean showWindow,
                int configChanges) {
            int seq = getLifecycleSeq();
            if (DEBUG_ORDER) Slog.d(TAG, "stopActivity " + ActivityThread.this
                    + " operation received seq: " + seq);
            sendMessage(
                // 发送一个消息STOP_ACTIVITY_SHOW,我们来搜这个STOP_ACTIVITY_SHOW
                showWindow ? H.STOP_ACTIVITY_SHOW : H.STOP_ACTIVITY_HIDE,
                token, 0, configChanges, seq);
     }

    case STOP_ACTIVITY_SHOW: {
                    Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "activityStop");
                    SomeArgs args = (SomeArgs) msg.obj;
                    // 关键方法
                    handleStopActivity((IBinder) args.arg1, true, args.argi2, args.argi3);
                    Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
                } break;

    private void handleStopActivity(IBinder token, boolean show, int configChanges, int seq) {
        // 关键方法
        performStopActivityInner(r, info, show, true, "handleStopActivity");
    }

    private void performStopActivityInner(ActivityClientRecord r,
            StopInfo info, boolean keepShown, boolean saveState, String reason) {
        
        r.activity.performStop(false /*preserveWindow*/);
    }

23、我们再次来到Activity类中

    final void performStop(boolean preserveWindow) {
            mInstrumentation.callActivityOnStop(this);
    }

24、我们再次来到Instrumentation类中:

    public void callActivityOnStop(Activity activity) {
        activity.onStop();
    }
总结:

A启动B方法顺序: A的onPause -> B的onCreate -> B的onStart方法 -> B的onResume -> A的onStop

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

推荐阅读更多精彩内容

  • 背景介绍 从事开发到了一定阶段,想要提高就必须搞明白系统的一些工作原理。为什么?因为只有明白了这些,你才能针对平台...
    CoorChice阅读 72,719评论 57 508
  • 1.前言 小时候听大人们讲四大组件的故事,以为Activity就是手机屏幕上被看到的那东西。长大以后才发现,原来这...
    吴愣阅读 12,197评论 12 32
  • 前言 Launcher启动app launcher就是android桌面应用程序。也是操作系统启动有第一个app。...
    第八区阅读 1,006评论 0 2
  • “起床没有?好饿,一起去食堂吃饭” 我刚拿起手机,就看到你发来的信息。 “早起了,但是不想出寝室...” “手...
    野野边阅读 553评论 2 1
  • 中秋到,三天假期不多,对上班上学的人来说却是难得的休息。多少人正在琢磨着出门玩耍。那么在这依旧炎热的天气,去三峡人...
    阿梨的小故事阅读 1,986评论 2 5