很多情况下,我们需要通过APK文件来得到它的一些信息。 (此时此APK不一定被安装了)

0. 基础知识:
可以通过android.content.Context的方法 getPackageManager()得到一个PackageManager实例。
PackageManager有一个方法:
PackageInfo getPackageArchiveInfo (String archiveFilePath, int flags);
其中参数一:archiveFilePath。就是指一个APK文件绝对路径。

请注意它的返回值。是个PackageInfo。它包含Pakcage所有信息,可以认为,这些信息都是从AndroidManifest.xml中收集而来。

1. 得到APK中Activity信息:
PackageInfo中包含:public ActivityInfo[] activities
它会得到AndroidManifest.xml中<application>中所有<Activity> tag.

例如:

<application android:label="@string/app_name"
                 android:debuggable="true">
        <activity android:name=".HelloJni"
                  android:label="@string/app_name">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

ActivityInfo:从 AndroidManifest.xml's <activity> and <receiver> tags得到的信息。

可以从这一项中得到和Activity有关的所有信息。这些内容都是从<activity>或<receiver>中得到。

例1:得到Activity各信息:
想要得到Activity信息,需要参数2设置为: PackageManager.GET_ACTIVITIES 

PackageInfo PI = PM.getPackageArchiveInfo("/data/hellojni.apk",PackageManager.GET_ACTIVITIES);

 //PackageInfo中得到的PackageName
Log.w("3DiJoy", "PackageInfo . PackageName:" + PI.packageName); Log.w("3DiJoy", String.format("There are [%d] Activity", PI.activities.length));
Log.w("3DiJoy", String.format("Activity in Porcess Name: [%s]", PI.activities[0].processName));

//注意:此处第一次看到ProcessName:
//The name of the process this component should run in. From the "android:process" attribute or, //if not set, the same as applicationInfo.processName
//即此项从main.xml中得到,为android:process项目内容。如果没有设置,则与applicationInfo.processName相同。(注1)

//Activity 所在的PackageName。应该与上面PockageInfo中得到的PackageName相同。

Log.w("3DiJoy", "Activity in Package PackageName:" + PI.activities[0].packageName);
//Activity Name:<activity android:name=".HelloJni"。中读取到的
Log.w("3DiJoy", "Activity Name:" + PI.activities[0].name);

2. 得到APK中Application信息:
Information collected from the <application> tag, or null if there was none。
呵呵,如之前所述,其实PackageInfo中所有项目都是从Main.xml中取到的,之前的Activity如此,现在的Application也是如此。

PackageInfo中有一项:
public ApplicationInfo applicationInfo

Log.w("3DiJoy", "Appliction DataDir:" + PI.applicationInfo.dataDir);
Log.w("3DiJoy", "Application SourceDir:" + PI.applicationInfo.sourceDir);
//Log.w("3DiJoy", "Application JNI LIbdir:" + PI.applicationInfo.nativeLibraryDir);
Log.w("3DiJoy", "Application permission:" + PI.applicationInfo.permission);
Log.w("3DiJoy", "Application in porcessName is :" + PI.applicationInfo.processName);//(注2)
Log.w("3DiJoy", String.format("minSDK Version: [%d]", PI.applicationInfo.targetSdkVersion));
Log.w("3DiJoy", "Application Name:" + PI.applicationInfo.name);
Log.w("3DiJoy", "Application in Package " + PI.applicationInfo.packageName);

3.得到APK中版本信息:

//<manifest> tag's "name" attribute.
Log.w("3DiJoy", "PackageInfo . PackageName:" + PI.packageName);

Log.w("3DiJoy", "PackageInfo . Version String:" + PI.versionName);
Log.w("3DiJoy", String.format("Version code:[%d]", PI.versionCode));
分别从: <manifest> tag's versionCode attribute和 <manifest> tag's versionName attribute.中得到信息。


注1: 
ProcessName相关内容:
PI.activities[0].processName .它其实是activityInfo的父类: android.content.pm.ComponentInfo

的成员变量。是指此activity所运行的Process的名字。
此信息从main.xml的 "android:process" attribute中得到。如果没有设置,则与applicationInfo.processName相同。

