Android源碼解析 - Launcher啟動流程

Launcher 概述

我們知道,Android 系統(tǒng)啟動的最后一步叠赐,就是去啟動一個桌面應(yīng)用程序痊土,這個應(yīng)用程序就是 Launcher塞帐。

Launcher 其實就是一個普通的 App 應(yīng)用程序,只是它的功能是可以顯示 Android 系統(tǒng)中所有已經(jīng)安裝的程序沪羔,然后提供用戶點擊相應(yīng)的應(yīng)用快捷圖標(biāo)就可以啟動相關(guān)應(yīng)用的功能······饥伊。

那么,Launcher 進程的具體啟動流程是怎樣的呢蔫饰?我們下面就來結(jié)合源碼進行分析琅豆。

Launcher 啟動流程解析(源碼:android-6.0.1_r81

在前面的文章 Android 系統(tǒng)啟動流程簡析 有講過,在 System Server 進程啟動各種系統(tǒng)服務(wù)后篓吁,Android 系統(tǒng)就已經(jīng)啟動完成了茫因。Launcher 進程的創(chuàng)建,其實就存在于 System Server 啟動系統(tǒng)服務(wù)過程中杖剪,具體代碼如下:frameworks\base\services\java\com\android\server\SystemServer.java

private void run() {
    ...
    // Start services.
    startBootstrapServices();
    startCoreServices();
    startOtherServices();
    ...
}

System Server 在其run方法內(nèi)部開啟了3類系統(tǒng)服務(wù):BootstrapServices冻押,CoreServicesOtherServices

我們進入startOtherServices方法內(nèi)部看下:

private void startOtherServices() {
    ...
    // We now tell the activity manager it is okay to run third party
    // code.  It will call back into us once it has gotten to the state
    // where third party code can really run (but before it has actually
    // started launching the initial applications), for us to complete our
    // initialization.
    mActivityManagerService.systemReady(new Runnable() {
            @Override
            public void run() {
              /*
              *執(zhí)行各種SystemService的啟動方法盛嘿,各種SystemService的systemReady方法...
              */
            }
    });
}

startOtherServices方法內(nèi)調(diào)用了ActivityManagerService.systemReady(Runnable)方法洛巢,其參數(shù)Runnable是一個回調(diào),當(dāng)系統(tǒng)達到可以運行第三方代碼的狀態(tài)后次兆,會回調(diào)該Runnable稿茉,執(zhí)行各種SystemService的啟動方法,各種SystemServicesystemReady方法。不過這個過程我們并不關(guān)心漓库,我們主要要看下ActivityManagerService.systemReady(Runnable)這個方法的內(nèi)部實現(xiàn):frameworks\base\services\core\java\com\android\server\am\ActivityManagerService.java

public void systemReady(final Runnable goingCallback) {
    ...
    // Start up initial activity.
    mBooting = true;
    startHomeActivityLocked(mCurrentUserId, "systemReady");
    ...
}

可以看到恃慧,在systemReady里面調(diào)用了startHomeActivityLocked方法,見名知意渺蒿,該方法應(yīng)該就是用來啟動 Launcher 桌面程序痢士。那么我們就接著看下這個方法的內(nèi)部實現(xiàn):

boolean startHomeActivityLocked(int userId, String reason) {
    ...
    Intent intent = getHomeIntent();
    ActivityInfo aInfo =
        resolveActivityInfo(intent, STOCK_PM_FLAGS, userId);
    if (aInfo != null) {
        intent.setComponent(new ComponentName(
                    aInfo.applicationInfo.packageName, aInfo.name));
        // Don't do this if the home app is currently being
        // instrumented.
        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) {
            intent.setFlags(intent.getFlags() | Intent.FLAG_ACTIVITY_NEW_TASK);
            mStackSupervisor.startHomeActivity(intent, aInfo, reason);
        }
    }
    return true;
}

ComponentName mTopComponent;
String mTopAction = Intent.ACTION_MAIN;// "android.intent.action.MAIN"
String mTopData;

