亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > Android > 正文

Android高性能日志寫入方案的實現

2019-10-21 21:30:26
字體:
來源:轉載
供稿:網友

前言

公司目前在做一款企業級智能客服系統,對于系統穩定性要求很高,不過難保用戶在使用中不會出現問題,而 Android SDK 集成在客戶的 APP 中,同時由于 Android 碎片化的問題,對于 SDK 的問題排查就顯得尤為困難,因此記錄下用戶的操作日志就顯得極為重要。

初始方案

一開始,SDK 記錄日志的方式是直接通過寫文件,當有一條日志要寫入的時候,首先,打開文件,然后寫入日志,最后關閉文件。這樣做的問題就在于頻繁的IO操作,影響程序的性能,而且 SDK 為了保證消息的及時性,還維護了一個后臺進程,當其中一個進程進行日志寫入時,另一個就會被鎖在門外等著,問題就愈發嚴重。使用這種方案雖然當前看上去對程序的影響不大,但是隨著日志量的增加,更多的IO操作,一定會造成性能瓶頸。

下面我們來分析下直接寫入文件的流程:

  • 用戶發起 write 操作
  • 操作系統查找頁緩存
    a.若未命中,則產生缺頁異常,然后創建頁緩存,將用戶傳入的內容寫入頁緩存
    b.若命中,則直接將用戶傳入的內容寫入頁緩存
  • 用戶 write 調用完成
  • 頁被修改后成為臟頁,操作系統有兩種機制將臟頁寫回磁盤
    a.用戶手動調用 fsync()
    b.由 pdflush 進程定時將臟頁寫回磁盤

可以看出,數據從程序寫入到磁盤的過程中,其實牽涉到兩次數據拷貝:一次是用戶空間內存拷貝到內核空間的緩存,一次是回寫時內核空間的緩存到硬盤的拷貝。當發生回寫時也涉及到了內核空間和用戶空間頻繁切換。

而且相對于機械硬盤,SSD 存儲還有一個“寫入放大”的問題。這個問題主要和 SSD 存儲的物理結構有關。當 SSD 被全部寫過一遍之后,再寫入的數據是不可以直接更新,只可以通過覆蓋重寫,在覆蓋之前需要先擦除數據。但寫入的最小單位是 Page,擦除的最小單位是 Block,而 Block 遠大于 Page,所以在寫入新數據時就需要先把 Block 上的數據讀出來和要寫入的數據合并在一起,再把 Block 擦除,最后把讀出來的數據重新寫入到存儲上,這樣導致實際寫入的數據可能遠遠大于最開始需要寫入的數據。

沒想到簡單的寫文件竟然涉及了這么多操作,只是對于應用層透明而已。

既然每寫一次文件會執行這么多次操作,那么我們能不能將日志緩存起來,當達到一定的數量后再一次性的寫入磁盤中呢?
這樣確實能夠大量減少 IO 次數,但是卻會引發另一個更嚴重的問題——丟日志

把日志緩存在內存中,當程序發生 Crash 或進程被殺后就無法保證日志的完整性,而且由于 SDK 存在多進程,也無法保證多進程下日志的順序。

一個完善的日志方案,需要滿足

  • 高效,不能影響系統性能,不能因為引入了日志模塊而造成應用卡頓
  • 保證日志的完整性,如果不能保證日志完整,那么日志收集就沒有意義了
  • 對于多進程應用,要保證最終看到的日志順序的準確性

高性能方案

既然無法減少寫入次數,那么我們能不能在寫文件的過程中去優化呢?

答案是可以的,使用 mmap

mmap是一種內存映射文件的方法,即將一個文件或者其它對象映射到進程的地址空間,實現文件磁盤地址和進程虛擬地址空間中一段虛擬地址的一一對映關系,函數原型如下

void *mmap(void *addr, size_t length, int prot, int flags, int fd, off_t offset);

 

mAndroid,高性能,日志寫入map操作提供了一種機制,讓用戶程序直接訪問設備內存,這種機制,相比較在用戶空間和內核空間互相拷貝數據,效率更高。在要求高性能的應用中比較常用。

