Android系統(tǒng)啟動流程(四)Launcher啟動過程與系統(tǒng)啟動流程

相關(guān)文章
Android系統(tǒng)架構(gòu)與系統(tǒng)源碼目錄
Android系統(tǒng)啟動流程(一)解析init進(jìn)程啟動過程
Android系統(tǒng)啟動流程(二)解析Zygote進(jìn)程啟動過程
Android系統(tǒng)啟動流程(三)解析SyetemServer進(jìn)程啟動過程

前言

此前的文章我們學(xué)習(xí)了init進(jìn)程余黎、Zygote進(jìn)程和SyetemServer進(jìn)程的啟動過程,這一篇文章我們就來學(xué)習(xí)Android系統(tǒng)啟動流程的最后一步:Launcher的啟動流程巡扇,并結(jié)合本系列的前三篇文章的內(nèi)容來講解Android系統(tǒng)啟動流程厅翔。建議讀這篇文章前要通讀本系列的前三篇文章搀突,否則你可能不會理解我在講什么描姚。

1.Launcher概述

Android系統(tǒng)啟動的最后一步是啟動一個Home應(yīng)用程序,這個應(yīng)用程序用來顯示系統(tǒng)中已經(jīng)安裝的應(yīng)用程序筒扒,這個Home應(yīng)用程序就叫做Launcher花墩。應(yīng)用程序Launcher在啟動過程中會請求PackageManagerService返回系統(tǒng)中已經(jīng)安裝的應(yīng)用程序的信息澄步,并將這些信息封裝成一個快捷圖標(biāo)列表顯示在系統(tǒng)屏幕上村缸,這樣用戶可以通過點(diǎn)擊這些快捷圖標(biāo)來啟動相應(yīng)的應(yīng)用程序。

2.Launcher啟動流程