Intent getHomeIntent() {
    Intent intent = new Intent(mTopAction, mTopData != null ? Uri.parse(mTopData) : null);
    intent.setComponent(mTopComponent);
    // 系統(tǒng)運行模式不是低級工廠模式,則添加CATEGORY_HOME
    if (mFactoryTest != FactoryTest.FACTORY_TEST_LOW_LEVEL) {
        intent.addCategory(Intent.CATEGORY_HOME);
    }
    return intent;
}

startHomeActivityLocked主要做了3件事:

  1. 首先是調(diào)用getHomeIntent構(gòu)建一個Intent茂装,
    IntentActionmTopAction良瞧,ComponentNamemTopComponentmTopAction默認為Intent.ACTION_MAIN训唱,mTopComponent默認為空褥蚯,只有當(dāng)系統(tǒng)處于低級工廠模式時,mTopActionmTopComponent才會被改變况增,具體代碼如下:
public void systemReady(final Runnable goingCallback) {
    ...
    if (mFactoryTest == FactoryTest.FACTORY_TEST_LOW_LEVEL) {
        ResolveInfo ri = mContext.getPackageManager()
            .resolveActivity(new Intent(Intent.ACTION_FACTORY_TEST),
                    STOCK_PM_FLAGS);
        CharSequence errorMsg = null;
        if (ri != null) {
            ActivityInfo ai = ri.activityInfo;
            ApplicationInfo app = ai.applicationInfo;
            if ((app.flags&ApplicationInfo.FLAG_SYSTEM) != 0) {
                //設(shè)置為工廠模式Action
                mTopAction = Intent.ACTION_FACTORY_TEST;
                mTopData = null;
                //設(shè)置工廠模式App的ComponentName
                mTopComponent = new ComponentName(app.packageName,
                        ai.name);
            } else {
                errorMsg = mContext.getResources().getText(
                        com.android.internal.R.string.factorytest_not_system);
            }
        } else {
            errorMsg = mContext.getResources().getText(
                    com.android.internal.R.string.factorytest_no_action);
        }
        if (errorMsg != null) {
            mTopAction = null;
            mTopData = null;
            mTopComponent = null;
            Message msg = Message.obtain();
            msg.what = SHOW_FACTORY_ERROR_MSG;
            msg.getData().putCharSequence("msg", errorMsg);
            mUiHandler.sendMessage(msg);
        }
    }
    ...
}

因此赞庶,系統(tǒng)正常啟動時,getHomeIntent構(gòu)建的IntentComponentNamenull澳骤,Action"android.intent.action.MAIN"歧强,Category"android.intent.category.HOME",(所以为肮,Launcher 與普通 App 的區(qū)別就是多了一個<category android:name="android.intent.category.HOME">)摊册。

  1. 獲取到HomeIntent后,就又通過resolveActivityInfo方法結(jié)合HomeIntent獲取到要啟動的應(yīng)用相關(guān)信息颊艳,具體代碼如下:
private ActivityInfo resolveActivityInfo(Intent intent, int flags, int userId) {
    ActivityInfo ai = null;
    //系統(tǒng)正常啟動時茅特,comp為null
    ComponentName comp = intent.getComponent();
    try {
        if (comp != null) {
            // Factory test.
            ai = AppGlobals.getPackageManager().getActivityInfo(comp, flags, userId);
        } else {
            ResolveInfo info = AppGlobals.getPackageManager().resolveIntent(
                    intent,
                    intent.resolveTypeIfNeeded(mContext.getContentResolver()),
                    flags, userId);

            if (info != null) {
                ai = info.activityInfo;
            }
        }
    } catch (RemoteException e) {
        // ignore
    }
    return ai;
}

我們在前面分析getHomeIntent方法時,已經(jīng)知道resolveActivityInfo傳入的IntentComponentNamenull棋枕,因此白修,resolveActivityInfo最終調(diào)用的是AppGlobals.getPackageManager().resolveIntent方法:

那么我們首先來看下AppGlobals.getPackageManager()的具體實現(xiàn):
frameworks\base\core\java\android\app\AppGlobals.java

 public static IPackageManager getPackageManager() {
        return ActivityThread.getPackageManager();
    }

