仿QQ获取设备中APK并分享

onau7184 8年前
   <p><strong>我和郭霖</strong></p>    <p>13年,郭霖开始写技术博客。我在CSDN的博客频道逛着玩,刚好看到他的文章,于是点击进去阅读。看完之后最大的感受就是:代码整齐,技术深厚;文章写得图文并茂,有条有理。从那时起,我就成了郭霖的一个粉丝,每周四早晨都等着看他的更新。诚实地说:没有脑残地追过星,但这么实在地喜欢看一个人写的技术博客,这还是头一回。其实,从他每篇文章的点击量和评论数也可以看出:大家对郭霖非常的认可。他也常出现在CSDN的首页,有时还无耻地连续出现几次。去年,部门经理交给我十几个大学毕业生,老大问我:他们适合从哪里学起?我基本没有过脑子地说:每个人买一本《第一行代码》,快去吧,京西十一点前下单下午就能送到。从那以后,部门来的新手,公司都会让他们照着这本书夯实一下基础。</p>    <p>嗯哼,期待郭大婶带给我们更多的分享。</p>    <p>书归正传。</p>    <p>前几天看到同事里有一个界面绚丽的应用,觉得有点意思,就让他把APK发给我,我想反编译看看里面的代码。结果,这哥们在手机里找了好一阵子,最后给我说:手机没有root,找不到APK文件在哪里。我再让他试试其他机子,结果都差不多:要不然找起来很麻烦,要不然根本都找不到。这时,测试的妹子说:手机QQ有这个功能。我打开手机QQ一看,果然有,平时都没有注意到啊。</p>    <p><img alt="这里写图片描述" src="https://simg.open-open.com/show/2d35e7d1b8b9379d063a5a04ab49f548.jpg"></p>    <p>这个功能点稍作总结:<br> - <strong>每个item包括:应用的icon,名字,安装文件的大小,最后更新时间</strong><br> - <strong>点击item分享其对应的APK文件</strong></p>    <p>看到这里,心里怪痒痒的,我们也能做这么个类似的东西么?<br> 能的!必须能!否则在测试的妹子面前怎么能抬起头!?</p>    <p>我们先获取手机中已经安装的应用:</p>    <pre>  <code class="language-java">List<PackageInfo> packageInfoList = mPackageManager.getInstalledPackages(0);</code></pre>    <p>这些应用已经都躺在这里了,我们现在就一步一步地来找出每个应用的相关信息。</p>    <p><strong>(1) 获取应用的名称</strong></p>    <pre>  <code class="language-java">    /** * 获取应用的名称 */     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;      }</code></pre>    <p>嗯哼,这个不难,刚上路的小司机也可以轻松的搞定。</p>    <p><strong>(2) 获取应用的icon</strong></p>    <p>应用的名字我们容易获得,那它的icon又在哪里呢?<br> 既然获得应用名字用的是:</p>    <blockquote>     <p>packageManager.getApplicationLabel(applicationInfo);</p>    </blockquote>    <p>那么是不是有类似的getApplicationIcon方法呢?赶紧试着敲一下代码,AS提示果然有我们想要的东西:</p>    <blockquote>     <p>packageManager.getApplicationIcon(ApplicationInfo info)</p>    </blockquote>    <p>嗯哼,真愉快,我们猜对啦。把代码运行起来瞅瞅,我的华为手机没有问题。换个三星试试,也对呢;再用HTC跑跑,获取到的居然是个小绿人——系统默认的图标!再从测试妹子那里拿个小米过来,一样啊,没有正确获取到应用对应的图标。</p>    <p>不开心了,喝口水,默默地打开度娘找答案。看到这样的方式:</p>    <pre>  <code class="language-java">//.........          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);  //.........</code></pre>    <p>居然用的是反射,复杂就不说了而且还影响效率。所以,在不得已的情况下还是不要这么做了。看来想通过PackageManager获取应用的Icon是不行了,那就换个角度从PackageInfo入手试试,看到一个字段:</p>    <blockquote>     <p>public ApplicationInfo applicationInfo;</p>    </blockquote>    <p>官方文档的解释是: Information collected from the application。也就是说这个字段包含了App的众多信息。所以,接着看ApplicationInfo里面有啥东西,扫了一眼,看到一个东西:</p>    <pre>  <code class="language-java">//Retrieve the current graphical icon associated with this item.   public Drawable loadIcon(PackageManager pm) {          return pm.loadItemIcon(this, getApplicationInfo());      }</code></pre>    <p>利用该方法才可以避免在某些机型上无法获取应用的icon的bug。手边的机子试了一遍,都没问题。这个小问题解决了,就接着往下走。</p>    <p><strong>(3) 获取应用的最后更新时间</strong><br> 这个也挺容易的,PackageInfo中有相应的字段:</p>    <blockquote>     <p>public long lastUpdateTime;</p>    </blockquote>    <p>当然这个值是个毫秒值,需要利用SimpleDateFormat将其转换成项目需要的日期格式。<br> 有些情况下还需要获取应用的第一次安装时间,PackageInfo中也有相应的字段:</p>    <blockquote>     <p>public long firstInstallTime;</p>    </blockquote>    <p>同理,也需要对其进行格式化。</p>    <p><strong>(4) 获取Apk文件大小</strong><br> 要获取Apk文件大小,首先得找到Apk文件。就像我想周末和妹子去逛街,前提是我得有个妹子啊(打住,不说了,眼泪滴到键盘上了)<br> 但是它到底在哪里呢?幻想着利用PackageManager是不行的,它根本没有类似于getApplicationApk( )的方法。<br> 那怎么办呢?喔,还记得前面提到的PackageInfo中的ApplicationInfo字段么?我们继续去里面找,看看有没有啥收获,在源码501行发现一个字段:</p>    <blockquote>     <p>public String sourceDir;</p>    </blockquote>    <p>官方文档是这么描述的:Full path to the base APK for this application。嗯哼,bingo!找到了就是它,它代表了APK文件的完整路径。文件路径已经拿到了,啥都好办了(就像知道了妹子住哪里,就可以……..)</p>    <blockquote>     <p>File apkFile = new File(packageInfo.applicationInfo.sourceDir);</p>    </blockquote>    <p>我们将该路径封装成一个文件,再获取它的大小即可.</p>    <blockquote>     <p>apkFile.length() / 1024 / 1024</p>    </blockquote>    <p>在此处将文件大小转换成了MB单位,比如豌豆荚的APK文件为6.46MB</p>    <p>好了,想要的东西我们都找到了,我们用一个ListView把每个APK的相关信息作为item展示出来就行了。有个小问题请注意:获取手机中APK信息,这是一个耗时的过程,所以我们要在子线程中来做这个事情。</p>    <p>好了,看看做出来的效果</p>    <p><img alt="这里写图片描述" src="https://simg.open-open.com/show/959b90cc397596b07243fde71cff5159.jpg"></p>    <p>我们接着实现点击item分享Apk文件:</p>    <pre>  <code class="language-java"> 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);          }      }  </code></pre>    <p>在此处Apk文件放到intent中再调用系统自带的分享功能即可。</p>    <p>界面做出来了,功能也实现了,再回过头来看,其实不是很难。难点在于我们不知道——不知道它的icon和Apk文件到底放在哪里了。这让我想起了拉姆斯菲尔德的一段话:</p>    <blockquote>     <p>We also know there are known unknowns; that is to say we know there<br> are some things we do not know. But there are also unknown unknowns –<br> the ones we don’t know we don’t know.</p>    </blockquote>    <p>嘿嘿,我们不是不会,只是不知道罢了。</p>    <p><a href="/misc/goto?guid=4959672241934952977">源码下载</a></p>    <p>来自: http://blog.csdn.net/lfdfhl/article/details/51286284</p>