前几天看到同事里有一个界面绚丽的应用,觉得有点意思,就让他把APK发给我,我想反编译看看里面的代码。结果,这哥们在手机里找了好一阵子,最后给我说:手机没有root,找不到APK文件在哪里。我再让他试试其他机子,结果都差不多:要不然找起来很麻烦,要不然根本都找不到。这时,测试的妹子说:手机QQ有这个功能。我打开手机QQ一看,果然有,平时都没有注意到啊。

android app QQ客服功能怎么做 手机qq客服在哪里_分享

这个功能点稍作总结:
- 每个item包括:应用的icon,名字,安装文件的大小,最后更新时间
- 点击item分享其对应的APK文件

看到这里,心里怪痒痒的,我们也能做这么个类似的东西么?
能的!必须能!否则在测试的妹子面前怎么能抬起头!?

我们先获取手机中已经安装的应用:

List<PackageInfo> packageInfoList = mPackageManager.getInstalledPackages(0);

这些应用已经都躺在这里了,我们现在就一步一步地来找出每个应用的相关信息。

(1) 获取应用的名称

/**
     * 获取应用的名称
     */
   public String getApplicationName(String packageName,PackageManager packageManager) {
        String applicationName=null;
        try {
            ApplicationInfo applicationInfo = packageManager.getApplicationInfo(packageName, 0);
            applicationName = (String) packageManager.getApplicationLabel(applicationInfo);
        } catch (PackageManager.NameNotFoundException e) {

        }
        return applicationName;
    }

嗯哼,这个不难,刚上路的小司机也可以轻松的搞定。

(2) 获取应用的icon

应用的名字我们容易获得,那它的icon又在哪里呢?
既然获得应用名字用的是:

packageManager.getApplicationLabel(applicationInfo);

那么是不是有类似的getApplicationIcon方法呢?赶紧试着敲一下代码,AS提示果然有我们想要的东西:

packageManager.getApplicationIcon(ApplicationInfo info)

嗯哼,真愉快,我们猜对啦。把代码运行起来瞅瞅,我的华为手机没有问题。换个三星试试,也对呢;再用HTC跑跑,获取到的居然是个小绿人——系统默认的图标!再从测试妹子那里拿个小米过来,一样啊,没有正确获取到应用对应的图标。

不开心了,喝口水,默默地打开度娘找答案。看到这样的方式:

//.........
        Class pkgParserCls = Class.forName(PATH_PackageParser);
        Class[] typeArgs = new Class[1];
        typeArgs[0] = String.class;
        Constructor pkgParserCt = pkgParserCls.getConstructor(typeArgs);
        Object[] valueArgs = new Object[1];
        valueArgs[0] = apkPath;
        Object pkgParser = pkgParserCt.newInstance(valueArgs);
//.........


居然用的是反射,复杂就不说了而且还影响效率。所以,在不得已的情况下还是不要这么做了。看来想通过PackageManager获取应用的Icon是不行了,那就换个角度从PackageInfo入手试试,看到一个字段:

public ApplicationInfo applicationInfo;

官方文档的解释是: Information collected from the application。也就是说这个字段包含了App的众多信息。所以,接着看ApplicationInfo里面有啥东西,扫了一眼,看到一个东西:

//Retrieve the current graphical icon associated with this item.
 public Drawable loadIcon(PackageManager pm) {
        return pm.loadItemIcon(this, getApplicationInfo());
    }

利用该方法才可以避免在某些机型上无法获取应用的icon的bug。手边的机子试了一遍,都没问题。这个小问题解决了,就接着往下走。

(3) 获取应用的最后更新时间
这个也挺容易的,PackageInfo中有相应的字段:

public long lastUpdateTime;

当然这个值是个毫秒值,需要利用SimpleDateFormat将其转换成项目需要的日期格式。
有些情况下还需要获取应用的第一次安装时间,PackageInfo中也有相应的字段:

public long firstInstallTime;

同理,也需要对其进行格式化。

(4) 获取Apk文件大小
要获取Apk文件大小,首先得找到Apk文件。就像我想周末和妹子去逛街,前提是我得有个妹子啊(打住,不说了,眼泪滴到键盘上了)
但是它到底在哪里呢?幻想着利用PackageManager是不行的,它根本没有类似于getApplicationApk( )的方法。
那怎么办呢?喔,还记得前面提到的PackageInfo中的ApplicationInfo字段么?我们继续去里面找,看看有没有啥收获,在源码501行发现一个字段:

public String sourceDir;

官方文档是这么描述的:Full path to the base APK for this application。嗯哼,bingo!找到了就是它,它代表了APK文件的完整路径。文件路径已经拿到了,啥都好办了(就像知道了妹子住哪里,就可以……..)

File apkFile = new File(packageInfo.applicationInfo.sourceDir);

我们将该路径封装成一个文件,再获取它的大小即可.

apkFile.length() / 1024 / 1024

在此处将文件大小转换成了MB单位,比如豌豆荚的APK文件为6.46MB

好了,想要的东西我们都找到了,我们用一个ListView把每个APK的相关信息作为item展示出来就行了。有个小问题请注意:获取手机中APK信息,这是一个耗时的过程,所以我们要在子线程中来做这个事情。

好了,看看做出来的效果

android app QQ客服功能怎么做 手机qq客服在哪里_字段_02

我们接着实现点击item分享Apk文件:

private class ItemClickListenerImpl implements AdapterView.OnItemClickListener{
        @Override
        public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
            File apkFile = mAppInfoList.get(position).getApkFile();
            Intent intent = new Intent();
            intent.setAction(Intent.ACTION_SEND);
            intent.setType("*/*");
            intent.putExtra(Intent.EXTRA_STREAM, Uri.fromFile(apkFile));
            startActivity(intent);
        }
    }

在此处Apk文件放到intent中再调用系统自带的分享功能即可。

界面做出来了,功能也实现了,再回过头来看,其实不是很难。难点在于我们不知道——不知道它的icon和Apk文件到底放在哪里了。这让我想起了拉姆斯菲尔德的一段话:

We also know there are known unknowns; that is to say we know there
are some things we do not know. But there are also unknown unknowns –
the ones we don’t know we don’t know.

嘿嘿,我们不是不会,只是不知道罢了。