tinker熱修復(fù)

1.熱修復(fù):

熱修復(fù)從原理上說應(yīng)該是屬于插件化的一類,我們可以用熱修復(fù)來處理線上緊急的bug,而不需要提示用戶重新發(fā)版

這里對比下常見的熱修復(fù)優(yōu)缺點(diǎn):

tinker對比圖

2.插件化:

插件化中通過DexClassLoader來加載類,將各個子bundle載入到宿主apk中,這個過程在第一次啟動的時候會耗費(fèi)一定時間,相關(guān)博客請參考http://www.reibang.com/p/43a8a9b932de,這里不做詳細(xì)描述.

3.增量更新:

增量更新的原理就是通過比較新apk和舊的apk,通過工具可以生成拆分包patch,然后客戶端需要處理的就是將old.apk與patch合并成新的apk,最后進(jìn)行安裝.但是這里合并完成后會進(jìn)行對比,兩個apk的md5或者sha1是否一致.參見鴻洋的博客


下面我們就來具體實(shí)現(xiàn)下騰訊tinker的集成與使用(也可以參考官方wiki):

步驟1:

在工程外部添加對tinker的支持:

dependencies {

classpath'com.android.tools.build:gradle:2.3.3'

classpath ('com.tencent.tinker:tinker-patch-gradle-plugin:1.8.0')

}

在工程里面添加需要的jar:

//apply tinker插件

applyplugin:'com.tencent.tinker.patch'

//可選奠骄,用于生成application類

provided('com.tencent.tinker:tinker-android-anno:1.8.0')

//tinker的核心庫

compile('com.tencent.tinker:tinker-android-lib:1.8.0')

然后配置gradle中新增tinker組在工具欄選項(xiàng),配置編譯輸出路徑和基本參數(shù):

defaultConfig中添加

/**

* buildConfig can change during patch!

* we can use the newly value when patch

*/

buildConfigField"String","MESSAGE","\"I am the base apk\""

//? ? ? ? buildConfigField "String", "MESSAGE", "\"I am the patch apk\""

/**

* client version would update with patch

* so we can get the newly git version easily!

*/

buildConfigField"String","TINKER_ID","\"${getTinkerIdValue()}\""

buildConfigField"String","PLATFORM","\"all\""


其他需要添加的直接貼出來:

defgitSha() {

try{

String gitRev ='git rev-parse --short HEAD'.execute(null, project.rootDir).text.trim()

if(gitRev ==null) {

throw newGradleException("can't get git rev, you should add git to system path or just input test value, such as 'testTinkerId'")

}

returngitRev

}catch(Exception e) {

throw newGradleException("can't get git rev, you should add git to system path or just input test value, such as 'testTinkerId'")

}

}

defbakPath = file("${buildDir}/bakApk/")

/**

* you can use assembleRelease to build you base apk

* use tinkerPatchRelease -POLD_APK=? -PAPPLY_MAPPING=? -PAPPLY_RESOURCE= to build patch

* add apk from the build/bakApk

*/

ext {

//for some reason, you may want to ignore tinkerBuild, such as instant run debug build?

tinkerEnabled =true

//for normal build

//old apk file to build patch apk

tinkerOldApkPath ="${bakPath}/Codes-old.apk"

//proguard mapping file to build patch apk

tinkerApplyMappingPath ="${bakPath}/Codes-debug-mapping.txt"

//resource R.txt to build patch apk, must input if there is resource changed

tinkerApplyResourcePath ="${bakPath}/Codes-debug-R.txt"

//only use for build all flavor, if not, just ignore this field

tinkerBuildFlavorDirectory ="${bakPath}/Codes-debug"

}

defgetOldApkPath() {

returnhasProperty("OLD_APK") ? OLD_APK : ext.tinkerOldApkPath

}

defgetApplyMappingPath() {

returnhasProperty("APPLY_MAPPING") ? APPLY_MAPPING : ext.tinkerApplyMappingPath

}

defgetApplyResourceMappingPath() {

returnhasProperty("APPLY_RESOURCE") ? APPLY_RESOURCE : ext.tinkerApplyResourcePath

}

defgetTinkerIdValue() {

returnhasProperty("TINKER_ID") ? TINKER_ID : gitSha()

}

defbuildWithTinker() {

returnhasProperty("TINKER_ENABLE") ? TINKER_ENABLE : ext.tinkerEnabled

}

defgetTinkerBuildFlavorDirectory() {

returnext.tinkerBuildFlavorDirectory

}

