第一篇随笔就此开始。
1. 起源
思路源自于项目开发过程中。需要确认apk文件版本以验证其功能差异以便于定位问题,于是度娘,得到APK信息查看器(APK-info)这个工具,其版本号为0.2。
它能显示apk详细的信息,如下图示:
但它使用不够方便,只能通过双击打开对话框找到apk文件然后显示,或者拖放apk到其图标上打开,不支持拖放至界面打开。它也没有再打开入口,且对中文支持很不好,如右图百度手机卫士apk信息。
2. 原理
析其原理,原来通过aapt.exe解开apk包中AndroidManifest.xml文件来实现信息展示。
而aapt.exe使用方法,网上诸多教程。apk信息尽存于AndroidManifest.xml中,它是加密的xml文件,用aapt之dump命令做个解析尝试,其语法如下:
>aapt dump badging QQ_482.apk
似乎想要的信息都有了……慢着,中文呢?
>aapt dump badging QQ_482.apk
中文显示乱码。乱就乱吧,咱转码!
如何不自己写个类似工具?好,整!
3. 实现
以此为思路,c#实现解析,获取cmd管道输出数据,核心代码如下(为界面响流畅,我置解析入一线程中):
private void Decoder(object state)
{
if (!File.Exists(this.apkPath))
return;
string aaptPath = Path.Combine(this.appPath, @"tools\aapt.exe");
if (!File.Exists(aaptPath))
aaptPath = Path.Combine(this.appPath, @"aapt.exe");
if (!File.Exists(aaptPath))
{
var handler = AaptNotFoundEvent;
if (handler != null)
handler();
return;
} var startInfo = new ProcessStartInfo(aaptPath);
string args = string.Format("dump badging \"{0}\"", this.apkPath);
startInfo.Arguments = args;
startInfo.UseShellExecute = false;
startInfo.RedirectStandardOutput = true;
startInfo.CreateNoWindow = true;
using (var process = Process.Start(startInfo))
{
var sr = process.StandardOutput;
while (!sr.EndOfStream)
{
infos.Add(sr.ReadLine());
}
process.WaitForExit();
//解析
ParseInfo(sr.CurrentEncoding);
}
}
//application: label='MobileGo™' icon='r/l/icon.png'
if (info.IndexOf("application:") == )
{
string appName = GetKeyValue(info, "label=");
this.AppName = Encoding.UTF8.GetString(currentEncoding.GetBytes(appName));
this.IconPath = GetKeyValue(info, "icon=");
GetAppIcon(this.IconPath);
}
其执行界面如下:
可以看得出,对中文支持,仍然不够友好,即便通过utf-8到默认中文编码转换。
怎么办呢?
4. 改进
而用其直接输出于一外部文件中,中文显示却是正确的:
>aapt dump badging QQshurufa_1991.apk > info.txt
解析输出文件吧!对这种方案,我一开始是抵触的,因为不想额外成生文件;能截用内部cmd管道输出,我就不愿生成外部文件,说是情怀也好洁癖也罢,只是个人喜好。
但目前转码方案尝试无效,就只得用吧,获取输出信息代码如下:
private void Decoder(object state)
{
if (!File.Exists(this.apkPath))
return;
string aaptPath = Path.Combine(this.appPath, @"tools\aapt.exe");
if (!File.Exists(aaptPath))
aaptPath = Path.Combine(this.appPath, @"aapt.exe");
if (!File.Exists(aaptPath))
{
var handler = AaptNotFound;
if (handler != null)
handler();
return;
} StringBuilder sb = new StringBuilder();
int result = GetShortPathName(aaptPath, sb, );
if (result != )
aaptPath = sb.ToString(); var startInfo = new ProcessStartInfo("cmd.exe");
string dumpFile = Path.GetTempFileName();
//如此费事做中转,只为处理中文乱码
string args = string.Format("/k {0} dump badging \"{1}\" > \"{2}\" &exit", aaptPath, this.apkPath, dumpFile);
startInfo.Arguments = args;
startInfo.WindowStyle = ProcessWindowStyle.Hidden;
this.infos.Clear();
using (var process = Process.Start(startInfo))
{
process.WaitForExit();
}
if (File.Exists(dumpFile))
{
//解析
using (var sr = new StreamReader(dumpFile, Encoding.UTF8))
{
string line;
while ((line = sr.ReadLine()) != null)
{
this.infos.Add(line);
}
ParseInfo();
}
try
{
File.Delete(dumpFile);
}
catch
{
}
}
}
好吧,一切正常了……只是其中构建cmd管道脚本,颇为费些工夫。
看看下图中文信息显示,一切正常:
5. 后记
这是我第一篇博客,写代码十数年,虽时时想记录经验与心得,奈何懒惰,至此方忍不住,终于写出来。
其实此例成完成已久,只是近来更做完善,查询资料过程中看到Meteoric_cry的博文:windows下apk查看工具的原理,颇觉异曲同工之妙,而他也因此写了APK Helper这一工具,简单而易用。
本欲置源代码入Github,奈何折腾半天,亦未成功,因此计划暂且搁置。
此工具我置于网上,亦根据需要作不定时更新,如有需要伙伴尽管拿去用,下载地址为:ApkInfo.zip。若小伙伴有其它需要,可留言以待
6. 追加
2017-03-14更新:费了许多工夫,终于传代码至GitHub,今夜可以安睡。项目址为:https://github.com/awei78/ApkInfo
希望有需要的朋友以做参考。或者,咱们一起更完善它,以适合实际需要。