時 mmap 能夠保證日志的完整性,mmap 的回寫時機:

  • 內存不足
  • 進程退出
  • 調用 msync 或者 munmap
  • 不設置 MAP_NOSYNC 情況下 30s-60s(僅限FreeBSD)

當映射一個文件后,程序就會在 native 內存中申請一塊相同大小的空間,因此建議每次映射一小段內容,如 64k,寫滿后再重新映射文件后面的內容。

日志寫入性能和完整性的問題解決了,那么如何保證多進程下日志的順序呢?

由于 mmap 是采用共享內存的方式寫入數據,如果兩個進程同時映射一個文件,那么一定會造成日志覆蓋的問題。

既然不能直接保證順序,那我們只能退而求其次,兩個進程分別映射不同的文件,每天合并一次,合并時對日志進行排序。

繼續優化

根據上述方案,設計 jni 接口,打包 so,引入 SDK,看似沒什么問題了,但是作為一款 SDK,總覺得包含 so 不太友好,在一定程度上會增加接入的難度。

那么能不能不用 so 呢?

其實 Java 中已經提供了內存映射的實現——MappedByteBuffer

MappedByteBuffer 位于 Java NIO 包下,用于將文件內容映射到緩沖區,使用的即是 mmap 技術。通過 FileChannel 的 map 方法可以創建緩沖區

RandomAccessFileraf = new RandomAccessFile(file, "rw");MappedByteBuffer buffer = raf.getChannel().map(FileChannel.MapMode.READ_WRITE, position, size);

為了測試 MappedByteBuffer 的效率,我們把 64byte 的數據分別寫入內存、MappedByteBuffer 和磁盤文件 50 萬次,并統計耗時

 

方法 耗時
內存 384ms
MappedByteBuffer 700ms
磁盤文件 16805ms

 

可以看出 MappedByteBuffer 雖然不及寫入內存的性能,但是相比較寫入磁盤文件,已經有了質的提升。

總結

本文主要分析了直接寫文件記錄日志方式存在的問題,并引申出高性能文件寫入方案 mmap,兼顧了寫入性能和完整性,并通過補償方案確保多進程下日志的順序。最后發現了內存映射在 Java 層的實現,避免了引入 so。

好了,以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網的支持。


