Android OOM 分析

OOM簡(jiǎn)介

OOM全稱為Out of memory,解釋為內(nèi)存溢出敬锐。

  • 為了整個(gè)Android系統(tǒng)的內(nèi)存控制需要,Android系統(tǒng)為每一個(gè)應(yīng)用程序都設(shè)置了一個(gè)硬性的Dalvik Heap Size最大限制閾值,這個(gè)閾值在不同的設(shè)備上會(huì)因?yàn)镽AM大小不同而各有差異。如果你的應(yīng)用占用內(nèi)存空間已經(jīng)接近這個(gè)閾值始绍,此時(shí)再嘗試分配內(nèi)存的話,很容易引起OutOfMemoryError的錯(cuò)誤盏浇。
  • ActivityManager.getMemoryClass()可以用來查詢當(dāng)前應(yīng)用的Heap Size閾值(prop dalvik.vm.heapgrowthlimit 也可以)童擎,這個(gè)方法會(huì)返回一個(gè)整數(shù),表明你的應(yīng)用的Heap Size閾值是多少M(fèi)b(megabates)。
OOM產(chǎn)生原因

關(guān)于Native Heap末购,Dalvik Heap婴噩,Pss等內(nèi)存管理機(jī)制比較復(fù)雜,這里不展開描述站欺。簡(jiǎn)單的說贾虽,通過不同的內(nèi)存分配方式(malloc/mmap/JNIEnv/etc)對(duì)不同的對(duì)象(bitmap绰咽,etc)進(jìn)行操作會(huì)因?yàn)锳ndroid系統(tǒng)版本的差異而產(chǎn)生不同的行為,對(duì)Native Heap與Dalvik Heap以及OOM的判斷條件都會(huì)有所影響笑跛。在2.x的系統(tǒng)上,我們常称薨樱可以看到Heap Size的total值明顯超過了通過getMemoryClass()獲取到的閾值而不會(huì)發(fā)生OOM的情況圣拄,那么針對(duì)2.x與4.x的Android系統(tǒng)凭疮,到底是如何判斷會(huì)發(fā)生OOM呢寞肖?

  • Android 2.x系統(tǒng) GC LOG中的dalvik allocated + external allocated + 新分配的大小 >= getMemoryClass()值的時(shí)候就會(huì)發(fā)生OOM桶唐。 例如规脸,假設(shè)有這么一段Dalvik輸出的GC LOG:GC_FOR_MALLOC free 2K, 13% free 32586K/37455K, external 8989K/10356K, paused 20ms横殴,那么32586+8989+(新分配23975)=65550>64M時(shí)梨与,就會(huì)發(fā)生OOM瞄崇。
  • Android 4.x系統(tǒng) Android 4.x的系統(tǒng)廢除了external的計(jì)數(shù)器,類似bitmap的分配改到dalvik的java heap中申請(qǐng)- rt運(yùn)行環(huán)境筹燕,但是統(tǒng)計(jì)規(guī)則還是和dalvik保持一致)


    20180426160641182.png
如何避免OOM

前面介紹了OOM 的基礎(chǔ)知識(shí)糠涛,那么在實(shí)踐中有什么方法來減少OOM的出現(xiàn)呢砸脊?總結(jié)下來大概分下面幾個(gè)方面:

  • 減小對(duì)象的內(nèi)存占用
  • 內(nèi)存對(duì)象的重復(fù)使用
  • 避免對(duì)象的內(nèi)存泄漏
  • 內(nèi)存使用策略優(yōu)化
減小對(duì)象的內(nèi)存占用

1诗芜、使用更輕量級(jí)的數(shù)據(jù)結(jié)構(gòu)
例如孩哑,我們可以考慮使用ArrayMap/SparseArray而不是HashMap等傳統(tǒng)數(shù)據(jù)結(jié)構(gòu),下圖演示了HashMap的簡(jiǎn)要工作原理仅炊,相比起Android系統(tǒng)專門為移動(dòng)操作系統(tǒng)編寫的ArrayMap容器瓶竭,在大多數(shù)情況下次询,都顯示效率低下盒卸,更占內(nèi)存摘投。通常的HashMap的實(shí)現(xiàn)方式更加消耗內(nèi)存薇组,因?yàn)樗枰粋€(gè)額外的實(shí)例對(duì)象來記錄Mapping操作挑童。另外菇民,SparseArray更加高效在于他們避免了對(duì)key與value的autobox自動(dòng)裝箱,并且避免了裝箱后的解箱。

