最近有客戶(hù)提到一個(gè)需求,如下:
他要導(dǎo)出whatsapp程序下面的xml數(shù)據(jù),并再手動(dòng)輸入用戶(hù)手機(jī)號(hào)以及國(guó)家區(qū)號(hào)之后生成一個(gè)json文件,我們都知道/data/data目錄存放存儲(chǔ)包私有數(shù)據(jù)魏烫,對(duì)于設(shè)備中每一個(gè)安裝的 App,系統(tǒng)都會(huì)在內(nèi)部存儲(chǔ)空間的 data/data 目錄下以應(yīng)用包名為名字自動(dòng)創(chuàng)建與之對(duì)應(yīng)的文件夾肝箱。
用戶(hù)卸載 App 時(shí)哄褒,系統(tǒng)自動(dòng)刪除 data/data 目錄下對(duì)應(yīng)包名的文件夾及其內(nèi)容。
該目錄下又把存儲(chǔ)內(nèi)容進(jìn)行了分類(lèi):
data/data/包名/cache: 存放的 APP 的緩存信息
data/data/包名/databases: 存放 APP 的數(shù)據(jù)庫(kù)信息
data/data/包名/files: 存放 APP 的文件信息
data/data/包名/shared_prefs: 存放 APP 內(nèi)的 SharedPreferences
所以他這個(gè)數(shù)據(jù)比較隱私煌张,如果手機(jī)沒(méi)有root的話是沒(méi)有權(quán)限獲取的呐赡,為了做這個(gè)小功能再把我小米手機(jī)root一下?感覺(jué)不太好骏融,于是下載了genymotion模擬器链嘀,我也不知道為啥我的genymotion好像一直都有root權(quán)限井辜,這塊我沒(méi)太弄明白,有清楚的人可以給我留言管闷。
因?yàn)橛脩?hù)要的是macos的app,所以需要在macos app里面獲取genymotion模擬器里面的/data/data數(shù)據(jù)窃肠,經(jīng)過(guò)一番研究最終通過(guò):
在程序你面添加:
rm -rf ~/Downloads/keystore.xml
~/Library/Android/sdk/platform-tools/adb pull /data/data/com.whatsapp/shared_prefs/keystore.xml ~/Downloads/keystore.xml
通過(guò)adb的pull命令可以把設(shè)備里面的文件拷到外面包个,這里需要設(shè)置程序權(quán)限:
最后效果:
https://www.youtube.com/watch?v=mpXtJ-lM480&feature=youtu.be
拓展
如果在安卓上訪問(wèn)別的apk的私密數(shù)據(jù)該怎么做呢?
Java的File類(lèi)可以對(duì)文件做一系列操作冤留。開(kāi)始的思路是通過(guò)File.listFiles()來(lái)獲取目標(biāo)應(yīng)用的子文件列表碧囊,然后對(duì)指定的文件進(jìn)行exists()判斷。但是當(dāng)進(jìn)入到包名下的一個(gè)目錄時(shí)纤怒,里面的文件夾用isDirectory()和isFile()判斷糯而,返回的都是false,同時(shí)用canRead()做判斷泊窘,返回的也是false熄驼。判斷是文件訪問(wèn)權(quán)限的問(wèn)題,但是通過(guò)chmod 744之后烘豹,canRead()仍然返回false瓜贾。看樣子通過(guò)File對(duì)/data分區(qū)的操作仍然受系統(tǒng)安全性限制携悯。
后來(lái)找到了另外一種方法來(lái)判斷祭芦,通過(guò)執(zhí)行shell命令ls -R ,獲取指定應(yīng)用目錄下文件的列表憔鬼,從而判斷文件是否存在龟劲。
這是執(zhí)行shell命令的方法
public static ArrayList execCmdsforResult(String[] cmds) {
ArrayList<String> list = new ArrayList<String>();
try {
Process process = Runtime.getRuntime().exec("su");
OutputStream os = process.getOutputStream();
process.getErrorStream();
InputStream is = process.getInputStream();
int i = cmds.length;
for (int j = 0; j < i; j++) {
String str = cmds[j];
os.write((str + "\n").getBytes());
}
os.write("exit\n".getBytes());
os.flush();
os.close();
BufferedReader reader = new BufferedReader(new InputStreamReader(is));
while (true) {
String str = reader.readLine();
if (str == null)
break;
list.add(str);
}
reader.close();
process.waitFor();
process.destroy();
return list;
} catch (Exception localException) {
}
return list;
}