注2:
PI.applicationInfo.processName 
此Application将要运行的ProcessName.从Main.xml的"process" attribute得到。如果没有设置,则应该与PackageName相同。

注3:
PI.applicationInfo.packageName 
Application所在的PackageName。


但Sam发现:
在Android2.2下。
从Activity中取得ProcessName,为null.(
PI.activities[0].processName)

从Application中取得ProcessName也为空。(PI.applicationInfo.processName)
但从Application中取得packageName却是正确的(PI.applicationInfo.packageName, 与PI.packageName相同)

注4:
getPackageArchiveInfo () 的参数1, archive 指的就是APK文件。
 
 

1)获取apk文件的图标

  1. public static Drawable getApkFileIcon(String apkPath, Context context) {
  2. PackageManager pm = context.getPackageManager();
  3. PackageInfo info = pm.getPackageArchiveInfo(apkPath,
  4. PackageManager.GET_ACTIVITIES);
  5. if (info != null) {
  6. ApplicationInfo appInfo = info.applicationInfo;
  7. if (appInfo != null) {
  8. try {
  9. appInfo.publicSourceDir = apkPath;
  10. return pm.getApplicationIcon(appInfo);
  11. } catch (OutOfMemoryError e) {
  12. }
  13. }
  14. }
  15. return null;
  16. }

 

2)获取AndroidManifest.xml中<meta-data>标签中定义的值

  1. PackageManager manager = this.getPackageManager();
  2. try {
  3. // 设置PackageManager.GET_META_DATA标识位是必须的
  4. PackageInfo info = manager.getPackageInfo(this.getPackageName(),
  5. PackageManager.GET_CONFIGURATIONS | PackageManager.GET_META_DATA);
  6. Object myChannel = info.applicationInfo.metaData.get("meta-data-key");
  7. if ((myChannel != null) && (myChannel instanceof Integer)) {
  8. // do something
  9. }
  10. } catch (NameNotFoundException e) {
  11. e.printStackTrace();
  12. }
 
 
public class test4 extends Activity {
    
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);       
        setContentView(R.layout.main);    
        String archiveFilePath="sdcard/download/Law.apk";//安装包路径
        PackageManager pm = getPackageManager();  
        PackageInfo info = pm.getPackageArchiveInfo(archiveFilePath,PackageManager.GET_ACTIVITIES);  
        if(info != null){  
            ApplicationInfo appInfo = info.applicationInfo;  
            String appName = pm.getApplicationLabel(appInfo).toString();  
            String packageName = appInfo.packageName;  //得到安装包名称
            String version=info.versionName;       //得到版本信息     
            Toast.makeText(test4.this, "packageName:"+packageName+";version:"+version, Toast.LENGTH_LONG).show();
            Drawable icon = pm.getApplicationIcon(appInfo);//得到图标信息
            TextView tv = (TextView)findViewById(R.id.tv); //显示图标
            tv.setBackgroundDrawable(icon);
        }  
    }
    
}
 
 

