Android 編譯速度優(yōu)化黑科技 - RocketX

一、背景描述

二嘱丢、效果展示

三薪介、思路問題分析與模塊搭建

四、問題解決與實現(xiàn)

五屿讽、一天一個小驚喜

六昭灵、下一步展望

一、背景描述

在項目體量越來越大的情況下伐谈,編譯速度也隨著增長烂完,有時候一個修改需要等待長達好幾分鐘的編譯時間。
基于這種普遍的情況诵棵,推出了 RocketX ,通過在編譯流程 動態(tài) 替換 moduleaar 抠蚣,提高全量編譯的速度。讓你體驗到所有模塊都是 aar 的速度履澳,又能保留所有的 module 便于修改嘶窄,簡直完美!

二距贷、效果展示

2.1柄冲、測試項目介紹
  • 目標項目一共 3W+ 個類與資源文件,全量編譯 4min 左右

  • 通過 RocketX 全量增速之后的效果(每一個操作取 3 次平均值)

    build-speed.png

  • 項目依賴關(guān)系如下圖忠蝗,app 依賴 bm 業(yè)務(wù)模塊现横,bm 業(yè)務(wù)模塊依賴頂層 base/comm 模塊

架構(gòu)圖.png
  • rx(RocketX) 編譯 - 可以看到 rx(RocketX) 在無論哪一個模塊的編譯速度基本都是在控制在 30s 左右,因為只編譯 app 和 改動的模塊阁最,其他模塊是 aar 包不參與編譯戒祠。
  • 原生編譯 - 當 base/comm 模塊改動,底部的所有模塊都必須參與編譯速种。因為 app/bmxxx 模塊可能使用了 base 模塊中的接口或變量等姜盈,并且不知道是否有改動到。(那么速度就非常慢)
  • 原生編譯和RocketX 的編譯差距就體現(xiàn)在這里配阵,RocketX 少編了 60+ 個模塊馏颂,從而實現(xiàn)提速:
1641484922(1).png

三示血、思路問題分析與模塊搭建:

3.1、思路問題分析
  • 上個思維導(dǎo)圖饱亮,涉及到以下問題:


    未命名文件 (3).jpg
  1. 需要通過 gradle plugin 的形式動態(tài)修改沒有改動過的 module 依賴為 相對應(yīng)的 aar 依賴矾芙,如果 module 改動,退化成 project 工程依賴近上,這樣每次只有改動的 moduleapp 兩個模塊編譯剔宪。
  2. 需要把 implement/api moduleB,修改為implement/api aarB壹无,并且需要知道插件中如何加入 aar 依賴和剔除原有依賴
  3. 需要構(gòu)建 local maven 存儲未被修改的 module 對應(yīng)的 aar(也可以通過 flatDir 代替速度更快)
  4. 編譯流程啟動葱绒,需要找到哪一個 module 做了修改
  5. 需要遍歷每一個 module 的依賴關(guān)系進行置換, module 依賴怎么獲榷范А地淀?一次性能獲取到所有模塊依賴,還是分模塊各自回調(diào)岖是?修改其中一個模塊依賴關(guān)系會阻斷后面模塊依賴回調(diào)帮毁?
  6. 每一個 module 換變成 aar 之后,自身依賴的 child 依賴 (網(wǎng)絡(luò)依賴豺撑,aar),給到 parent module (如何找到所有 parent module) ? 還是直接給 app module ? 有沒有 appmodule 依賴斷掉的風(fēng)險烈疚? 這里需要出一個技術(shù)方案。
  7. 需要hook 編譯流程聪轿,完成后置換 loacal maven 中被修改的 aar
  8. 提供 AS 狀態(tài)欄 button, 實現(xiàn)開啟關(guān)閉功能爷肝,加速編譯還是讓開發(fā)者使用已經(jīng)習(xí)慣性的三角形 run 按鈕
3.2、模塊搭建
  • 依照上面的分析陆错,雖然問題很多灯抛,但是大致可以把整個項目分成以下幾塊:
image.png

四、問題解決與實現(xiàn):

4.1音瓷、如何手動添加 aar 依賴对嚼,分析implement 源碼實現(xiàn)入口在 DynamicAddDependencyMethods 中的 tryInvokeMethod 方法。他是一個動態(tài)語言的 methodMissing 功能
  • tryInvokeMethod 代碼分析
 public DynamicInvokeResult tryInvokeMethod(String name, Object... arguments) {
       //省略部分代碼 ...
       return DynamicInvokeResult.found(this.dependencyAdder.add(configuration, normalizedArgs.get(0), (Closure)null));
 }
  • dependencyAdder 實現(xiàn)是一個 DirectDependencyAdder
