打赏

第一篇随笔就此开始。

 

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:") == 0)
{
    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(255);
            int result = GetShortPathName(aaptPath, sb, 255);
            if (result != 0)
                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(2000);
            }
            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
希望有需要的朋友以做参考。或者,咱们一起更完善它,以适合实际需要。

posted on 2017-03-12 23:45  楚人无衣  阅读(2320)  评论(0编辑  收藏  举报