国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統 > Android > 正文

Android中如何安全地打印日志詳解

2019-10-22 18:21:32
字體:
來源:轉載
供稿:網友

前言

在Android開發過程中,不管是寫Demo還是實戰項目中,都會打印一些日志用于記錄數據,調試來著,Android中的日志工具類是Log,這個類提供了一些方法來android/81703.html">android/286262.html">打印日志。五個級別,v、d、i、w、e,各有不同的重載。

當談到如何打印日志?很多人會想這不是很簡單,直接使用android.util.Log這個類不就行了?然而,日志屬于非常敏感的信息;逆向工程師在逆向你的程序的時候,本來需要捕捉你程序的各種輸出,然后進行推測,順藤摸瓜然后得到需要的信息;一旦你的日志泄漏,無異于門戶洞開,破解你的程序如入無人之境。

安全的概念本來就是相對的,如果破解你程序的代價遠遠大于破解得到的價值,那么就可以認為程序是“安全的”;這里就分析一下,為了提高程序的安全性,在打印日志的時候應該注意什么。

首先看看絕大部分公司以及開發者的做法:

日志開關+日志類

為了在release版本里面沒有日志輸出,一個最簡單的想法是:把所有打印日志的語句放在一個if(DEBUG)的語句里面;在日常開發的時候,DEBUG開關打開,發布正式版本的時候關閉這個開關即可,大致思路如下:

// LogUtil.javapublic class LogUtil { private static boolean DEBUG = true;// 發布的時候修改為false  public static void d(String tag, String msg) {  if (DEBUG) android.util.Log.d(TAG, msg); } // 其他debug方法}

接下來看一個真實的例子,國外的一個apk,名字叫做powerclean;包名:com.lionmobi.powerclean;我們安裝這個包;發現很正常,沒有任何日志輸出;然后我們逆向這個apk;隨便翻看幾個類,發現很多地方有類似日志輸出:

android,打印日志,怎么打印日志,jni日志輸出圖片

我們打開這個叫做x的類,雖然被混淆過了,但是意思很明白,跟我們上面的思路一樣:

package com.lionmobi.util;import android.util.Log;public class x { private static boolean a; static {  x.a = false; } public static void d(String arg1, String arg2) {  if(x.a) {   Log.d(arg1, arg2);  } } public static void e(String arg1, String arg2) {  if(x.a) {   Log.e(arg1, arg2);  } } public static void i(String arg1, String arg2) {  if(x.a) {   Log.i(arg1, arg2);  } }}

這是一個真實的例子,而且這個app的用戶還不少;接下來我們看看這種方式有什么問題。

靜態反編譯打開日志開關

上面的那種方式有一個問題:雖然在release版本里面,確實沒有日志輸出;但是輸出日志的代碼依然存在,只是沒有執行到!(if條件不成立)所以,有沒有辦法讓這些代碼執行到呢?簡單來說,就是能不能在release版本里面把這個DEBUG變量弄成true呢?當然可以!而且做法還非常簡單。

我們使用apktool反編譯得到這個apk的smali代碼;然后上面的反編譯告訴我們,這個日志類的位置是:com.lionmobi.util.x我們打開這個x.smali文件,內容如下:

.class public Lcom/lionmobi/util/x;.super Ljava/lang/Object;# static fields.field private static a:Z# direct methods.method static constructor <clinit>()V .locals 1 const/4 v0, 0x0 # 修改為0x1 (True) sput-boolean v0, Lcom/lionmobi/util/x;->a:Z #初始化位置 return-void.end method.method public static d(Ljava/lang/String;Ljava/lang/String;)V .locals 1 sget-boolean v0, Lcom/lionmobi/util/x;->a:Z if-eqz v0, :cond_0 invoke-static {p0, p1}, Landroid/util/Log;->d(Ljava/lang/String;Ljava/lang/String;)I :cond_0 return-void.end method.method public static e(Ljava/lang/String;Ljava/lang/String;)V .locals 1 sget-boolean v0, Lcom/lionmobi/util/x;->a:Z if-eqz v0, :cond_0 invoke-static {p0, p1}, Landroid/util/Log;->e(Ljava/lang/String;Ljava/lang/String;)I :cond_0 return-void.end method.method public static i(Ljava/lang/String;Ljava/lang/String;)V .locals 1 sget-boolean v0, Lcom/lionmobi/util/x;->a:Z if-eqz v0, :cond_0 invoke-static {p0, p1}, Landroid/util/Log;->i(Ljava/lang/String;Ljava/lang/String;)I :cond_0 return-void.end method

很明白,那個叫做a的靜態變量就是我們的開關, 它的初始化在哪個靜態代碼塊里面;新建了一個局部變量0x0然后賦值給了a;因此,我們把這個0x0修改為0x1就打開了這個開關。很簡單吧,接下來我們把修改好的smali打包回去,然后簽名得到一個新的可以運行的apk;運行一下看看結果。果然,一大堆的日志輸出了出來,你的程序每一步在干什么都自己告訴別人了,都不需要去猜;我就隨便截個圖,感受下:

android,打印日志,怎么打印日志,jni
泄漏的日志信息

讓release版本里面不包含日志代碼