private class DirectDependencyAdder implements DependencyAdder<Dependency> {
        private DirectDependencyAdder() {
        }
        public Dependency add(Configuration configuration, Object dependencyNotation, @Nullable Closure configureAction) {
            return DefaultDependencyHandler.this.doAdd(configuration, dependencyNotation, configureAction);
        }
    }
  • 最后是在 DefaultDependencyHandler.this.doAdd 進行添加進去绳慎,而 DefaultDependencyHandlerproject可以獲取
public interface Project extends Comparable<Project>, ExtensionAware, PluginAware {
     ...
     DependencyHandler getDependencies(); 
     ...
}

  • doAdd 方法三個參數(shù)通過 debug 源碼發(fā)現(xiàn)纵竖,configuration 就是 "implementation", "api", "compileOnly" 這三個字符串生成的對象,dependencyNotation 是一個 LinkHashMap 有兩個鍵值對偷线,分別是 name:aarName, ext:aar,最后一個configureActionnull 就可以了,調(diào)用 project.dependencies.add 最終會調(diào)到 doAdd 方法沽甥,也就是說直接調(diào)用 add 即可声邦。

 public Dependency add(String configurationName, Object dependencyNotation) {
        return this.add(configurationName, dependencyNotation, (Closure)null);
    }

    public Dependency add(String configurationName, Object dependencyNotation, Closure configureClosure) {
       //這里直接調(diào)用到了 doAdd 
        return this.doAdd(this.configurationContainer.getByName(configurationName), dependencyNotation, configureClosure);
    }
    
  • 那么依葫蘆畫瓢添加 aar/jar 的實現(xiàn)代碼:configNamechildProject 中的 configName ,也就是 "implementation", "api", "compileOnly" 這三個字符串摆舟,原封不動拿過來:
    fun addAarDependencyToProject(aarName: String, configName: String, project: Project) {
        //添加 aar 依賴 以下代碼等同于 api/implementation/xxx (name: 'libaccount-2.0.0', ext: 'aar'),源碼使用 linkedMap
        val map = linkedMapOf<String, String>()
        map.put("name", aarName)
        map.put("ext", "aar")
        project.dependencies.add(configName, map)
    }
4.2亥曹、localMave 優(yōu)先使用 flatDir 實現(xiàn)通過指定一個緩存目錄 getLocalMavenCacheDir 把生成 aar/jar 包丟進去,依賴修改時候通過 上面的 4.1 添加對應(yīng)的 aar 即可:
  fun flatDirs() {
        val map = mutableMapOf<String, File>()
        map.put("dirs", File(getLocalMavenCacheDir()))
        appProject.rootProject.allprojects {
            it.repositories.flatDir(map)
        }
    }
4.3邓了、編譯流程啟動,需要找到哪一個 module做了修改
  • 使用遍歷整個項目的文件的 lastModifyTime 去做實現(xiàn)
  • 已每一個 module 為一個粒度媳瞪,遞歸遍歷當前 module 的文件骗炉,把每個文件的 lastModifyTime 整合計算得出一個唯一標識 countTime
  • 通過 countTime 與上一次的作對比,相同說明沒改動,不同則改動. 并需要同步計算后的 countTime 到本地緩存中
  • 整體 3W 個文件耗時 1.2s 可以接受,目前在類 ChangeModuleUtils.kt 進行實現(xiàn)
4.4蛇受、 module 依賴關(guān)系獲取
  • 目的找到生成整個項目的依賴關(guān)系圖時機句葵,并在此處生成依賴圖解析器。時機要在run 編譯的 task之前兢仰,確保依賴關(guān)系獲取后替換能生效乍丈,而且要在全局module依賴圖已經(jīng)生成之后(也就是在 執(zhí)行完build.gradle),那么有目前有兩種方式:DependencyResolutionListenerprojectsEvaluated 把将,但是目前都有問題 轻专。
  1. 通過監(jiān)聽DependencyResolutionListener,并在beforeResolve回調(diào)方法處理:
  public interface DependencyResolutionListener {
    void beforeResolve(ResolvableDependencies var1);

    void afterResolve(ResolvableDependencies var1);
}

   project.gradle.addListener(DependencyResolutionListener listener)
  1. 但是出現(xiàn)的問題是beforeResolve 會回調(diào)多次察蹲,并且執(zhí)行完畢每一個modulebuild.gradle 把依賴解析出來則會回調(diào)请垛。那么如果在業(yè)務(wù)層處理一下,等待到最后一個module 回調(diào)完畢洽议,再通過 project.configurations 獲取到所有 module 的依賴圖宗收?答案是可以的,但是 時機已經(jīng)晚了绞铃,等到最后一個module 解析完畢之后 回調(diào) beforeResolve 镜雨,再去修改依賴關(guān)系會報以下異常(無法修改依賴關(guān)系):