if(buildWithTinker()) {

applyplugin:'com.tencent.tinker.patch'

tinkerPatch {

/**

* necessary撕彤,default 'null'

* the old apk path, use to diff with the new apk to build

* add apk from the build/bakApk

*/

oldApk = getOldApkPath()

/**

* optional涡匀,default 'false'

* there are some cases we may get some warnings

* if ignoreWarning is true, we would just assert the patch process

* case 1: minSdkVersion is below 14, but you are using dexMode with raw.

*? ? ? ? it must be crash when load.

* case 2: newly added Android Component in AndroidManifest.xml,

*? ? ? ? it must be crash when load.

* case 3: loader classes in dex.loader{} are not keep in the main dex,

*? ? ? ? it must be let tinker not work.

* case 4: loader classes in dex.loader{} changes,

*? ? ? ? loader classes is ues to load patch dex. it is useless to change them.

*? ? ? ? it won't crash, but these changes can't effect. you may ignore it

* case 5: resources.arsc has changed, but we don't use applyResourceMapping to build

*/

ignoreWarning =false

/**

* optional,default 'true'

* whether sign the patch file

* if not, you must do yourself. otherwise it can't check success during the patch loading

* we will use the sign config with your build type

*/

useSign =true

/**

* optional孝扛,default 'true'

* whether use tinker to build

*/

tinkerEnable = buildWithTinker()

/**

* Warning, applyMapping will affect the normal android build!

*/

buildConfig {

/**

* optional,default 'null'

* if we use tinkerPatch to build the patch apk, you'd better to apply the old

* apk mapping file if minifyEnabled is enable!

* Warning:

* you must be careful that it will affect the normal assemble build!

*/

applyMapping = getApplyMappingPath()

/**

* optional拨脉,default 'null'

* It is nice to keep the resource id from R.txt file to reduce java changes

*/

applyResourceMapping = getApplyResourceMappingPath()

/**

* necessary辨赐,default 'null'

* because we don't want to check the base apk with md5 in the runtime(it is slow)

* tinkerId is use to identify the unique base apk when the patch is tried to apply.

* we can use git rev, svn rev or simply versionCode.

* we will gen the tinkerId in your manifest automatic

*/

//? ? ? ? ? tinkerId = getTinkerIdValue()

tinkerId ="tinkerId"

/**

* if keepDexApply is true, class in which dex refer to the old apk.

* open this can reduce the dex diff file size.

*/

keepDexApply =false

/**

* optional, default 'false'

* Whether tinker should treat the base apk as the one being protected by app

* protection tools.

* If this attribute is true, the generated patch package will contain a

* dex including all changed classes instead of any dexdiff patch-info files.

*/

isProtectedApp =false

}

dex {

/**

* optional,default 'jar'

* only can be 'raw' or 'jar'. for raw, we would keep its original format

* for jar, we would repack dexes with zip format.

* if you want to support below 14, you must use jar

* or you want to save rom or check quicker, you can use raw mode also

*/

dexMode ="jar"

/**

* necessary诲祸,default '[]'

* what dexes in apk are expected to deal with tinkerPatch

* it support * or ? pattern.

*/

pattern = ["classes*.dex",

"assets/secondary-dex-?.jar"]

/**

* necessary浊吏,default '[]'

* Warning, it is very very important, loader classes can't change with patch.

* thus, they will be removed from patch dexes.

* you must put the following class into main dex.

* Simply, you should add your own application {@codetinker.sample.android.SampleApplication}

* own tinkerLoader, and the classes you use in them

*

*/

loader = [

//use sample, let BaseBuildInfo unchangeable with tinker

"com.zte.rs.RSApplication"

]

}

lib {

/**

* optional而昨,default '[]'

* what library in apk are expected to deal with tinkerPatch

* it support * or ? pattern.

* for library in assets, we would just recover them in the patch directory

* you can get them in TinkerLoadResult with Tinker

*/

pattern = ["lib/*/*.so"]

}

res {

/**

* optional,default '[]'

* what resource in apk are expected to deal with tinkerPatch

* it support * or ? pattern.

* you must include all your resources in apk here,

* otherwise, they won't repack in the new apk resources.

*/

pattern = ["res/*","assets/*","resources.arsc","AndroidManifest.xml"]

/**

* optional找田,default '[]'

* the resource file exclude patterns, ignore add, delete or modify resource change

* it support * or ? pattern.

* Warning, we can only use for files no relative with resources.arsc

*/

ignoreChange = ["assets/sample_meta.txt"]

/**

* default 100kb

* for modify resource, if it is larger than 'largeModSize'

* we would like to use bsdiff algorithm to reduce patch file size

*/

largeModSize =100

}

packageConfig {

/**

* optional歌憨,default 'TINKER_ID, TINKER_ID_VALUE' 'NEW_TINKER_ID, NEW_TINKER_ID_VALUE'

* package meta file gen. path is assets/package_meta.txt in patch file

* you can use securityCheck.getPackageProperties() in your ownPackageCheck method

* or TinkerLoadResult.getPackageConfigByName

* we will get the TINKER_ID from the old apk manifest for you automatic,

* other config files (such as patchMessage below)is not necessary

*/

configField("patchMessage","tinker is sample to use")

/**

* just a sample case, you can use such as sdkVersion, brand, channel...

* you can parse it in the SamplePatchListener.

* Then you can use patch conditional!

*/

configField("platform","all")

/**

* patch version via packageConfig

*/

configField("patchVersion","1.0")

}

//or you can add config filed outside, or get meta value from old apk

//project.tinkerPatch.packageConfig.configField("test1", project.tinkerPatch.packageConfig.getMetaDataFromOldApk("Test"))

//project.tinkerPatch.packageConfig.configField("test2", "sample")

/**

* if you don't use zipArtifact or path, we just use 7za to try

*/

sevenZip {

/**

* optional,default '7za'

* the 7zip artifact path, it will use the right 7za with your platform

*/

zipArtifact ="com.tencent.mm:SevenZip:1.1.10"

/**

* optional墩衙,default '7za'

* you can specify the 7za path yourself, it will overwrite the zipArtifact value

*/

//? ? ? ? path = "/usr/local/bin/7za"

}

}

List flavors =newArrayList<>();

project.android.productFlavors.each { flavor ->

flavors.add(flavor.name)

}

booleanhasFlavors = flavors.size() >0

defdate =newDate().format("MMdd-HH-mm-ss")

/**

* bak apk and mapping

*/

android.applicationVariants.all { variant ->

/**

* task type, you want to bak

*/

deftaskName = variant.name

tasks.all {

if("assemble${taskName.capitalize()}".equalsIgnoreCase(it.name)) {

it.doLast {

copy {

deffileNamePrefix ="${project.name}-${variant.baseName}"

defnewFileNamePrefix = hasFlavors ?"${fileNamePrefix}":"${fileNamePrefix}-${date}"

defdestPath = hasFlavors ? file("${bakPath}/${project.name}-${date}/${variant.flavorName}") : bakPath

from variant.outputs.outputFile

into destPath

rename { String fileName ->

fileName.replace("${fileNamePrefix}.apk","${newFileNamePrefix}.apk")

}

from"${buildDir}/outputs/mapping/${variant.dirName}/mapping.txt"

into destPath

rename { String fileName ->

fileName.replace("mapping.txt","${newFileNamePrefix}-mapping.txt")

}

from"${buildDir}/intermediates/symbols/${variant.dirName}/R.txt"

into destPath

rename { String fileName ->

fileName.replace("R.txt","${newFileNamePrefix}-R.txt")

}

}

}

}

}

}

project.afterEvaluate {

//sample use for build all flavor for one time

if(hasFlavors) {

task(tinkerPatchAllFlavorRelease) {

group ='tinker'

deforiginOldPath = getTinkerBuildFlavorDirectory()

for(String flavor : flavors) {

deftinkerTask = tasks.getByName("tinkerPatch${flavor.capitalize()}Release")

dependsOn tinkerTask

defpreAssembleTask = tasks.getByName("process${flavor.capitalize()}ReleaseManifest")

preAssembleTask.doFirst {

String flavorName = preAssembleTask.name.substring(7,8).toLowerCase() + preAssembleTask.name.substring(8, preAssembleTask.name.length() -15)

project.tinkerPatch.oldApk ="${originOldPath}/${flavorName}/${project.name}-${flavorName}-release.apk"

project.tinkerPatch.buildConfig.applyMapping ="${originOldPath}/${flavorName}/${project.name}-${flavorName}-release-mapping.txt"

project.tinkerPatch.buildConfig.applyResourceMapping ="${originOldPath}/${flavorName}/${project.name}-${flavorName}-release-R.txt"

}

}

}

task(tinkerPatchAllFlavorDebug) {

group ='tinker'

deforiginOldPath = getTinkerBuildFlavorDirectory()

for(String flavor : flavors) {

deftinkerTask = tasks.getByName("tinkerPatch${flavor.capitalize()}Debug")

dependsOn tinkerTask

defpreAssembleTask = tasks.getByName("process${flavor.capitalize()}DebugManifest")

preAssembleTask.doFirst {

String flavorName = preAssembleTask.name.substring(7,8).toLowerCase() + preAssembleTask.name.substring(8, preAssembleTask.name.length() -13)

project.tinkerPatch.oldApk ="${originOldPath}/${flavorName}/${project.name}-${flavorName}-debug.apk"

project.tinkerPatch.buildConfig.applyMapping ="${originOldPath}/${flavorName}/${project.name}-${flavorName}-debug-mapping.txt"

project.tinkerPatch.buildConfig.applyResourceMapping ="${originOldPath}/${flavorName}/${project.name}-${flavorName}-debug-R.txt"

}

}

}

}

}

這里更具具體需求去改變參數(shù)的值


2:分析生成的各個文件


目錄

以debug為例,首先我們先運(yùn)行assembleDebug,可以看到build下生成了bakApk文件夾,這里將outputs中apk的賦值到bakApk下取名Codes-old.apk,跟gradle中配置的文件名保存一致,然后我們修改bug后點(diǎn)擊TinkerPatchDebug,可以看到會生成timerPatch文件夾,這里面詳細(xì)的記錄了編譯中所對比的文件和加載中混淆的說明等等,這里我們只需要選著patch_signed_7zip.apk就可以,為了模擬運(yùn)行,我們將拆分apk復(fù)制到手機(jī)存儲,然后在程序的入口出載入patch,載入成功后程序會默認(rèn)退出,然后下次進(jìn)入看看bug是否被修護(hù).


對應(yīng)每個文件的說明

3.自定義application

目的:熱修復(fù)中為了能夠讓application更新

首先我們可以新建一個ApplicationLike類繼承DefaultApplicationLike,

@DefaultLifeCycle(

application ="com.xx.xxx.Application",//你自己的包名路徑

flags = ShareConstants.TINKER_ENABLE_ALL)

public class ApplicationLike extends DefaultApplicationLike

{

public ApplicationLike (Application application,inttinkerFlags,booleantinkerLoadVerifyFlag,longapplicationStartElapsedTime,longapplicationStartMillisTime, Intent tinkerResultIntent)

{

super(application, tinkerFlags, tinkerLoadVerifyFlag, applicationStartElapsedTime, applicationStartMillisTime, tinkerResultIntent);

}

@TargetApi(Build.VERSION_CODES.ICE_CREAM_SANDWICH)

public voidregisterActivityLifecycleCallbacks(Application.ActivityLifecycleCallbacks callback)

{

getApplication().registerActivityLifecycleCallbacks(callback);

}

@Override

public voidonBaseContextAttached(Context base)

{

MultiDex.install(base);

TinkerInstaller.install(this);

super.onBaseContextAttached(base);

}

然后把原有的application中的代碼復(fù)制到自定義的application中,這樣我們就可以在更新時候改動自定義的application了,原來的application


原有的application

放在com.tentcent.tinker.loader.app.TinkerApplication包下的是被保護(hù)不能被修改的.在添加完成后需要手動刪除自己原有的application類.


我們在來模擬測試下代碼:

/*

* Tencent is pleased to support the open source community by making Tinker available.

*

* Copyright (C) 2016 THL A29 Limited, a Tencent company. All rights reserved.

*

* Licensed under the BSD 3-Clause License (the "License"); you may not use this file except in

* compliance with the License. You may obtain a copy of the License at

*

* https://opensource.org/licenses/BSD-3-Clause

*

* Unless required by applicable law or agreed to in writing, software distributed under the License is

* distributed on an "AS IS" basis, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,

* either express or implied. See the License for the specific language governing permissions and

* limitations under the License.

*/

packagecom.zte.rs.test;

importandroid.app.AlertDialog;

importandroid.content.Context;

importandroid.graphics.Typeface;

importandroid.os.Bundle;

importandroid.os.Environment;

importandroid.support.v7.app.AppCompatActivity;

importandroid.util.Log;

importandroid.util.TypedValue;

importandroid.view.Gravity;

importandroid.view.View;

importandroid.view.ViewGroup;

importandroid.widget.Button;

importandroid.widget.TextView;

importcom.tencent.tinker.lib.library.TinkerLoadLibrary;

importcom.tencent.tinker.lib.tinker.Tinker;

importcom.tencent.tinker.lib.tinker.TinkerInstaller;

importcom.tencent.tinker.loader.shareutil.ShareConstants;

importcom.tencent.tinker.loader.shareutil.ShareTinkerInternals;

importcom.zte.rs.R;

importcom.zte.rs.util.ToastUtils;

public classTestActivityextendsAppCompatActivity

{

private static finalStringTAG="Tinker.TestActivity";

@Override

protected voidonCreate(Bundle savedInstanceState)

{

super.onCreate(savedInstanceState);

setContentView(R.layout.test_main);

Log.e(TAG,"i am on onCreate classloader:"+ TestActivity.class.getClassLoader().toString());

//test resource change

Log.e(TAG,"i am on onCreate string:"+"I am in the base apk");

Log.e(TAG,"i am on patch onCreate");

Button loadPatchButton = (Button) findViewById(R.id.loadPatch);

loadPatchButton.setOnClickListener(newView.OnClickListener()

{

@Override

public voidonClick(View v)

{

TinkerInstaller.onReceiveUpgradePatch(getApplicationContext(), Environment.getExternalStorageDirectory().getAbsolutePath() +"/patch_signed_7zip.apk");

}

});

Button error = (Button) findViewById(R.id.btn_error);

error.setOnClickListener(newView.OnClickListener()

{

@Override

public voidonClick(View v)

{

//? ? ? ? ? ? ? ? String a = null;

//? ? ? ? ? ? ? ? if(a.length() > 1)

//? ? ? ? ? ? ? ? {

//? ? ? ? ? ? ? ? ? ? ToastUtils.show(TestActivity.this, "永遠(yuǎn)不會彈出");

//? ? ? ? ? ? ? ? }

String a ="222";

if(a.length() >1)

{

ToastUtils.show(TestActivity.this,"bug修改了!這是第三次");

}

}

});

Button loadLibraryButton = (Button) findViewById(R.id.loadLibrary);

loadLibraryButton.setOnClickListener(newView.OnClickListener()

{

@Override

public voidonClick(View v)

{

// #method 1, hack classloader library path

TinkerLoadLibrary.installNavitveLibraryABI(getApplicationContext(),"armeabi");

System.loadLibrary("stlport_shared");

// #method 2, for lib/armeabi, just use TinkerInstaller.loadLibrary

//? ? ? ? ? ? ? ? TinkerLoadLibrary.loadArmLibrary(getApplicationContext(), "stlport_shared");

// #method 3, load tinker patch library directly

//? ? ? ? ? ? ? ? TinkerInstaller.loadLibraryFromTinker(getApplicationContext(), "assets/x86", "stlport_shared");

}

});

Button cleanPatchButton = (Button) findViewById(R.id.cleanPatch);

cleanPatchButton.setOnClickListener(newView.OnClickListener()

{

@Override

public voidonClick(View v)

{

Tinker.with(getApplicationContext()).cleanPatch();

}

});

Button killSelfButton = (Button) findViewById(R.id.killSelf);

killSelfButton.setOnClickListener(newView.OnClickListener()

{

@Override

public voidonClick(View v)

{

ShareTinkerInternals.killAllOtherProcess(getApplicationContext());

android.os.Process.killProcess(android.os.Process.myPid());

}

});

Button buildInfoButton = (Button) findViewById(R.id.showInfo);

buildInfoButton.setOnClickListener(newView.OnClickListener()

{

@Override

public voidonClick(View v)

{

showInfo(TestActivity.this);

}

});

}

public booleanshowInfo(Context context)

{

// add more Build Info

finalStringBuilder sb =newStringBuilder();

Tinker tinker = Tinker.with(getApplicationContext());

if(tinker.isTinkerLoaded())

{

sb.append(String.format("[patch is loaded]\n"));

sb.append(String.format("[TINKER_ID] %s\n", tinker.getTinkerLoadResultIfPresent().getPackageConfigByName(ShareConstants.TINKER_ID)));

sb.append(String.format("[packageConfig patchMessage] %s\n", tinker.getTinkerLoadResultIfPresent().getPackageConfigByName("patchMessage")));

sb.append(String.format("[TINKER_ID Rom Space] %d k\n", tinker.getTinkerRomSpace()));

}

else

{

sb.append(String.format("[patch is not loaded]\n"));

sb.append(String.format("[TINKER_ID] %s\n", ShareTinkerInternals.getManifestTinkerID(getApplicationContext())));

}

finalTextView v =newTextView(context);

v.setText(sb);

v.setGravity(Gravity.LEFT| Gravity.CENTER_VERTICAL);

v.setTextSize(TypedValue.COMPLEX_UNIT_DIP,10);

v.setLayoutParams(newViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT));

v.setTextColor(0xFF000000);

v.setTypeface(Typeface.MONOSPACE);

final intpadding =16;

v.setPadding(padding, padding, padding, padding);

finalAlertDialog.Builder builder =newAlertDialog.Builder(context);

builder.setCancelable(true);

builder.setView(v);

finalAlertDialog alert = builder.create();

alert.show();

return true;

}

@Override

protected voidonResume()

{

Log.e(TAG,"i am on onResume");

//? ? ? ? Log.e(TAG, "i am on patch onResume");

super.onResume();

Utils.setBackground(false);

}

@Override

protected voidonPause()

{

super.onPause();

Utils.setBackground(true);

}

}

加載patch:

TinkerInstaller.onReceiveUpgradePatch(getApplicationContext(), Environment.getExternalStorageDirectory().getAbsolutePath() +"/patch_signed_7zip.apk");

判斷是否加載完畢:

tinker.isTinkerLoaded()

運(yùn)行效果如下:

首先在程序中模擬一個異常:


異常

這里點(diǎn)擊異常會奔潰,然后我們修改代碼,點(diǎn)擊loadpatch,載入成功后再進(jìn)入程序,點(diǎn)擊異常,如圖:


修改后

到這里就全部結(jié)束了,這里需要注意幾點(diǎn):

基準(zhǔn)版本不改變

改變基準(zhǔn)版本時候改變tinkerid的值

自定義的application可以更新,但是manifest.xml是不能通過熱修復(fù)更新的

debug的差分apk大

release的查分apk小

運(yùn)行時候需要取消runtimeinstall選項(xiàng), settings-->Instant Run ---Enalbe instant Runto hot取消打勾


先build再tinker!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末务嫡,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子漆改,更是在濱河造成了極大的恐慌心铃,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,277評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件籽懦,死亡現(xiàn)場離奇詭異于个,居然都是意外死亡氛魁,警方通過查閱死者的電腦和手機(jī)暮顺,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評論 3 393
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來秀存,“玉大人捶码,你說我怎么就攤上這事』蛄矗” “怎么了惫恼?”我有些...
    開封第一講書人閱讀 163,624評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長澳盐。 經(jīng)常有香客問我祈纯,道長,這世上最難降的妖魔是什么叼耙? 我笑而不...
    開封第一講書人閱讀 58,356評論 1 293
  • 正文 為了忘掉前任腕窥,我火速辦了婚禮,結(jié)果婚禮上筛婉,老公的妹妹穿的比我還像新娘簇爆。我一直安慰自己,他們只是感情好爽撒,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評論 6 392
  • 文/花漫 我一把揭開白布入蛆。 她就那樣靜靜地躺著,像睡著了一般硕勿。 火紅的嫁衣襯著肌膚如雪哨毁。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,292評論 1 301
  • 那天源武,我揣著相機(jī)與錄音挑庶,去河邊找鬼言秸。 笑死,一個胖子當(dāng)著我的面吹牛迎捺,可吹牛的內(nèi)容都是我干的举畸。 我是一名探鬼主播,決...
    沈念sama閱讀 40,135評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼凳枝,長吁一口氣:“原來是場噩夢啊……” “哼抄沮!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起岖瑰,我...
    開封第一講書人閱讀 38,992評論 0 275
  • 序言:老撾萬榮一對情侶失蹤叛买,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后蹋订,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體率挣,經(jīng)...
    沈念sama閱讀 45,429評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評論 3 334
  • 正文 我和宋清朗相戀三年露戒,在試婚紗的時候發(fā)現(xiàn)自己被綠了椒功。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,785評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡智什,死狀恐怖动漾,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情荠锭,我是刑警寧澤旱眯,帶...
    沈念sama閱讀 35,492評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站证九,受9級特大地震影響删豺,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜愧怜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評論 3 328
  • 文/蒙蒙 一呀页、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧叫搁,春花似錦赔桌、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至惨奕,卻和暖如春雪位,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背梨撞。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評論 1 269
  • 我被黑心中介騙來泰國打工雹洗, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留香罐,地道東北人。 一個月前我還...
    沈念sama閱讀 47,891評論 2 370
  • 正文 我出身青樓时肿,卻偏偏與公主長得像庇茫,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子螃成,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評論 2 354

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