下圖是HashMap 的工作原理:

20180426165316367.png

下面是ArrayMap的delete 原理:
2018042616541494.png

2塞蹭、避免在Android 中使用enum
Android 官方的Training 中有這樣一句話>“Enums often require more than twice as much memory as static constants. You should strictly avoid using enums on Android.”
關(guān)于enum的效率漱办,請(qǐng)看下面的討論。假設(shè)我們有這樣一份代碼率碾,編譯之后的dex大小是2556 bytes仔粥,在此基礎(chǔ)之上麦向,添加一些如下代碼话告,這些代碼使用普通static常量相關(guān)作為判斷值:
20180426170014707.png

增加上面那段代碼之后,編譯成dex的大小是2680 bytes送挑,相比起之前的2556 bytes只增加124 bytes罢荡。假如換做使用enum,情況如下:
20180426170105515.png

使用enum之后的dex大小是4188 bytes,相比起2556增加了1632 bytes摔踱,增長(zhǎng)量是使用static int的13倍腐芍。不僅僅如此颠蕴,使用enum,運(yùn)行時(shí)還會(huì)產(chǎn)生額外的內(nèi)存占用襟沮,如下圖所示:
20180426200926112.png

Android官方強(qiáng)烈建議不要在Android程序里面使用到enum。
3祠汇、減小Bitmap對(duì)象的內(nèi)存占用
Bitmap是一個(gè)極容易消耗內(nèi)存的大胖子,減小創(chuàng)建出來的Bitmap的內(nèi)存占用是很重要的,通常來說有下面2個(gè)措施:
inSampleSize:縮放比例,在把圖片載入內(nèi)存之前,我們需要先計(jì)算出一個(gè)合適的縮放比例,避免不必要的大圖載入菠秒。
decode format:解碼格式禁灼,選擇ARGB_8888/RBG_565/ARGB_4444/ALPHA_8守谓,存在很大差異平酿。

4叨恨、使用更小的圖片
在設(shè)計(jì)給到資源圖片的時(shí)候送矩,我們需要特別留意這張圖片是否存在可以壓縮的空間,是否可以使用一張更小的圖片晌块。盡量使用更小的圖片不僅僅可以減少內(nèi)存的使用匆背,還可以避免出現(xiàn)大量的InflationException。假設(shè)有一張很大的圖片被XML文件直接引用括享,很有可能在初始化視圖的時(shí)候就會(huì)因?yàn)閮?nèi)存不足而發(fā)生InflationException铃辖,這個(gè)問題的根本原因其實(shí)是發(fā)生了OOM澳叉。

內(nèi)存對(duì)象的重復(fù)使用

大多數(shù)對(duì)象的復(fù)用沐悦,最終實(shí)施的方案都是利用對(duì)象池技術(shù),要么是在編寫代碼的時(shí)候顯式的在程序里面去創(chuàng)建對(duì)象池,然后處理好復(fù)用的實(shí)現(xiàn)邏輯碴裙,要么就是利用系統(tǒng)框架既有的某些復(fù)用特性達(dá)到減少對(duì)象的重復(fù)創(chuàng)建,從而減少內(nèi)存的分配與回收。

1辞做、復(fù)用系統(tǒng)自帶的資源
Android系統(tǒng)本身內(nèi)置了很多的資源晒杈,例如字符串/顏色/圖片/動(dòng)畫/樣式以及簡(jiǎn)單布局等等粪般,這些資源都可以在應(yīng)用程序中直接引用。這樣做不僅僅可以減少應(yīng)用程序的自身負(fù)重顾稀,減小APK的大小辐宾,另外還可以一定程度上減少內(nèi)存的開銷豺鼻,復(fù)用性更好儒飒。但是也有必要留意Android系統(tǒng)的版本差異性,對(duì)那些不同系統(tǒng)版本上表現(xiàn)存在很大差異桩了,不符合需求的情況蕉扮,還是需要應(yīng)用程序自身內(nèi)置進(jìn)去颗圣。