Cannot change dependencies of dependency configuration ':app:implementation' after it has been included in dependency resolution.
  1. 換個法子通過project.gradle.projectsEvaluated {}回調(diào)之后拿到所有 module 依賴關(guān)系并且去修改。依賴圖可以拿到儿捧,但是修改依賴關(guān)系還是會報異常荚坞,時機終究還是晚了。

  2. 那么過一遍 gradle 的生命周期(網(wǎng)上摳圖)

20190530121157928.png
  1. 發(fā)現(xiàn)執(zhí)行完畢build.gradle(解析完依賴)之后的生命周期只有:Project.afterEvaluateGradle.projectsEvaluated(在3中試過不行)菲盾。那么只有afterEvaluate 颓影,afterEvaluate 還是每 個module 執(zhí)行完畢依賴會回調(diào)一次,監(jiān)聽最后一個module 回調(diào)的時機懒鉴,并修改依舊還是報相同錯誤诡挂。綜上在生命周期找不到合適的時機
  2. 直接找 gradle 源碼 找到異常出現(xiàn)的代碼,具體是在 :DefaultConfiguration.preventIllegalMutation 這個方法
private void preventIllegalParentMutation(MutationType type) {
        if (type != MutationType.DEPENDENCY_ATTRIBUTES) {
            if (this.resolvedState == InternalState.ARTIFACTS_RESOLVED) {
                throw new InvalidUserDataException(String.format("Cannot change %s of parent of %s after it has been resolved", type, this.getDisplayName()));
            } else if (this.resolvedState == InternalState.GRAPH_RESOLVED && type == MutationType.DEPENDENCIES) {
                throw new InvalidUserDataException(String.format("Cannot change %s of parent of %s after task dependencies have been resolved", type, this.getDisplayName()));
            }
        }
    }
    
  1. 出現(xiàn)這個異常的原因主要是typeMutationType.DEPENDENCY_ATTRIBUTES临谱,那么在哪里賦值為 DEPENDENCY_ATTRIBUTES璃俗,并趕在它之前修改依賴不就解決了,給出相關(guān) gradle 源碼調(diào)用流程:
未命名文件 (1).png
  1. 那么基本是通過apply plugin: 'com.android.application' 開始調(diào)用進來并通過設(shè)置
    project.gradle.projectsEvaluated{} 監(jiān)聽回調(diào)悉默,把type 設(shè)置成 DEPENDENCY_ATTRIBUTES 城豁,通過這次排查源碼知道 gradle 也是通過projectsEvaluated 這個生命周期才開始解決依賴關(guān)系(這時候所有的module 的依賴圖才全部生成)

  2. 那么最后的解決方法 要在 projectsEvaluated 去修改依賴,但是要趕在 GradlePluginUtils 里面的一個監(jiān)聽者之前抄课,通過反射剔除設(shè)置進去的所有 projectsEvaluated監(jiān)聽者(其實就是 Action 的匿名內(nèi)部對象)唱星,先執(zhí)行 rockectXPlugin 的監(jiān)聽者雳旅,后面再執(zhí)行其他的監(jiān)聽者。大概是這樣子:

//AppProjectDependencies.kt
    init {
        val projectsEvaluatedList = hookProjectsEvaluatedAction()
        project.gradle.projectsEvaluated {
            //先執(zhí)行重依賴
            resolveDenpendency()
            //后執(zhí)行移除的監(jiān)聽(主要調(diào)整執(zhí)行順序间聊,重依賴才能生效和不報錯攒盈,可能有AGP 版本兼容問題)
            val clazz = Class.forName("org.gradle.api.invocation.Gradle")
            val method = clazz.getDeclaredMethod("projectsEvaluated", Action::class.java)
            val mMethodInvocation = MethodInvocation(method, arrayOf(it))
            projectsEvaluatedList.forEach {
                it.dispatch(mMethodInvocation)
            }

        }
    }

    //把所有 監(jiān)聽了 projectsEvaluated 的匿名內(nèi)部類移除
    fun hookProjectsEvaluatedAction(): List<BroadcastDispatch<BuildListener>> {
        var removeDispatch = mutableListOf<BroadcastDispatch<BuildListener>>()
        try {
            var buildListenerBroadcast: ListenerBroadcast<BuildListener>? = null
            val fBuildListenerBroadcast =
                    DefaultGradle::class.java.getDeclaredField("buildListenerBroadcast")
            fBuildListenerBroadcast.isAccessible = true
            buildListenerBroadcast =
                    fBuildListenerBroadcast.get(project.gradle) as? ListenerBroadcast<BuildListener>

            val fBroadcast = ListenerBroadcast::class.java.getDeclaredField("broadcast")
            fBroadcast.isAccessible = true
            val broadcast: BroadcastDispatch<BuildListener>? =
                    fBroadcast.get(buildListenerBroadcast) as? BroadcastDispatch<BuildListener>
            val fDispatchers = broadcast?.javaClass?.getDeclaredField("dispatchers")
            fDispatchers?.isAccessible = true
            val dispatchers: ArrayList<BroadcastDispatch<BuildListener>>? =
                    fDispatchers?.get(broadcast) as? ArrayList<BroadcastDispatch<BuildListener>>

            val clazz =
                    Class.forName("org.gradle.internal.event.BroadcastDispatch\$ActionInvocationHandler")
            val iterator = dispatchers?.iterator()
            iterator?.let {
                while (iterator.hasNext()) {
                    try {
                        val next = iterator.next()
                        val fDispatch = next.javaClass.getDeclaredField("dispatch")
                        fDispatch.isAccessible = true
                        val dispatch: Any? = fDispatch.get(next)
                        val fMethodName = clazz.getDeclaredField("methodName")
                        fMethodName.isAccessible = true
                        val methodName = fMethodName.get(dispatch) as? String
                        if (methodName?.contains("projectsEvaluated") == true) {
                            removeDispatch.add(next)
                            iterator.remove()
                        }
                    } catch (ignore: Exception) {
                    }
                }
            }
        } catch (ignore: Exception) {
        }
        return removeDispatch
    }
  1. 至此完美解決修改依賴時機問題,更多詳情可查看 issue18
  • 如何獲取每個module 的依賴哎榴,依賴就藏在 Configuration.dependencies型豁,那么通過project.configurations.maybeCreate(configName) 找到所有的 Configuration 對象,就能得到每個moduledependencies
4.5叹话、 module 依賴關(guān)系 project 替換成 aar 技術(shù)方案
  • 每一個 module 依賴關(guān)系替換的遍歷順序是無序的偷遗,所以技術(shù)方案需要支持無序的替換
  • 目前使用的方案是:如果當前模塊 A 未改動,需要把 A 通過 localMaven 置換成 A.aar,并把 A.aar 以及 Achild 依賴驼壶,給到第一層的 parent module 即可氏豌。(可能會質(zhì)疑如果 parent module 也是 aar 怎么辦,其實這塊也是沒有問題的热凹,這里就不展開說了泵喘,篇幅太長)
  • 為什么要給到 parent 不能直接給到 app ,下圖一個簡單的示例如果 B.aar 不給 A 模塊的話般妙,A 使用 B 模塊的接口不見了纪铺,會導(dǎo)致編譯不過
    8b6e18135662895b9fae8e9940f3aed.png
  • 給出整體項目替換的技術(shù)方案演示:


    RocketXPlugin (3).jpg
  • 整體的實現(xiàn)在 DependenciesHelper.kt 這個類中,由于講起來篇幅太長碟渺,有興趣可查閱開源庫代碼
4.5鲜锚、hook 編譯流程,完成后置換 loacal maven 中被修改的 aar
  • 點擊三角形 run苫拍,執(zhí)行的命令是 app:assembleDebug , 需要在 assembleDebug 后面補一個 uploadLocalMavenTask, 通過 finalizedBy 把我們的 task 運行起來去同步修改后的 aar
val localMavenTask = childProject.tasks.maybeCreate("uploadLocalMaven"+buildType.capitalize(),LocalMavenTask::class.java)
localMavenTask.localMaven = this@AarFlatLocalMaven
bundleTask?.finalizedBy(localMavenTask)
4.6芜繁、提供 AS 狀態(tài)欄 button,小火箭按鈕一個噴火一個沒有噴火绒极,代表 enable/disable , 一個 掃把clean rockectx 的緩存骏令,需要通過編寫 intellij idea plugin 即可,也就是 目前擁有兩個插件了垄提,一個 gradle 插件一個 AS 插件:
image.png