frameworks\base\core\java\android\app\ActivityThread.java

static IPackageManager sPackageManager;

public static IPackageManager getPackageManager() {
    if (sPackageManager != null) {
        //Slog.v("PackageManager", "returning cur default = " + sPackageManager);
        return sPackageManager;
    }
    IBinder b = ServiceManager.getService("package");
    //Slog.v("PackageManager", "default service binder = " + b);
    sPackageManager = IPackageManager.Stub.asInterface(b);
    //Slog.v("PackageManager", "default service = " + sPackageManager);
    return sPackageManager;
}

所以AppGlobals.getPackageManager()最終返回的其實就是一個IPackageManager實例,從其內(nèi)部實現(xiàn)ActivityThread.getPackageManager()可以很明顯地知道該IPackageManager實例是一個遠程Binder代理重斑,對應(yīng)的服務(wù)端其實就是PackageManagerService

public class PackageManagerService extends IPackageManager.Stub{···}

因此兵睛,AppGlobals.getPackageManager().resolveIntent最終調(diào)用的就是PackageManagerService.resolveIntent方法,我們來查看下該方法具體實現(xiàn):
frameworks\base\services\core\java\com\android\server\pm\PackageManagerService.java

@Override
public ResolveInfo resolveIntent(Intent intent, String resolvedType,
        int flags, int userId) {
    if (!sUserManager.exists(userId)) return null;
    //權(quán)限檢查
    enforceCrossUserPermission(Binder.getCallingUid(), userId, false, false, "resolve intent");
    //查找符合Intent的Activity相關(guān)信息
    List<ResolveInfo> query = queryIntentActivities(intent, resolvedType, flags, userId);
    //選擇一個最符合要求的Activity
    return chooseBestActivity(intent, resolvedType, flags, query, userId);
}

resolveIntent該方法主要做了2件事:

  1. 通過queryIntentActivities來查找符合HomeIntent需求Activities,我們來看下queryIntentActivities的具體實現(xiàn)代碼:
@Override
public List<ResolveInfo> queryIntentActivities(Intent intent,
        String resolvedType, int flags, int userId) {
    if (!sUserManager.exists(userId)) return Collections.emptyList();
    enforceCrossUserPermission(Binder.getCallingUid(), userId, false, false, "query intent activities");
    ComponentName comp = intent.getComponent();
    if (comp == null) {
        if (intent.getSelector() != null) {
            intent = intent.getSelector();
            comp = intent.getComponent();
        }
    }
    //如果ComponentName已指定,說明滿足條件的Activity只有一個
    if (comp != null) {
        final List<ResolveInfo> list = new ArrayList<ResolveInfo>(1);
        final ActivityInfo ai = getActivityInfo(comp, flags, userId);
        if (ai != null) {
            final ResolveInfo ri = new ResolveInfo();
            ri.activityInfo = ai;
            list.add(ri);
        }
        return list;
    }

    // reader
    synchronized (mPackages) {
        final String pkgName = intent.getPackage();
        //未指定包名
        if (pkgName == null) {
            List<CrossProfileIntentFilter> matchingFilters =
                getMatchingCrossProfileIntentFilters(intent, resolvedType, userId);
            // Check for results that need to skip the current profile.
            ResolveInfo xpResolveInfo  = querySkipCurrentProfileIntents(matchingFilters, intent,
                    resolvedType, flags, userId);
            if (xpResolveInfo != null && isUserEnabled(xpResolveInfo.targetUserId)) {
                List<ResolveInfo> result = new ArrayList<ResolveInfo>(1);
                result.add(xpResolveInfo);
                return filterIfNotPrimaryUser(result, userId);
            }

            // Check for results in the current profile.
            //從所有應(yīng)用中查找
            List<ResolveInfo> result = mActivities.queryIntent(
                    intent, resolvedType, flags, userId);

            // Check for cross profile results.
            xpResolveInfo = queryCrossProfileIntents(
                    matchingFilters, intent, resolvedType, flags, userId);
            if (xpResolveInfo != null && isUserEnabled(xpResolveInfo.targetUserId)) {
                result.add(xpResolveInfo);
                Collections.sort(result, mResolvePrioritySorter);
            }
            result = filterIfNotPrimaryUser(result, userId);
            if (hasWebURI(intent)) {
                CrossProfileDomainInfo xpDomainInfo = null;
                final UserInfo parent = getProfileParent(userId);
                if (parent != null) {
                    xpDomainInfo = getCrossProfileDomainPreferredLpr(intent, resolvedType,
                            flags, userId, parent.id);
                }
                if (xpDomainInfo != null) {
                    if (xpResolveInfo != null) {
                        // If we didn't remove it, the cross-profile ResolveInfo would be twice
                        // in the result.
                        result.remove(xpResolveInfo);
                    }
                    if (result.size() == 0) {
                        result.add(xpDomainInfo.resolveInfo);
                        return result;
                    }
                } else if (result.size() <= 1) {
                    return result;
                }
                result = filterCandidatesWithDomainPreferredActivitiesLPr(intent, flags, result,
                        xpDomainInfo, userId);
                Collections.sort(result, mResolvePrioritySorter);
            }
            return result;
        }
        final PackageParser.Package pkg = mPackages.get(pkgName);
        //如果Intent中已指定包名
        if (pkg != null) {
            return filterIfNotPrimaryUser(
            //從滿足包名的應(yīng)用中進行查找
                    mActivities.queryIntentForPackage(
                        intent, resolvedType, flags, pkg.activities, userId),
                    userId);
        }
        return new ArrayList<ResolveInfo>();
    }
}