SyetemServer進(jìn)程在啟動的過程中會啟動PackageManagerService仇箱,PackageManagerService啟動后會將系統(tǒng)中的應(yīng)用程序安裝完成剂桥。在此前已經(jīng)啟動的ActivityManagerService會將Launcher啟動起來。
啟動Launcher的入口為ActivityManagerService的systemReady函數(shù)美尸,如下所示斟薇。
frameworks/base/services/java/com/android/server/SystemServer.java

 private void startOtherServices() {
 ...
  mActivityManagerService.systemReady(new Runnable() {
            @Override
            public void run() {
                Slog.i(TAG, "Making services ready");
                mSystemServiceManager.startBootPhase(
                        SystemService.PHASE_ACTIVITY_MANAGER_READY);

...
}
...
}

在startOtherServices函數(shù)中奔垦,會調(diào)用ActivityManagerService的systemReady函數(shù):
frameworks/base/services/core/java/com/android/server/am/ActivityManagerService.java

public void systemReady(final Runnable goingCallback) {
...
synchronized (this) {
           ...
            mStackSupervisor.resumeFocusedStackTopActivityLocked();
            mUserController.sendUserSwitchBroadcastsLocked(-1, currentUserId);
        }
    }

systemReady函數(shù)中調(diào)用了ActivityStackSupervisor的resumeFocusedStackTopActivityLocked函數(shù):
frameworks/base/services/core/java/com/android/server/am/ActivityStackSupervisor.java

boolean resumeFocusedStackTopActivityLocked(
            ActivityStack targetStack, ActivityRecord target, ActivityOptions targetOptions) {
        if (targetStack != null && isFocusedStack(targetStack)) {
            return targetStack.resumeTopActivityUncheckedLocked(target, targetOptions);//1
        }
        final ActivityRecord r = mFocusedStack.topRunningActivityLocked();
        if (r == null || r.state != RESUMED) {
            mFocusedStack.resumeTopActivityUncheckedLocked(null, null);
        }
        return false;
    }

在注釋1處會調(diào)用ActivityStack的resumeTopActivityUncheckedLocked函數(shù)椿猎,ActivityStack對象是用來描述Activity堆棧的犯眠,resumeTopActivityUncheckedLocked函數(shù)如下所示症革。
frameworks/base/services/core/java/com/android/server/am/ActivityStack.java

 boolean resumeTopActivityUncheckedLocked(ActivityRecord prev, ActivityOptions options) {
        if (mStackSupervisor.inResumeTopActivity) {
            // Don't even start recursing.
            return false;
        }
        boolean result = false;
        try {
            // Protect against recursion.
            mStackSupervisor.inResumeTopActivity = true;
            if (mService.mLockScreenShown == ActivityManagerService.LOCK_SCREEN_LEAVING) {
                mService.mLockScreenShown = ActivityManagerService.LOCK_SCREEN_HIDDEN;
                mService.updateSleepIfNeededLocked();
            }
            result = resumeTopActivityInnerLocked(prev, options);//1
        } finally {
            mStackSupervisor.inResumeTopActivity = false;
        }
       return result;
    }

注釋1調(diào)用了resumeTopActivityInnerLocked函數(shù):

 private boolean resumeTopActivityInnerLocked(ActivityRecord prev, ActivityOptions options) {
   ...
   return isOnHomeDisplay() &&
                        mStackSupervisor.resumeHomeStackTask(returnTaskType, prev, "prevFinished");
   ...                 
}

resumeTopActivityInnerLocked函數(shù)的代碼很長量蕊,我們截取我們要分析的關(guān)鍵的一句:調(diào)用ActivityStackSupervisor的resumeHomeStackTask函數(shù)艇挨,代碼如下所示缩滨。
frameworks/base/services/core/java/com/android/server/am/ActivityStackSupervisor.java

boolean resumeHomeStackTask(int homeStackTaskType, ActivityRecord prev, String reason) {
    ...
    if (r != null && !r.finishing) {
        mService.setFocusedActivityLocked(r, myReason);
        return resumeFocusedStackTopActivityLocked(mHomeStack, prev, null);
    }
    return mService.startHomeActivityLocked(mCurrentUser, myReason);//1
}

在注釋1處調(diào)用了ActivityManagerService的startHomeActivityLocked函數(shù)脉漏,如下所示。
frameworks/base/services/core/java/com/android/server/am/ActivityManagerService.java

   boolean startHomeActivityLocked(int userId, String reason) {
        if (mFactoryTest == FactoryTest.FACTORY_TEST_LOW_LEVEL
                && mTopAction == null) {//1
            return false;
        }
        Intent intent = getHomeIntent();//2
        ActivityInfo aInfo = resolveActivityInfo(intent, STOCK_PM_FLAGS, userId);
        if (aInfo != null) {
            intent.setComponent(new ComponentName(aInfo.applicationInfo.packageName, aInfo.name));
            aInfo = new ActivityInfo(aInfo);
            aInfo.applicationInfo = getAppInfoForUser(aInfo.applicationInfo, userId);
            ProcessRecord app = getProcessRecordLocked(aInfo.processName,
                    aInfo.applicationInfo.uid, true);
            if (app == null || app.instrumentationClass == null) {//3
                intent.setFlags(intent.getFlags() | Intent.FLAG_ACTIVITY_NEW_TASK);
                mActivityStarter.startHomeActivityLocked(intent, aInfo, reason);//4
            }
        } else {
            Slog.wtf(TAG, "No home screen found for " + intent, new Throwable());
        }

        return true;
    }

注釋1處的mFactoryTest代表系統(tǒng)的運(yùn)行模式舅锄,系統(tǒng)的運(yùn)行模式分為三種皇忿,分別是非工廠模式、低級工廠模式和高級工廠模式撮胧,mTopAction則用來描述第一個被啟動Activity組件的Action芹啥,它的值為Intent.ACTION_MAIN铺峭。因此注釋1的代碼意思就是mFactoryTest為FactoryTest.FACTORY_TEST_LOW_LEVEL(低級工廠模式)并且mTopAction=null時卫键,直接返回false。注釋2處的getHomeIntent函數(shù)如下所示钓账。

Intent getHomeIntent() {
    Intent intent = new Intent(mTopAction, mTopData != null ? Uri.parse(mTopData) : null);
    intent.setComponent(mTopComponent);
    intent.addFlags(Intent.FLAG_DEBUG_TRIAGED_MISSING);
    if (mFactoryTest != FactoryTest.FACTORY_TEST_LOW_LEVEL) {
        intent.addCategory(Intent.CATEGORY_HOME);
    }
    return intent;
}

getHomeIntent函數(shù)中創(chuàng)建了Intent梆暮,并將mTopAction和mTopData傳入绍昂。mTopAction的值為Intent.ACTION_MAIN窘游,并且如果系統(tǒng)運(yùn)行模式不是低級工廠模式則將intent的Category設(shè)置為Intent.CATEGORY_HOME。我們再回到ActivityManagerService的startHomeActivityLocked函數(shù)贪嫂,假設(shè)系統(tǒng)的運(yùn)行模式不是低級工廠模式撩荣,在注釋3處判斷符合Action為Intent.ACTION_MAIN饶深,Category為Intent.CATEGORY_HOME的應(yīng)用程序是否已經(jīng)啟動敌厘,如果沒啟動則調(diào)用注釋4的方法啟動該應(yīng)用程序。
這個被啟動的應(yīng)用程序就是Launcher饱狂,因?yàn)長auncher的Manifest文件中的intent-filter標(biāo)簽匹配了Action為Intent.ACTION_MAIN,Category為Intent.CATEGORY_HOME讲婚。Launcher的Manifest文件如下所示筹麸。
packages/apps/Launcher3/AndroidManifest.xml

<manifest
    xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.android.launcher3">
    <uses-sdk android:targetSdkVersion="23" android:minSdkVersion="16"/>
 ...
 <application
        ...
        <activity
            android:name="com.android.launcher3.Launcher"
            android:launchMode="singleTask"
            android:clearTaskOnLaunch="true"
            android:stateNotNeeded="true"
            android:theme="@style/Theme"
            android:windowSoftInputMode="adjustPan"
            android:screenOrientation="nosensor"
            android:configChanges="keyboard|keyboardHidden|navigation"
            android:resumeWhilePausing="true"
            android:taskAffinity=""
            android:enabled="true">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.HOME" />
                <category android:name="android.intent.category.DEFAULT" />
                <category android:name="android.intent.category.MONKEY"/>
            </intent-filter>
        </activity>
...
  </application> 
</manifest>         

這樣物赶,應(yīng)用程序Launcher就會被啟動起來酵紫,并執(zhí)行它的onCreate函數(shù)错维。

3.Launcher中應(yīng)用圖標(biāo)顯示流程

Launcher的onCreate函數(shù)如下所示。
packages/apps/Launcher3/src/com/android/launcher3/Launcher.java

      @Override
    protected void onCreate(Bundle savedInstanceState) {
       ...
        LauncherAppState app = LauncherAppState.getInstance();//1
        mDeviceProfile = getResources().getConfiguration().orientation
                == Configuration.ORIENTATION_LANDSCAPE ?
                app.getInvariantDeviceProfile().landscapeProfile
                : app.getInvariantDeviceProfile().portraitProfile;

        mSharedPrefs = Utilities.getPrefs(this);
        mIsSafeModeEnabled = getPackageManager().isSafeMode();
        mModel = app.setLauncher(this);//2
        ....
        if (!mRestoring) {
            if (DISABLE_SYNCHRONOUS_BINDING_CURRENT_PAGE) {
                mModel.startLoader(PagedView.INVALID_RESTORE_PAGE);//2
            } else {
                mModel.startLoader(mWorkspace.getRestorePage());
            }
        }
...
    }

注釋1處獲取LauncherAppState的實(shí)例并在注釋2處調(diào)用它的setLauncher函數(shù)并將Launcher對象傳入轧坎,LauncherAppState的setLauncher函數(shù)如下所示。
packages/apps/Launcher3/src/com/android/launcher3/LauncherAppState.java

   LauncherModel setLauncher(Launcher launcher) {
        getLauncherProvider().setLauncherProviderChangeListener(launcher);
        mModel.initialize(launcher);//1
        mAccessibilityDelegate = ((launcher != null) && Utilities.ATLEAST_LOLLIPOP) ?
            new LauncherAccessibilityDelegate(launcher) : null;
        return mModel;
    }

注釋1處會調(diào)用LauncherModel的initialize函數(shù):

public void initialize(Callbacks callbacks) {
    synchronized (mLock) {
        unbindItemInfosAndClearQueuedBindRunnables();
        mCallbacks = new WeakReference<Callbacks>(callbacks);
    }
}

在initialize函數(shù)中會將Callbacks,也就是傳入的Launcher 封裝成一個弱引用對象捎泻。因此我們得知mCallbacks變量指的就是封裝成弱引用對象的Launcher笆豁,這個mCallbacks后文會用到它。
再回到Launcher的onCreate函數(shù)煞赢,在注釋2處調(diào)用了LauncherModel的startLoader函數(shù):
packages/apps/Launcher3/src/com/android/launcher3/LauncherModel.java

...
 @Thunk static final HandlerThread sWorkerThread = new HandlerThread("launcher-loader");//1
    static {
        sWorkerThread.start();
    }
    @Thunk static final Handler sWorker = new Handler(sWorkerThread.getLooper());//2
...
   public void startLoader(int synchronousBindPage, int loadFlags) {s
        InstallShortcutReceiver.enableInstallQueue();
        synchronized (mLock) {
            synchronized (mDeferredBindRunnables) {
                mDeferredBindRunnables.clear();
            }
            if (mCallbacks != null && mCallbacks.get() != null) {
                stopLoaderLocked();
                mLoaderTask = new LoaderTask(mApp.getContext(), loadFlags);//3
                if (synchronousBindPage != PagedView.INVALID_RESTORE_PAGE
                        && mAllAppsLoaded && mWorkspaceLoaded && !mIsLoaderTaskRunning) {
                    mLoaderTask.runBindSynchronousPage(synchronousBindPage);
                } else {
                    sWorkerThread.setPriority(Thread.NORM_PRIORITY);
                    sWorker.post(mLoaderTask);//4
                }
            }
        }
    }

注釋1處創(chuàng)建了具有消息循環(huán)的線程HandlerThread對象。注釋2處創(chuàng)建了Handler,并且傳入HandlerThread的Looper波俄。Hander的作用就是向HandlerThread發(fā)送消息懦铺。在注釋3處創(chuàng)建LoaderTask支鸡,在注釋4處將LoaderTask作為消息發(fā)送給HandlerThread 苍匆。
LoaderTask類實(shí)現(xiàn)了Runnable接口,當(dāng)LoaderTask所描述的消息被處理時則會調(diào)用它的run函數(shù)叔汁,代碼如下所示

 private class LoaderTask implements Runnable {
 ...
        public void run() {
            synchronized (mLock) {
                if (mStopped) {
                    return;
                }
                mIsLoaderTaskRunning = true;
            }
            keep_running: {
                if (DEBUG_LOADERS) Log.d(TAG, "step 1: loading workspace");
                loadAndBindWorkspace();//1
                if (mStopped) {
                    break keep_running;
                }
                waitForIdle();
                if (DEBUG_LOADERS) Log.d(TAG, "step 2: loading all apps");
                loadAndBindAllApps();//2
            }
            mContext = null;
            synchronized (mLock) {
                if (mLoaderTask == this) {
                    mLoaderTask = null;
                }
                mIsLoaderTaskRunning = false;
                mHasLoaderCompletedOnce = true;
            }
        }
   ...     
  }      

Launcher是用工作區(qū)的形式來顯示系統(tǒng)安裝的應(yīng)用程序的快捷圖標(biāo)据块,每一個工作區(qū)都是來描述一個抽象桌面的折剃,它由n個屏幕組成怕犁,每個屏幕又分n個單元格,每個單元格用來顯示一個應(yīng)用程序的快捷圖標(biāo)戈轿。注釋1處調(diào)用loadAndBindWorkspace函數(shù)用來加載工作區(qū)信息思杯,注釋2處的loadAndBindAllApps函數(shù)是用來加載系統(tǒng)已經(jīng)安裝的應(yīng)用程序信息挠进,loadAndBindAllApps函數(shù)代碼如下所示领突。

private void loadAndBindAllApps() {
    if (DEBUG_LOADERS) {
        Log.d(TAG, "loadAndBindAllApps mAllAppsLoaded=" + mAllAppsLoaded);
    }
    if (!mAllAppsLoaded) {
        loadAllApps();//1
        synchronized (LoaderTask.this) {
            if (mStopped) {
                return;
            }
        }
        updateIconCache();
        synchronized (LoaderTask.this) {
            if (mStopped) {
                return;
            }
            mAllAppsLoaded = true;
        }
    } else {
        onlyBindAllApps();
    }
}

如果系統(tǒng)沒有加載已經(jīng)安裝的應(yīng)用程序信息攘须,則會調(diào)用注釋1處的loadAllApps函數(shù):

  private void loadAllApps() {
...
        mHandler.post(new Runnable() {
            public void run() {
                final long bindTime = SystemClock.uptimeMillis();
                final Callbacks callbacks = tryGetCallbacks(oldCallbacks);
                if (callbacks != null) {
                    callbacks.bindAllApplications(added);//1
                    if (DEBUG_LOADERS) {
                        Log.d(TAG, "bound " + added.size() + " apps in "
                                + (SystemClock.uptimeMillis() - bindTime) + "ms");
                    }
                } else {
                    Log.i(TAG, "not binding apps: no Launcher activity");
                }
            }
        });
       ...
    }

在注釋1處會調(diào)用callbacks的bindAllApplications函數(shù),在前面我們得知這個callbacks實(shí)際是指向Launcher的浮驳,因此我們來查看Launcher的bindAllApplications函數(shù)至会,代碼如下所示。
packages/apps/Launcher3/src/com/android/launcher3/Launcher.java

public void bindAllApplications(final ArrayList<AppInfo> apps) {
    if (waitUntilResume(mBindAllApplicationsRunnable, true)) {
        mTmpAppsList = apps;
        return;
    }
    if (mAppsView != null) {
        mAppsView.setApps(apps);//1
    }
    if (mLauncherCallbacks != null) {
        mLauncherCallbacks.bindAllApplications(apps);
    }
}

在注釋1處會調(diào)用AllAppsContainerView的setApps函數(shù)宵蛀,并將包含應(yīng)用信息的列表apps傳進(jìn)去术陶,AllAppsContainerView的setApps函數(shù)如下所示梧宫。
packages/apps/Launcher3/src/com/android/launcher3/allapps/AllAppsContainerView.java

  public void setApps(List<AppInfo> apps) {
        mApps.setApps(apps);
    }

包含應(yīng)用信息的列表apps已經(jīng)傳給了AllAppsContainerView摆碉,查看AllAppsContainerView的onFinishInflate函數(shù):

 @Override
    protected void onFinishInflate() {
        super.onFinishInflate();
...
        // Load the all apps recycler view
        mAppsRecyclerView = (AllAppsRecyclerView) findViewById(R.id.apps_list_view);//1
        mAppsRecyclerView.setApps(mApps);//2
        mAppsRecyclerView.setLayoutManager(mLayoutManager);
        mAppsRecyclerView.setAdapter(mAdapter);//3
        mAppsRecyclerView.setHasFixedSize(true);
        mAppsRecyclerView.addOnScrollListener(mElevationController);
        mAppsRecyclerView.setElevationController(mElevationController);
...
    }

onFinishInflate函數(shù)在加載完xml文件時就會調(diào)用巷帝,在注釋1處得到AllAppsRecyclerView用來顯示App列表,并在注釋2處將apps的信息列表傳進(jìn)去驰徊,并在注釋3處為AllAppsRecyclerView設(shè)置Adapter辣垒。這樣應(yīng)用程序快捷圖標(biāo)的列表就會顯示在屏幕上印蔬。
到這里L(fēng)auncher啟動流程就講到這侥猬,接下來講Android系統(tǒng)啟動流程捐韩。

4.Android系統(tǒng)啟動流程

那么結(jié)合本篇以及本系列的前三篇文章荤胁,我們就可以得出Android系統(tǒng)啟動流程,如下所示垢油。
1.啟動電源以及系統(tǒng)啟動
當(dāng)電源按下時引導(dǎo)芯片代碼開始從預(yù)定義的地方(固化在ROM)開始執(zhí)行滩愁。加載引導(dǎo)程序Bootloader到RAM,然后執(zhí)行廉丽。
2.引導(dǎo)程序BootLoader
引導(dǎo)程序BootLoader是在Android操作系統(tǒng)開始運(yùn)行前的一個小程序妻味,它的主要作用是把系統(tǒng)OS拉起來并運(yùn)行责球。
3.Linux內(nèi)核啟動
內(nèi)核啟動時,設(shè)置緩存裁良、被保護(hù)存儲器价脾、計劃列表笛匙、加載驅(qū)動妹孙。當(dāng)內(nèi)核完成系統(tǒng)設(shè)置,它首先在系統(tǒng)文件中尋找init.rc文件骇笔,并啟動init進(jìn)程嚣崭。
4.init進(jìn)程啟動
初始化和啟動屬性服務(wù)雹舀,并且啟動Zygote進(jìn)程说榆。
5.Zygote進(jìn)程啟動
創(chuàng)建JavaVM并為JavaVM注冊JNI寸认,創(chuàng)建服務(wù)端Socket串慰,啟動SystemServer進(jìn)程模庐。
6.SystemServer進(jìn)程啟動
啟動Binder線程池和SystemServiceManager,并且啟動各種系統(tǒng)服務(wù)怜姿。
7.Launcher啟動
被SystemServer進(jìn)程啟動的ActivityManagerService會啟動Launcher沧卢,Launcher啟動后會將已安裝應(yīng)用的快捷圖標(biāo)顯示到界面上醉者。

結(jié)合上面的流程撬即,給出Android系統(tǒng)啟動流程圖:



歡迎關(guān)注我的微信公眾號剥槐,第一時間獲得博客更新提醒,以及更多成體系的Android相關(guān)原創(chuàng)技術(shù)干貨颅崩。
掃一掃下方二維碼即可關(guān)注:

enter image description here

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市尖滚,隨后出現(xiàn)的幾起案子锅移,更是在濱河造成了極大的恐慌非剃,老刑警劉巖推沸,帶你破解...
    沈念sama閱讀 206,214評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異恨锚,居然都是意外死亡猴伶,警方通過查閱死者的電腦和手機(jī)塌西,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評論 2 382
  • 文/潘曉璐 我一進(jìn)店門捡需,熙熙樓的掌柜王于貴愁眉苦臉地迎上來站辉,“玉大人,你說我怎么就攤上這事殊霞”炼祝” “怎么了古沥?”我有些...
    開封第一講書人閱讀 152,543評論 0 341
  • 文/不壞的土叔 我叫張陵岩齿,是天一觀的道長盹沈。 經(jīng)常有香客問我,道長做裙,這世上最難降的妖魔是什么肃晚? 我笑而不...
    開封第一講書人閱讀 55,221評論 1 279
  • 正文 為了忘掉前任拧廊,我火速辦了婚禮,結(jié)果婚禮上凰盔,老公的妹妹穿的比我還像新娘倦春。我一直安慰自己睁本,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,224評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著暮胧,像睡著了一般往衷。 火紅的嫁衣襯著肌膚如雪席舍。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,007評論 1 284
  • 那天,我揣著相機(jī)與錄音福铅,去河邊找鬼滑黔。 笑死笆包,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的略荡。 我是一名探鬼主播庵佣,決...
    沈念sama閱讀 38,313評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼汛兜!你這毒婦竟也來了巴粪?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,956評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎验毡,沒想到半個月后衡创,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體晶通,經(jīng)...
    沈念sama閱讀 43,441評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,925評論 2 323
  • 正文 我和宋清朗相戀三年哟玷,在試婚紗的時候發(fā)現(xiàn)自己被綠了狮辽。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,018評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡巢寡,死狀恐怖喉脖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情抑月,我是刑警寧澤树叽,帶...
    沈念sama閱讀 33,685評論 4 322
  • 正文 年R本政府宣布,位于F島的核電站谦絮,受9級特大地震影響题诵,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜层皱,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,234評論 3 307
  • 文/蒙蒙 一性锭、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧叫胖,春花似錦草冈、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至绷跑,卻和暖如春拳恋,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背你踩。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評論 1 261
  • 我被黑心中介騙來泰國打工诅岩, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人带膜。 一個月前我還...
    沈念sama閱讀 45,467評論 2 352
  • 正文 我出身青樓吩谦,卻偏偏與公主長得像,于是被迫代替她去往敵國和親膝藕。 傳聞我的和親對象是個殘疾皇子式廷,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,762評論 2 345

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 171,500評論 25 707
  • 最近發(fā)現(xiàn)了自己存在很多的缺點(diǎn),這點(diǎn)是在非常清醒的情況下意識到的芭挽,也隱約預(yù)感到潛在的危害性滑废。在別人的眼里具體是什么情...
    小城情懷閱讀 319評論 0 0
  • 好久沒有就兩三個朋友一起出來聚餐蝗肪,想想也是這樣的機(jī)會也會變得越來越少。大學(xué)之前我想著以后我要多交一些朋友蠕趁,我的...
    吳_JiaJia閱讀 500評論 0 1
  • 14年的那個夏天俺陋,我結(jié)束了12年寒窗苦讀的生涯豁延,懷著喜悅的心情來到營口理工。但這喜悅轉(zhuǎn)瞬即逝腊状,站臺上诱咏,一聲長鳴,和...
    YNM解說閱讀 241評論 6 4
  • 空閑時看看和兒子游戲的視頻缴挖,空虛時間一下子被快樂填滿袋狞。因?yàn)樾r候游戲的時間太少,如今才這么珍惜吧映屋?
    楊知行閱讀 129評論 0 0