從上面的分析我們得到一個結論:如果需要程序是“日志安全的”,那么release版本里面不應該存在輸出日志的代碼。

如何做到這一點呢?我們可以做一個工具,開發的時候,正常打印日志;一旦需要發布版本,把所有打印日志的語句代碼,全部刪除掉。代碼很簡單,用一些正則表達式就可以做到。

事實上,我們也可以使用一些別的工具,來實現這個類似的功能;那就是proguard;提到這個工具,很多認只是覺得他是一個代碼混淆的工具,實際上,它還可以幫你剔除無用代碼!什么樣的代碼是無用代碼呢?

if (true) { // statement;}

類似于這樣,靜態編譯的時候被認為“永遠不會執行的代碼”,就被認為是無用代碼,會被這個工具直接優化掉,生成的class文件里面,這個if語句直接就沒有了。這個功能,完美符合我們的需求;我們只需要把輸出日志的代碼用這樣的if語句包圍起來,然后release的時候肯定會用這個工具混淆;然后,在release版本里面,所有的輸出日志的代碼全部都沒有了!不會像以前一樣,留下一個影子,只是不做事。

正確的做法

最終,我們所有打印日志的語句應該如下:

private static final boolean DEBUG = true; // 必須是static final 也就是常量,這樣才能在編譯器優化;刪除if塊if (DEBUG) { android.util.Log.d(TAG, "msg to print");}

然后,使用proguard優化代碼即可。

看起來簡單,好像也與最初的“日志開關”沒有什么區別,仔細分析一下:

日志開關必須是靜態常量

對比一下正確的做法與最開始的日志開關,一個是一個靜態變量,一個是靜態常量;如果是常量的話,那么就是永遠不變的,那么當DEBUG變量為False的時候proguard可以理所當然地認為,這一部分代碼時絕對不會被執行的,這樣,打印日志的語句就會被優化(刪除)掉;如果是一個變量,那么在運行期間就有可能改變它的值(private僅僅是對于程序員的改變,對于編譯器以及運行時,沒有什么改不了),這樣proguard就會置之不理,這樣你的日志代碼就暴露出來了,一字之差,失之千里。

拋棄日志類

假設我們使用了靜態常量代碼塊以及proguard優化代碼的技術;但是依然采用上面的日志類的技術,會發生什么呢?

public class LogUtil { private static final boolean DEBUG = false; public static void d(String tag, String msg) {  if (DEBUG) android.util.Log.d(tag, msg); }}

我寫了一個demo,自己打包然后反編譯,得到這個日志類如下(為了方便看,沒有混淆):

package com.example.test.app;public class LogUtil { private static final boolean DEBUG; public LogUtil() {  super(); } public static void d(String tag, String msg) { }}

我們看到,if代碼塊已經沒有了,確實不會輸出任何日志;但是,我們看看調用這個類的地方!

android,打印日志,怎么打印日志,jni
掩耳盜鈴的日志

這個LogUtil.d的調用,無異于掩耳盜鈴;雖然破解者沒辦法讓android.util.Log這個類輸出任何日志,但是你這里的這個調用還是告訴了別人你在干什么;所以,要屏蔽日志的輸出,必須使用if代碼塊直接包含要被剔除的日志。上面的那個日志類,要被優化掉,那就是:

if (DEBUG) { LogUtil.d(TAG, "msg");}

這里,不是多此一舉嗎,寫一個日志類就是想不想重復地寫if (DEBUG) ,這里為了使這一句隱藏,還是逃不掉;但是很抱歉,逃得了和尚逃不了廟,這種方法沒辦法做到完全隱藏信息;必須拋棄日志類包裹日志代碼的做法!

解放雙手的補充

也許有人說,為了這個所謂的日志安全,每次輸出日志都的寫一個if語句,那不麻煩死;簡直反人類,我懶!實際上,要少寫幾行代碼,我們可以選擇復用(代碼級別,比如上面的日志類),也可以選擇生成(直接生成代碼);在支持元編程的語言里面,生成代碼是很常見的事情,比如C++的模版元編程以及ruby吹噓的DSL能力;這里沒有那么高大上,用代碼生成代碼,我們直接借助編輯器幫助我們少寫幾行代碼萬事。

IDEA/Android Studio

可以使用live template的功能;比如我的做法是,寫一個ifd的template,每次我輸入ifd然后自動展開成if語句,光標停在最中間:

android,打印日志,怎么打印日志,jni使用live template簡化輸入

vim/emacs

可以使用宏錄制的功能,實現上面的live template。

總結

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


注:相關教程知識閱讀請移步到Android開發頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 陆丰市| 永春县| 舟山市| 平遥县| 高邮市| 莫力| 荔浦县| 三台县| 澎湖县| 余姚市| 宜春市| 景德镇市| 栾川县| 星子县| 双辽市| 吕梁市| 玉龙| 洛川县| 邓州市| 琼海市| 西乌珠穆沁旗| 保德县| 同江市| 揭东县| 新和县| 时尚| 通州市| 禹城市| 青冈县| 灵丘县| 方正县| 瑞金市| 宜宾县| 玉龙| 牡丹江市| 荔波县| 廊坊市| 忻州市| 福清市| 九江县| 苏尼特左旗|