2在岂、注意在ListView/GridView等出現(xiàn)大量重復(fù)子組件的視圖里面對(duì)ConvertView的復(fù)用

3蔽午、Bitmap對(duì)象的復(fù)用

4、避免在onDraw方法里面執(zhí)行對(duì)象的創(chuàng)建

類似onDraw等頻繁調(diào)用的方法疾呻,一定需要注意避免在這里做創(chuàng)建對(duì)象的操作岸蜗,因?yàn)樗麜?huì)迅速增加內(nèi)存的使用叠蝇,而且很容易引起頻繁的gc铃慷,甚至是內(nèi)存抖動(dòng)。

5蜕该、StringBuilder
在有些時(shí)候犁柜,代碼中會(huì)需要使用到大量的字符串拼接的操作,這種時(shí)候有必要考慮使用StringBuilder來替代頻繁的“+”堂淡。

避免對(duì)象的內(nèi)存泄漏
內(nèi)存對(duì)象的泄漏馋缅,會(huì)導(dǎo)致一些不再使用的對(duì)象無法及時(shí)釋放扒腕,這樣一方面占用了寶貴的內(nèi)存空間,很容易導(dǎo)致后續(xù)需要分配內(nèi)存的時(shí)候萤悴,空閑空間不足而出現(xiàn)OOM瘾腰。顯然,這還使得每級(jí)Generation的內(nèi)存區(qū)域可用空間變小覆履,gc就會(huì)更容易被觸發(fā)栖雾,容易出現(xiàn)內(nèi)存抖動(dòng),從而引起性能問題噪径。

最新的LeakCanary開源控件,可以很好的幫助我們發(fā)現(xiàn)內(nèi)存泄露的情況车摄,更多關(guān)于LeakCanary的介紹眼俊,請(qǐng)看這里https://github.com/square/leakcanary(中文使用說明http://www.liaohuqiu.net/cn/posts/leak-canary-read-me/)环戈。另外也可以使用傳統(tǒng)的MAT工具查找內(nèi)存泄露,請(qǐng)參考這里http://android-developers.blogspot.pt/2011/03/memory-analysis-for-android.html(便捷的中文資料http://androidperformance.com/2015/04/11/AndroidMemory-Usage-Of-MAT/

1、注意Activity 的泄漏

通常來說创泄,Activity的泄漏是內(nèi)存泄漏里面最嚴(yán)重的問題,它占用的內(nèi)存多搁拙,影響面廣朋譬,我們需要特別注意以下兩種情況導(dǎo)致的Activity泄漏:

  • 內(nèi)部類引用導(dǎo)致Activity的泄漏

最典型的場(chǎng)景是Handler導(dǎo)致的Activity泄漏,如果Handler中有延遲的任務(wù)或者是等待執(zhí)行的任務(wù)隊(duì)列過長(zhǎng)窑业,都有可能因?yàn)镠andler繼續(xù)執(zhí)行而導(dǎo)致Activity發(fā)生泄漏。此時(shí)的引用關(guān)系鏈?zhǔn)荓ooper -> MessageQueue -> Message -> Handler -> Activity。為了解決這個(gè)問題喷市,可以在UI退出之前,執(zhí)行remove Handler消息隊(duì)列中的消息與runnable對(duì)象缭黔。或者是使用Static + WeakReference的方式來達(dá)到斷開Handler與Activity之間存在引用關(guān)系的目的哎媚。

  • Activity Context被傳遞到其他實(shí)例中,這可能導(dǎo)致自身被引用而發(fā)生泄漏。
    內(nèi)部類引起的泄漏不僅僅會(huì)發(fā)生在Activity上,其他任何內(nèi)部類出現(xiàn)的地方低淡,都需要特別留意滥壕!我們可以考慮盡量使用static類型的內(nèi)部類胁孙,同時(shí)使用WeakReference的機(jī)制來避免因?yàn)榛ハ嘁枚霈F(xiàn)的泄露冈止。

2、考慮使用Application Context而不是Activity Contex