queryIntentActivities對傳入的Intent的不同情形分別做了對應(yīng)的處理旺拉,主要包括:

  • 如果Intent設(shè)置了ComponentName,說明系統(tǒng)中滿足該條件的Activity只存在一個假颇,通過getActivityInfo獲取得到;
  • 如果Intent沒有指定應(yīng)用包名符相,那么通過mActivities.queryIntent從系統(tǒng)所有應(yīng)用中查找滿足IntentActivity拆融;
  • 如果Intent已指定應(yīng)用包名,那么通過mActivities.queryIntentForPackage從滿足包名的應(yīng)用中查找符合IntentActivity啊终;

最終镜豹,queryIntentActivities方法會返回一個或多個符合Intent需求的Activity相關(guān)信息。

  1. 通過chooseBestActivity找到最符合Intent需求的Activity信息蓝牲,代碼具體實現(xiàn)如下:
private ResolveInfo chooseBestActivity(Intent intent, String resolvedType,
        int flags, List<ResolveInfo> query, int userId) {
    if (query != null) {
        final int N = query.size();
        if (N == 1) {
            //只有一個滿足趟脂,直接返回
            return query.get(0);
        } else if (N > 1) { //存在多個滿足
            final boolean debug = ((intent.getFlags() & Intent.FLAG_DEBUG_LOG_RESOLUTION) != 0);
            // If there is more than one activity with the same priority,
            // then let the user decide between them.
            //存在多個相同priority,只獲取前兩個Activity
            ResolveInfo r0 = query.get(0);
            ResolveInfo r1 = query.get(1);
            if (DEBUG_INTENT_MATCHING || debug) {
                Slog.v(TAG, r0.activityInfo.name + "=" + r0.priority + " vs "
                        + r1.activityInfo.name + "=" + r1.priority);
            }
            // If the first activity has a higher priority, or a different
            // default, then it is always desireable to pick it.
            //前兩個Activity的priority或default不同例衍,則默認選擇第一個Activity
            if (r0.priority != r1.priority
                    || r0.preferredOrder != r1.preferredOrder
                    || r0.isDefault != r1.isDefault) {
                return query.get(0);
            }
            // If we have saved a preference for a preferred activity for
            // this Intent, use that.
            //如果已保存了默認選擇昔期,則使用
            ResolveInfo ri = findPreferredActivity(intent, resolvedType,
                    flags, query, r0.priority, true, false, debug, userId);
            if (ri != null) {
                return ri;
            }
            ri = new ResolveInfo(mResolveInfo);
            ri.activityInfo = new ActivityInfo(ri.activityInfo);
            ri.activityInfo.applicationInfo = new ApplicationInfo(
                    ri.activityInfo.applicationInfo);
            if (userId != 0) {
                ri.activityInfo.applicationInfo.uid = UserHandle.getUid(userId,
                        UserHandle.getAppId(ri.activityInfo.applicationInfo.uid));
            }
            // Make sure that the resolver is displayable in car mode
            if (ri.activityInfo.metaData == null) ri.activityInfo.metaData = new Bundle();
            ri.activityInfo.metaData.putBoolean(Intent.METADATA_DOCK_HOME, true);
            return ri;
        }
    }
    return null;
}