Android 下得到 未安装APK包含信息 等的更多相关文章

  1. android获取未安装APK签名信息及MD5指纹

    站在巨人的肩膀上写博客: http://blog.csdn.net/wulianghuan/article/details/18400581 http://www.jb51.net/article/7 ...

  2. Android 8.0+ 更新安装apk失败的问题

    最近做项目发现Android 8.0+ 更新安装apk时 出现安装失败的情况  总结原因是 缺少安装的权限 Android 8.0 (Android O)为了针对一些流氓软件引导用户安装其他无关应用. ...

  3. Android应用开发提高系列(4)——Android动态加载(上)——加载未安装APK中的类

    前言 近期做换肤功能,由于换肤程度较高,受限于平台本身,实现起来较复杂,暂时搁置了该功能,但也积累了一些经验,将分两篇文章来写这部分的内容,欢迎交流! 关键字:Android动态加载 声明 欢迎转载, ...

  4. 加载未安装APK中的类

    一.前提 目的:动态加载SD卡中Apk的类. 注意:被加载的APK是未安装的. 相关:本文是本博另外一篇文章:Android动态加载jar/dex的升级版. 截图: 成功截图: 二.准备 准备被调用A ...

  5. 我的Android进阶之旅------>解决Android Studio编译后安装apk报错:The APK file does not exist on disk

    1.错误描述 今天用Android Studio编译应用后安装APK的时候,报错了,错误如下所示: The APK file build\outputs\apk\OYP_2.3.4_I2Base_64 ...

  6. Android 在代码中安装 APK 文件

    废话不说,上代码 private void install(String filePath) { Log.i(TAG, "开始执行安装: " + filePath); File a ...

  7. 安卓(Android)手机如何安装APK?

    我大天朝的安卓手机只能在一个被阉割的APP市场里玩耍,有些APP可能需要直接安装APK文件.APK 是 Android Package (安卓安装包)安卓手机如何安装APK呢? 在电脑上下载安装APK ...

  8. Android手机提示“未安装应用程序”

    用eclipse调试应用时,遇到了这个问题,网上给出的解决方案倒是挺多,但似乎一个都没奏效,而且我手机也重启了,还是有问题,郁闷ing-   然后看到一篇文章指出,可能不是签名和SD的卡问题,而是我们 ...

  9. Android笔记(预安装APK)

    一般一个安卓的产品在出厂时,会预安装许多APK,关于这些APP,主要分为下面这几类 1.系统级别APK 这一类应用一般是:电话/设置或者厂家自己特定的应用. 2.系统预安装APK 因为商业原因,产品出 ...

随机推荐

  1. XCode工程内多Targets

    XCode工程内多Targets 可以认为一个target对应一个新的product(基于同一份代码的情况下). 虽然代码是同一份, 但编译设置(比如编译条件), 以及包含的资源文件却可以有很大的差别 ...

  2. 在VS中生成后拷贝文件

    环境:win7_64旗舰版,VS2013 工作项目中,一般会使用第三方库,当修改并重新编译第三方库后,需要将DLL文件拷贝到工作项目下的生成目录中,每次手动拷贝比较繁琐,VS提供自定义生成事件,允许我 ...

  3. python 字符转换

    我们所看到的“明文字符串”,都是经过编码(比如ASCII.Uncoded.UTF-8.GB-2312等)后呈现在我们面前的. 文本中“3082”想要“所见到所得”到内存中处理,必须decode('he ...

  4. 照着例子学习protobuf-python

    以下是照着python操作protobuf进行的protobuf-python的学习笔记: 首先是protobuf的下载与安装: 1 由于google被墙,所以去github上面搜索了一下protob ...

  5. HDU 3338 Kakuro Extension

    网络最大流 TLE了两天的题目.80次Submit才AC,发现是刘汝佳白书的Dinic代码还可以优化.....瞬间无语..... #include<cstdio> #include< ...

  6. 句柄C++

    C++中的句柄 这个句柄只是从英文handle翻译过来的,只问句是什么意思难以解释,这个是我从别人的空间收集的信息, 功能上的理解: 什么是"句柄"(handle),handle的 ...

  7. static静态初始化块

    Java 中可以通过初始化块进行数据赋值.如: 在类的声明中,可以包含多个初始化块,当创建类的实例时,就会依次执行这些代码块.如果使用 static 修饰初始化块,就称为静态初始化块. 需要特别注意: ...

  8. 依赖注入容器Autofac与MVC集成

    Autofac是应用于.Net平台的依赖注入(DI,Dependency Injection)容器,具有贴近.契合C#语言的特点.随着应用系统的日益庞大与复杂,使用Autofac容器来管理组件之间的关 ...

  9. Entity Framework技巧系列之十四 - Tip 56

    提示56. 使用反射提供程序编写一个OData Service 在TechEd我收到一大堆有关将数据作为OData暴露的问题. 到目前为止你大概知道可以使用数据服务与Entity Framework将 ...

  10. 命令 shell 学习

    for i in a b c do echo $i done !ser  历史补全 > 正确信息输出文件 >>正确信息输出文件  ,追加 2>错误信息输出文件 2>> ...