五榔袋、一天一個小驚喜( bug 較多)

5.1、發(fā)現(xiàn)點擊 run 按鈕 铡俐,執(zhí)行的命令是 app:assembleDebug 凰兑,各個子 moduleoutput 并沒有打包出 aar

解決:通過研究 gradle 源碼發(fā)現(xiàn)打包是由 bundle${Flavor}${BuildType}Aar 這個task執(zhí)行出來,那么只需要將各個模塊對應(yīng)的 task 找到并注入到 app:assembleDebug 之后運行即可:

        android.applicationVariants.forEach {
            getAppAssembleTask(ASSEMBLE + it.flavorName.capitalize() + it.buildType.name.capitalize())?.let { task ->
                    hookBundleAarTask(task, it.buildType.name)
                }
        }
5.2审丘、發(fā)現(xiàn)運行起來后存在多個 jar 包重復(fù)問題

解決: implementation fileTree(dir: "libs", include: ["*.jar"]) jar 依賴不能交到 parent module吏够,jar 包會打進 aar 中的lib 可直接剔除。通過以下代碼可以判斷:

// 這里的依賴是以下兩種: 無需添加在 parent ,因為 jar 包直接進入 自身的 aar 中的libs 文件夾
if (childDepency is DefaultSelfResolvingDependency && (childDepency.files is DefaultConfigurableFileCollection || childDepency.files is DefaultConfigurableFileTree)) {
// 這里的依賴是以下兩種: 無需添加在 parent 稿饰,因為 jar 包直接進入 自身的 aar 中的libs 文件夾
//    implementation rootProject.files("libs/tingyun-ea-agent-android-2.15.4.jar")
//    implementation fileTree(dir: "libs", include: ["*.jar"])
} else { 
    parentProject.key.dependencies.add(childConfig.name, childDepency)
}
5.3、發(fā)現(xiàn) aar/jar 存在多種依賴方式
 implementation (name: 'libXXX', ext: 'aar') 
 implementation files("libXXX.aar")

解決:使用第一種露泊,第二種會合并進aar,導(dǎo)致類重復(fù)問題

5.4喉镰、發(fā)現(xiàn) aar 新姿勢依賴
configurations.maybeCreate("default")
artifacts.add("default", file('lib-xx.aar'))

上面代碼把 aar 做了一個單獨的 module 給到其他 module 依賴,default config 其實是 module 最終輸出 aar 的持有者惭笑,default config 可以持有一個 列表的aar 侣姆,所以把 aar 手動添加到 default config,也相當于當前 module 打包出來的產(chǎn)物沉噩。

解決:通過 childProject.configurations.maybeCreate("default").artifacts 找到所有添加進來的 aar 捺宗,單獨發(fā)布 localmaven

   fun getAarByArtifacts(childProject: Project): MutableList<String> {
        //找到當前所有通過 artifacts.add("default", file('xxx.aar')) 依賴進來的 aar
        var listArtifact = mutableListOf<DefaultPublishArtifact>()
        var aarList = mutableListOf<String>()
        childProject.configurations.maybeCreate("default").artifacts?.forEach {
            if (it is DefaultPublishArtifact && "aar".equals(it.type)) {
                listArtifact.add(it)
            }
        }

        //拷貝一份到 localMaven
        listArtifact.forEach {
            it.file.copyTo(File(FileUtil.getLocalMavenCacheDir(), it.file.name), true)
            //剔除后綴 (.aar)
            aarList.add(removeExtension(it.file.name))
        }

        return aarList
    }
    
5.5、發(fā)現(xiàn) android module 打包出來可以是 jar

解決:通過找到名字叫做 jartask川蒙,并且在 jar task 后面注入 uploadLocalMaven task,代碼實現(xiàn)在 JarFlatLocalMaven.kt

5.6蚜厉、發(fā)現(xiàn) arouterbugtransform 沒有通過 outputProvider.deleteAll() 清理舊的緩存

解決:詳情查看 issue畜眨,結(jié)果arouter 問題是解決了昼牛,代碼也是合并了。但并沒有發(fā)布新的插件版本到 mavenCentral康聂,于是先自行幫 arouter 解決一下贰健。然而arouter 并沒有啟動 增量編譯,導(dǎo)致 DexArchiveBuilderTask 運行巨慢恬汁,也就是打 dex 包很慢伶椿,項目中我重改了 arouter 插件源碼支持 TransForm 增量速度提升一倍, 具體細節(jié)就下節(jié)和 dex 速度優(yōu)化一起講。