chooseBestActivity從多個維度進行抉擇獲取一個最符合Intent條件的Activity,返回對應(yīng)的ResolveInfo:
1). 如果符合條件的只有一個應(yīng)用佛玄,則返回該應(yīng)用信息(源碼中確實只有 Launcher3 符合)硼一;
2). 如果存在多個Activity符合條件,則只選出前兩個梦抢,比較優(yōu)先級priority般贼、preferredOrderisDefault三個參數(shù)奥吩,選出優(yōu)先級最高的Activity哼蛆;
3). 如果prioritypreferredOrder霞赫、isDefault三個參數(shù)都一致腮介,則看下用戶是否配置了默認使用的Activity,是則返回該Activity對應(yīng)的ResolveInfo端衰;

到此叠洗,ActivityManagerService.startHomeActivityLocked從構(gòu)建一個HomeIntent,到使用Binder跨進程通訊通知PackageManagerService查找符合HomeIntent的一個最佳Activity就已經(jīng)完成了旅东。

接下來惕味,我們繼續(xù)分析ActivityManagerService.startHomeActivityLocked最后做的一件事。

  1. ActivityManagerService.startHomeActivityLocked最后通過mStackSupervisor.startHomeActivity啟動了 Launcher 應(yīng)用玉锌,具體代碼如下:
    frameworks\base\services\core\java\com\android\server\am\ActivityStackSupervisor.java
void startHomeActivity(Intent intent, ActivityInfo aInfo, String reason) {
    moveHomeStackTaskToTop(HOME_ACTIVITY_TYPE, reason);
    startActivityLocked(null /* caller */, intent, null /* resolvedType */, aInfo,
            null /* voiceSession */, null /* voiceInteractor */, null /* resultTo */,
            null /* resultWho */, 0 /* requestCode */, 0 /* callingPid */, 0 /* callingUid */,
            null /* callingPackage */, 0 /* realCallingPid */, 0 /* realCallingUid */,
            0 /* startFlags */, null /* options */, false /* ignoreTargetSecurity */,
            false /* componentSpecified */,
            null /* outActivity */, null /* container */,  null /* inTask */);
    if (inResumeTopActivity) {
        // If we are in resume section already, home activity will be initialized, but not
        // resumed (to avoid recursive resume) and will stay that way until something pokes it
        // again. We need to schedule another resume.
        scheduleResumeTopActivities();
    }
}

可以看到名挥,startHomeActivity方法最終調(diào)用的是startActivityLocked,這其實就進入Activity的啟動流程了主守,這里就不展開進行簡述了(關(guān)于Activity的啟動流程可以參考:TODO::)

到此禀倔,Launcher 應(yīng)用就會被啟動起來了。

下面参淫,我們繼續(xù)深入分析 Launcher 的源碼救湖,來看下 Launcher 界面是如何顯示 Android 系統(tǒng)所有應(yīng)用圖標(biāo),以及如何啟動對應(yīng)應(yīng)用涎才。

總結(jié)

最后鞋既,簡單總結(jié)一下 Launcher 啟動流程:

首先力九,System Server 通過調(diào)用startOtherServices啟動系統(tǒng)其他服務(wù),startOtherServices內(nèi)部會調(diào)用ActivityManagerService.systemReady(Runnable)邑闺,該方法內(nèi)部又會調(diào)用startHomeActivityLocked開始準(zhǔn)備開啟 Launcher跌前;startHomeActivityLocked內(nèi)部首先構(gòu)建了一個HomeIntent(具體內(nèi)容就是一個ComponentName=null,action="android.intent.action.MAIN"陡舅,category ="android.intent.category.HOME"Intent)抵乓,然后通過Binder跨進程通訊通知PackageManagerService從系統(tǒng)已安裝應(yīng)用中找到一個最符合該HomeIntentActivity信息;找到滿足HomeIntentActivity信息后靶衍,最后通過StackSupervisor.startHomeActivity啟動 Launcher 應(yīng)用灾炭。

Launcher 啟動流程圖如下圖所示:

Launcher 啟動流程

參考

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市颅眶,隨后出現(xiàn)的幾起案子蜈出,更是在濱河造成了極大的恐慌,老刑警劉巖涛酗,帶你破解...
    沈念sama閱讀 206,126評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件掏缎,死亡現(xiàn)場離奇詭異,居然都是意外死亡煤杀,警方通過查閱死者的電腦和手機眷蜈,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評論 2 382
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來沈自,“玉大人酌儒,你說我怎么就攤上這事】萃荆” “怎么了忌怎?”我有些...
    開封第一講書人閱讀 152,445評論 0 341
  • 文/不壞的土叔 我叫張陵,是天一觀的道長酪夷。 經(jīng)常有香客問我榴啸,道長,這世上最難降的妖魔是什么晚岭? 我笑而不...
    開封第一講書人閱讀 55,185評論 1 278
  • 正文 為了忘掉前任鸥印,我火速辦了婚禮,結(jié)果婚禮上坦报,老公的妹妹穿的比我還像新娘库说。我一直安慰自己,他們只是感情好片择,可當(dāng)我...
    茶點故事閱讀 64,178評論 5 371
  • 文/花漫 我一把揭開白布潜的。 她就那樣靜靜地躺著,像睡著了一般字管。 火紅的嫁衣襯著肌膚如雪啰挪。 梳的紋絲不亂的頭發(fā)上信不,一...
    開封第一講書人閱讀 48,970評論 1 284
  • 那天,我揣著相機與錄音亡呵,去河邊找鬼抽活。 笑死,一個胖子當(dāng)著我的面吹牛政己,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播掏愁,決...
    沈念sama閱讀 38,276評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼歇由,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了果港?” 一聲冷哼從身側(cè)響起沦泌,我...
    開封第一講書人閱讀 36,927評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎辛掠,沒想到半個月后谢谦,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,400評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡萝衩,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,883評論 2 323
  • 正文 我和宋清朗相戀三年回挽,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片猩谊。...
    茶點故事閱讀 37,997評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡千劈,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出牌捷,到底是詐尸還是另有隱情墙牌,我是刑警寧澤,帶...
    沈念sama閱讀 33,646評論 4 322
  • 正文 年R本政府宣布暗甥,位于F島的核電站喜滨,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏虽风。R本人自食惡果不足惜寄月,卻給世界環(huán)境...
    茶點故事閱讀 39,213評論 3 307
  • 文/蒙蒙 一内舟、第九天 我趴在偏房一處隱蔽的房頂上張望验游。 院中可真熱鬧耕蝉,春花似錦、人聲如沸蒜魄。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽签舞。三九已至儒搭,卻和暖如春搂鲫,著一層夾襖步出監(jiān)牢的瞬間默穴,已是汗流浹背蓄诽。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評論 1 260
  • 我被黑心中介騙來泰國打工仑氛, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留锯岖,地道東北人出吹。 一個月前我還...
    沈念sama閱讀 45,423評論 2 352
  • 正文 我出身青樓鸠珠,卻偏偏與公主長得像渐排,于是被迫代替她去往敵國和親驯耻。 傳聞我的和親對象是個殘疾皇子可缚,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,722評論 2 345

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