對(duì)于大部分非必須使用Activity Context的情況(Dialog的Context就必須是Activity Context),我們都可以考慮使用Application Context而不是Activity的Context国瓮,這樣可以避免不經(jīng)意的Activity泄露。

3播歼、Bitmap 對(duì)象的及時(shí)回收
雖然在大多數(shù)情況下,我們會(huì)對(duì)Bitmap增加緩存機(jī)制雾狈,但是在某些時(shí)候,部分Bitmap是需要及時(shí)回收的。例如臨時(shí)創(chuàng)建的某個(gè)相對(duì)比較大的bitmap對(duì)象咒循,在經(jīng)過變換得到新的bitmap對(duì)象之后,應(yīng)該盡快回收原始的bitmap,這樣能夠更快釋放原始bitmap所占用的空間。

需要特別留意的是Bitmap類里面提供的createBitmap()方法:


2018042620264266.png

這個(gè)函數(shù)返回的bitmap有可能和source bitmap是同一個(gè)筷弦,在回收的時(shí)候奸绷,需要特別檢查source bitmap與return bitmap的引用是否相同畔派,只有在不等的情況下,才能夠執(zhí)行source bitmap的recycle方法。

4、注意監(jiān)聽器的注銷
在Android程序里面存在很多需要register與unregister的監(jiān)聽器括细,我們需要確保在合適的時(shí)候及時(shí)unregister那些監(jiān)聽器。自己手動(dòng)add的listener,需要記得及時(shí)remove這個(gè)listener。

5尤筐、注意緩存容器中的對(duì)象泄漏
有時(shí)候,我們?yōu)榱颂岣邔?duì)象的復(fù)用性把某些對(duì)象放到緩存容器中,可是如果這些對(duì)象沒有及時(shí)從容器中清除,也是有可能導(dǎo)致內(nèi)存泄漏的。例如井仰,針對(duì)2.3的系統(tǒng),如果把drawable添加到緩存容器,因?yàn)閐rawable與View的強(qiáng)應(yīng)用,很容易導(dǎo)致activity發(fā)生泄漏蛾茉。而從4.0開始,就不存在這個(gè)問題。解決這個(gè)問題,需要對(duì)2.3系統(tǒng)上的緩存drawable做特殊封裝碱茁,處理引用解綁的問題蜓氨,避免泄漏的情況港令。

6淋淀、注意WebView的泄漏
Android中的WebView存在很大的兼容性問題徽缚,不僅僅是Android系統(tǒng)版本的不同對(duì)WebView產(chǎn)生很大的差異那婉,另外不同的廠商出貨的ROM里面WebView也存在著很大的差異呛谜。更嚴(yán)重的是標(biāo)準(zhǔn)的WebView存在內(nèi)存泄露的問題割坠,看這里WebView causes memory leak - leaks the parent Activity沪羔。所以通常根治這個(gè)問題的辦法是為WebView開啟另外一個(gè)進(jìn)程茫因,通過AIDL與主進(jìn)程進(jìn)行通信洛巢,WebView所在的進(jìn)程可以根據(jù)業(yè)務(wù)的需要選擇合適的時(shí)機(jī)進(jìn)行銷毀恃慧,從而達(dá)到內(nèi)存的完整釋放怠蹂。

7训挡、注意Cursor對(duì)象是否及時(shí)關(guān)閉
在程序中我們經(jīng)常會(huì)進(jìn)行查詢數(shù)據(jù)庫(kù)的操作,但時(shí)常會(huì)存在不小心使用Cursor之后沒有及時(shí)關(guān)閉的情況。這些Cursor的泄露,反復(fù)多次出現(xiàn)的話會(huì)對(duì)內(nèi)存管理產(chǎn)生很大的負(fù)面影響祖很,我們需要謹(jǐn)記對(duì)Cursor對(duì)象的及時(shí)關(guān)閉蠢琳。

內(nèi)存使用策略優(yōu)化

1例衍、Try catch 某些大內(nèi)存的操作
在某些情況下梦抢,我們需要事先評(píng)估那些可能發(fā)生OOM的代碼,對(duì)于這些可能發(fā)生OOM的代碼甘改,加入catch機(jī)制,可以考慮在catch里面嘗試一次降級(jí)的內(nèi)存分配操作。例如decode bitmap的時(shí)候鞋既,catch到OOM,可以嘗試把采樣比例再增加一倍之后灾炭,再次嘗試decode偷厦。