六氓侧、下一步展望

目前初步的版本已經(jīng)能夠在在項目 run 起來脊另,但是還是有很多小問題不斷的冒出并解決,路漫漫其修遠兮甘苍,吾將上下而求索尝蠕。。

下步計劃:

  • dexBuild task 優(yōu)化
  • 解決各種兼容性問題

目前插件趨于穩(wěn)定载庭,喜歡嘗鮮的朋友可以通過github教程接入看彼,一起關(guān)注后期進展。后續(xù) issue 的解決思路都會在本文不斷更新囚聚,如果你喜歡本文就給我們 star 吧靖榕。
github開源地址

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市顽铸,隨后出現(xiàn)的幾起案子茁计,更是在濱河造成了極大的恐慌,老刑警劉巖谓松,帶你破解...
    沈念sama閱讀 218,755評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件星压,死亡現(xiàn)場離奇詭異践剂,居然都是意外死亡,警方通過查閱死者的電腦和手機娜膘,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評論 3 395
  • 文/潘曉璐 我一進店門逊脯,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人竣贪,你說我怎么就攤上這事军洼。” “怎么了演怎?”我有些...
    開封第一講書人閱讀 165,138評論 0 355
  • 文/不壞的土叔 我叫張陵匕争,是天一觀的道長。 經(jīng)常有香客問我爷耀,道長甘桑,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,791評論 1 295
  • 正文 為了忘掉前任歹叮,我火速辦了婚禮扇住,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘盗胀。我一直安慰自己艘蹋,他們只是感情好,可當我...
    茶點故事閱讀 67,794評論 6 392
  • 文/花漫 我一把揭開白布票灰。 她就那樣靜靜地躺著女阀,像睡著了一般。 火紅的嫁衣襯著肌膚如雪屑迂。 梳的紋絲不亂的頭發(fā)上浸策,一...
    開封第一講書人閱讀 51,631評論 1 305
  • 那天,我揣著相機與錄音惹盼,去河邊找鬼庸汗。 笑死,一個胖子當著我的面吹牛手报,可吹牛的內(nèi)容都是我干的蚯舱。 我是一名探鬼主播,決...
    沈念sama閱讀 40,362評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼掩蛤,長吁一口氣:“原來是場噩夢啊……” “哼枉昏!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起揍鸟,我...
    開封第一講書人閱讀 39,264評論 0 276
  • 序言:老撾萬榮一對情侶失蹤兄裂,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體晰奖,經(jīng)...
    沈念sama閱讀 45,724評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡谈撒,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了匾南。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片港华。...
    茶點故事閱讀 40,040評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖午衰,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情冒萄,我是刑警寧澤臊岸,帶...
    沈念sama閱讀 35,742評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站尊流,受9級特大地震影響帅戒,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜崖技,卻給世界環(huán)境...
    茶點故事閱讀 41,364評論 3 330
  • 文/蒙蒙 一逻住、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧迎献,春花似錦瞎访、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,944評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至冀瓦,卻和暖如春伴奥,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背翼闽。 一陣腳步聲響...
    開封第一講書人閱讀 33,060評論 1 270
  • 我被黑心中介騙來泰國打工拾徙, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人感局。 一個月前我還...
    沈念sama閱讀 48,247評論 3 371
  • 正文 我出身青樓尼啡,卻偏偏與公主長得像,于是被迫代替她去往敵國和親询微。 傳聞我的和親對象是個殘疾皇子玄叠,可洞房花燭夜當晚...
    茶點故事閱讀 44,979評論 2 355

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

  • 16宿命:用概率思維提高你的勝算 以前的我是風(fēng)險厭惡者,不喜歡去冒險拓提,但是人生放棄了冒險读恃,也就放棄了無數(shù)的可能。 ...
    yichen大刀閱讀 6,052評論 0 4
  • 公元:2019年11月28日19時42分農(nóng)歷:二零一九年 十一月 初三日 戌時干支:己亥乙亥己巳甲戌當月節(jié)氣:立冬...
    石放閱讀 6,880評論 0 2
  • 今天上午陪老媽看病,下午健身房跑步寺惫,晚上想想今天還沒有斷舍離疹吃,馬上做,衣架和旁邊的的布衣架西雀,一看亂亂萨驶,又想想自己是...
    影子3623253閱讀 2,913評論 1 8