注:相關教程知識閱讀請移步到Android開發頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
色中色综合影院手机版在线观看| 亚洲性生活视频在线观看| 亚洲欧美制服第一页| 91在线观看免费高清完整版在线观看| 亚洲人精品午夜在线观看| 国产999精品久久久影片官网| 欧美午夜美女看片| 日韩精品视频在线观看网址| 国产精品丝袜高跟| 亚洲一区免费网站| 色久欧美在线视频观看| 国产91对白在线播放| 久久躁狠狠躁夜夜爽| 日韩欧美极品在线观看| 欧美亚洲视频在线看网址| 亚洲免费视频观看| 成人精品一区二区三区电影免费| 亚洲精品v天堂中文字幕| 国内免费精品永久在线视频| 精品中文字幕视频| 成人午夜在线影院| 精品高清美女精品国产区| 国产精品第三页| 日韩av一区二区在线| 日韩在线视频一区| 狠狠做深爱婷婷久久综合一区| 日韩电影大片中文字幕| 91av视频在线观看| 亚洲综合成人婷婷小说| 欧美激情一区二区三区高清视频| 国产在线精品一区免费香蕉| 欧美激情精品久久久| 色偷偷88888欧美精品久久久| 色偷偷av一区二区三区乱| 亚洲一区二区三区视频播放| 国模视频一区二区三区| 中文字幕综合一区| 日韩hd视频在线观看| 欧美日韩国产精品一区| 国产日韩精品综合网站| 欧洲成人免费aa| 这里只有精品视频在线| 中日韩美女免费视频网址在线观看| 欧美日韩在线视频一区| 国产z一区二区三区| 日韩欧美在线视频免费观看| 国产成人精品在线播放| 国内精品国产三级国产在线专| 欧美激情图片区| 亚洲色图色老头| 日韩视频亚洲视频| 亚洲国产精品免费| 亚洲天堂男人天堂女人天堂| zzijzzij亚洲日本成熟少妇| 亚洲最新av在线| 国产精品久久久| 欧美一级视频在线观看| 国产精品亚发布| 成人欧美在线视频| 亚洲第一综合天堂另类专| 一区二区亚洲欧洲国产日韩| 综合国产在线观看| 91精品国产91久久久久久不卡| 亚洲一级免费视频| 2020久久国产精品| 国产女精品视频网站免费| 亚洲免费电影一区| 精品久久中文字幕| 亚洲精品乱码久久久久久按摩观| 国内久久久精品| 18一19gay欧美视频网站| 日韩电影中文 亚洲精品乱码| 久久久这里只有精品视频| 国产91在线高潮白浆在线观看| 久久久久久久国产精品| 懂色aⅴ精品一区二区三区蜜月| 欧美精品在线视频观看| 欧美日韩电影在线观看| 欧洲亚洲在线视频| 日韩欧美一区二区三区久久| 国产精品久久久久7777婷婷| 欧美一级片免费在线| 欧美在线中文字幕| 91在线免费看网站| 久久影视电视剧凤归四时歌| 亚洲精品久久久一区二区三区| 久久久久久久久91| 日韩不卡中文字幕| 久久久久久香蕉网| 日本午夜精品理论片a级appf发布| 国产丝袜一区视频在线观看| 久久久国产影院| 亚洲免费一级电影| 欧美激情视频网站| 久久色免费在线视频| 亚洲视频在线免费看| www.亚洲天堂| 91精品在线观| 国产福利成人在线| 日韩av片永久免费网站| 亚洲精品国精品久久99热| 欧美一级高清免费播放| 成人做爰www免费看视频网站| 色综合色综合久久综合频道88| 亚洲男人的天堂在线| 亚洲精品99久久久久中文字幕| 精品美女永久免费视频| 中文字幕一区二区三区电影| 欧美激情亚洲精品| 97精品国产91久久久久久| 日韩高清人体午夜| 日韩成人在线电影网| 日本欧美在线视频| 久久久伊人日本| 亚洲乱码国产乱码精品精天堂| 中文字幕精品www乱入免费视频| 精品一区二区三区三区| 国产成人精品日本亚洲专区61| 精品美女永久免费视频| 成人av电影天堂| 亚洲精品国产福利| 亚洲国产一区自拍| 亚洲美女av电影| 亚洲а∨天堂久久精品9966| 国产精品激情av在线播放| 一区二区国产精品视频| 欧洲精品久久久| 欧美日韩国产中字| 成人欧美在线观看| 日本一区二区不卡| 成人性生交大片免费观看嘿嘿视频| 最新的欧美黄色| 亚洲色图狂野欧美| 亚洲jizzjizz日本少妇| 欧美精品成人91久久久久久久| 国产精品色午夜在线观看| 国产精品第8页| 国产精品美女呻吟| 亚洲女人天堂成人av在线| 最近的2019中文字幕免费一页| 欧美xxxx做受欧美.88| 欧美大片在线免费观看| 国产精品成人av性教育| 亚洲第一区第二区| 欧美大尺度在线观看| 欧美视频免费在线观看| 久久综合伊人77777尤物| 国产精品视频最多的网站| 成人免费视频a| 狠狠色狠狠色综合日日小说| 欧美精品久久久久久久| 国产精品美乳在线观看| 中文字幕国产精品| 久久久久久久久中文字幕| 狠狠干狠狠久久| 最新国产成人av网站网址麻豆| 亚洲精品久久久久| 精品久久久久久久久中文字幕| 亚洲国产另类 国产精品国产免费| 一本色道久久88亚洲综合88| 最近中文字幕2019免费| 97国产精品人人爽人人做| 色偷偷av一区二区三区乱| 美女视频久久黄|