2、謹(jǐn)慎使用static 對(duì)象
static是Java中的一個(gè)關(guān)鍵字榴啸,當(dāng)用它來修飾成員變量時(shí)狂鞋,那么該變量就屬于該類嘲叔,而不是該類的實(shí)例卵牍。 不少程序員喜歡用static這個(gè)關(guān)鍵字修飾變量萝衩,因?yàn)樗沟米兞康纳芷诖蟠笱娱L(zhǎng)啦,并且訪問的時(shí)候撤防,也極其的方便,用類名就能直接訪問充岛,各個(gè)資源間 傳值也極其的方便扔亥,所以秕脓,它經(jīng)常被我們使用。但如果用它來引用一些資源耗費(fèi)過多的實(shí)例(Context的情況最多)薛训,這時(shí)就要謹(jǐn)慎對(duì)待了。

 public class ClassName {  
      private static Context mContext;  
      //省略  
}   

以上的代碼是很危險(xiǎn)的巍耗,如果將Activity賦值到么mContext的話。那么即使該Activity已經(jīng)onDestroy女气,但是由于仍有對(duì)象保存它的引用,因此該Activity依然不會(huì)被釋放观游,并且,如果該activity里面再持有一些資源况木,那就糟糕了宴倍。

上面是直接的引用泄露,我們?cè)倏磄oogle文檔中的一個(gè)例子:

 private static Drawable sBackground;   
 
  @Override  
  protected void onCreate(Bundle state) {  
    super.onCreate(state);   
 
    TextView label = new TextView(this);  
    label.setText("Leaks are bad");   
 
    if (sBackground == null) {  
      sBackground = getDrawable(R.drawable.large_bitmap);  
    }  
    label.setBackgroundDrawable(sBackground);   
 
    setContentView(label);  
  }   

sBackground, 是一個(gè)靜態(tài)的變量,但是我們發(fā)現(xiàn)瓦灶,我們并沒有顯式的保存Contex的引用眨层,但是,當(dāng)Drawable與View連接之后卧秘,Drawable就將View 設(shè)置為一個(gè)回調(diào)遭顶,由于View中是包含Context的引用的婿滓,所以嗡官,實(shí)際上我們依然保存了Context的引用。這個(gè)引用鏈如下:
Drawable->TextView->Context
所以询兴,最終該Context也沒有得到釋放,也發(fā)生了內(nèi)存泄露。
那我們?nèi)绾蔚谋苊膺@種泄露的發(fā)生呢厌丑?

  • 應(yīng)該盡量避免static成員變量引用資源耗費(fèi)過多的實(shí)例朦肘,比如Context媒抠。
  • Context盡量使用Application Context,因?yàn)锳pplication的Context的生命周期比較長(zhǎng),引用它不會(huì)出現(xiàn)內(nèi)存泄露的問題。
  • 使用WeakReference代替強(qiáng)引用苟耻。比如可以使用WeakReference<Context> mContextRef;
    該部分的詳細(xì)內(nèi)容也可以參考Android文檔中Article部分。

3毛秘、特別留意單例模式的不合理持有

4饭寺、珍惜service 資源
如果你的應(yīng)用需要在后臺(tái)使用service,除非它被觸發(fā)并執(zhí)行一個(gè)任務(wù)叫挟,否則其他時(shí)候Service都應(yīng)該是停止?fàn)顟B(tài)艰匙。另外需要注意當(dāng)這個(gè)service完成任務(wù)之后因?yàn)橥V箂ervice失敗而引起的內(nèi)存泄漏。 當(dāng)你啟動(dòng)一個(gè)Service抹恳,系統(tǒng)會(huì)傾向?yàn)榱吮A暨@個(gè)Service而一直保留Service所在的進(jìn)程员凝。這使得進(jìn)程的運(yùn)行代價(jià)很高,因?yàn)橄到y(tǒng)沒有辦法把Service所占用的RAM空間騰出來讓給其他組件奋献,另外Service還不能被Paged out健霹。這減少了系統(tǒng)能夠存放到LRU緩存當(dāng)中的進(jìn)程數(shù)量旺上,它會(huì)影響應(yīng)用之間的切換效率,甚至?xí)?dǎo)致系統(tǒng)內(nèi)存使用不穩(wěn)定糖埋,從而無法繼續(xù)保持住所有目前正在運(yùn)行的service宣吱。 建議使用IntentService,它會(huì)在處理完交代給它的任務(wù)之后盡快結(jié)束自己瞳别。更多信息征候,請(qǐng)閱讀Running in a Background Service。

5祟敛、優(yōu)化布局層次疤坝,減少內(nèi)存消耗
越扁平化的視圖布局,占用的內(nèi)存就越少馆铁,效率越高跑揉。我們需要盡量保證布局足夠扁平化,當(dāng)使用系統(tǒng)提供的View無法實(shí)現(xiàn)足夠扁平的時(shí)候考慮使用自定義View來達(dá)到目的埠巨。

6畔裕、謹(jǐn)慎使用“抽象”編程
很多時(shí)候,開發(fā)者會(huì)使用抽象類作為”好的編程實(shí)踐”,因?yàn)槌橄竽軌蛱嵘a的靈活性與可維護(hù)性。然而枝嘶,抽象會(huì)導(dǎo)致一個(gè)顯著的額外內(nèi)存開銷:他們需要同等量的代碼用于可執(zhí)行周崭,那些代碼會(huì)被mapping到內(nèi)存中,因此如果你的抽象沒有顯著的提升效率扛点,應(yīng)該盡量避免他們哥遮。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市陵究,隨后出現(xiàn)的幾起案子眠饮,更是在濱河造成了極大的恐慌,老刑警劉巖铜邮,帶你破解...
    沈念sama閱讀 218,386評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件仪召,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡松蒜,警方通過查閱死者的電腦和手機(jī)扔茅,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,142評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來秸苗,“玉大人召娜,你說我怎么就攤上這事【ィ” “怎么了玖瘸?”我有些...
    開封第一講書人閱讀 164,704評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵秸讹,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我雅倒,道長(zhǎng)璃诀,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,702評(píng)論 1 294
  • 正文 為了忘掉前任屯断,我火速辦了婚禮文虏,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘殖演。我一直安慰自己氧秘,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,716評(píng)論 6 392
  • 文/花漫 我一把揭開白布趴久。 她就那樣靜靜地躺著丸相,像睡著了一般。 火紅的嫁衣襯著肌膚如雪彼棍。 梳的紋絲不亂的頭發(fā)上灭忠,一...
    開封第一講書人閱讀 51,573評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音座硕,去河邊找鬼弛作。 笑死,一個(gè)胖子當(dāng)著我的面吹牛华匾,可吹牛的內(nèi)容都是我干的映琳。 我是一名探鬼主播,決...
    沈念sama閱讀 40,314評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼蜘拉,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼萨西!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起旭旭,我...
    開封第一講書人閱讀 39,230評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤谎脯,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后持寄,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體源梭,經(jīng)...
    沈念sama閱讀 45,680評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,873評(píng)論 3 336
  • 正文 我和宋清朗相戀三年稍味,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了咸产。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,991評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡仲闽,死狀恐怖脑溢,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤屑彻,帶...
    沈念sama閱讀 35,706評(píng)論 5 346
  • 正文 年R本政府宣布验庙,位于F島的核電站,受9級(jí)特大地震影響社牲,放射性物質(zhì)發(fā)生泄漏粪薛。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,329評(píng)論 3 330
  • 文/蒙蒙 一搏恤、第九天 我趴在偏房一處隱蔽的房頂上張望违寿。 院中可真熱鬧,春花似錦熟空、人聲如沸藤巢。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,910評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)掂咒。三九已至,卻和暖如春迈喉,著一層夾襖步出監(jiān)牢的瞬間绍刮,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,038評(píng)論 1 270
  • 我被黑心中介騙來泰國(guó)打工挨摸, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留孩革,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,158評(píng)論 3 370
  • 正文 我出身青樓得运,卻偏偏與公主長(zhǎng)得像膝蜈,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子澈圈,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,941評(píng)論 2 355

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