Android系统为我们提供了很多服务管理的类,包括ActivityManager、PowerManager(电源管理)、AudioManager(音频管理)

等。除此之外,还提供了一个PackageManger管理类,它的主要职责是管理应用程序包。 通过它,我们就可以获取应用程序信息。

引入: AnroidManifest.xml文件节点说明:

Android中获取应用程序(包)的信息-----PackageManager的使用_第1张图片

一、相关类的介绍

PackageItemInfo类

说明: AndroidManifest.xml文件中所有节点的基类,提供了这些节点的基本信息:a label、icon、 meta-data。它并不

直接使用,而是由子类继承然后调用相应方法。

常用字段

public int icon 获得该资源图片在R文件中的值 (对应于android:icon属性)

public int labelRes 获得该label在R文件中的值(对应于android:label属性)

public String name 获得该节点的name值(对应于android:name属性)

publicString packagename 获得该应用程序的包名(对应于android:packagename属性)

常用方法

Drawable loadIcon(PackageManager pm) 获得当前应用程序的图像

CharSequence loadLabel(PackageManager pm) 获得当前应用程序的label

ActivityInfo类 继承自 PackageItemInfo

说明: 获得应用程序中<activity/>或者 <receiver />节点的信息 。我们可以通过它来获取我们设置的任何属性,包括

theme 、launchMode、launchmode等

常用方法继承至PackageItemInfo类中的loadIcon()和loadLabel()

ServiceInfo 类

说明: 同ActivityInfo类似 ,同样继承自 PackageItemInfo,只不过它表示的是<service>节点信息。

ApplicationInfo类 继承自 PackageItemInfo

说明:获取一个特定引用程序中<application>节点的信息。

字段说明

    flags字段:FLAG_SYSTEM 系统应用程序

       FLAG_EXTERNAL_STORAGE 表示该应用安装在sdcard中

常用方法继承至PackageItemInfo类中的loadIcon()和loadLabel()

ResolveInfo类

说明:根据<intent>节点来获取其上一层目录的信息,通常是<activity>、<receiver>、<service>节点信息。

常用字段

public ActivityInfo activityInfo 获取 ActivityInfo对象,即<activity>或<receiver >节点信息

public ServiceInfo serviceInfo 获取 ServiceInfo对象,即<activity>节点信息

常用方法

Drawable loadIcon(PackageManager pm) 获得当前应用程序的图像

CharSequence loadLabel(PackageManager pm)获得当前应用程序的label

PackageInfo类

说明:手动获取AndroidManifest.xml文件的信息 。

常用字段

public StringpackageName 包名

public ActivityInfo[] activities 所有<activity>节点信息

public ApplicationInfo applicationInfo<application>节点信息,只有一个

publicActivityInfo[] receivers 所有<receiver>节点信息,多个

public ServiceInfo[] services 所有<service>节点信息 ,多个

PackageManger 类

说明: 获得已安装的应用程序信息 。可以通过getPackageManager()方法获得。

常用方法

public abstract PackageManager getPackageManager()

功能:获得一个PackageManger对象

public abstrac tDrawable getApplicationIcon(StringpackageName)

参数: packageName 包名

功能:返回给定包名的图标,否则返回null

public abstractApplicationInfogetApplicationInfo(StringpackageName, int flags)

参数:packagename 包名

flags 该ApplicationInfo是此flags标记,通常可以直接赋予常数0即可

功能:返回该ApplicationInfo对象

public abstractList<ApplicationInfo> getInstalledApplications(int flags)

参数:flag为一般为GET_UNINSTALLED_PACKAGES,那么此时会返回所有ApplicationInfo。我们可以对ApplicationInfo

的flags过滤,得到我们需要的。

功能:返回给定条件的所有PackageInfo

public abstractList<PackageInfo> getInstalledPackages(int flags)

参数如上

功能:返回给定条件的所有PackageInfo

publicabstractResolveInfo resolveActivity(Intentintent, int flags)

参数: intent查寻条件,Activity所配置的action和category

flags:MATCH_DEFAULT_ONLY :Category必须带有CATEGORY_DEFAULT的Activity,才匹配

GET_INTENT_FILTERS :匹配Intent条件即可

GET_RESOLVED_FILTER匹配Intent条件即可

功能 :返回给定条件的ResolveInfo对象(本质上是Activity)

public abstractList<ResolveInfo> queryIntentActivities(Intentintent, int flags)

参数同上

功能 :返回给定条件的所有ResolveInfo对象(本质上是Activity),集合对象

public abstractResolveInfo resolveService(Intentintent, int flags)

参数同上

功能 :返回给定条件的ResolveInfo对象(本质上是Service)

public abstractList<ResolveInfo>queryIntentServices(Intentintent, int flags)

参数同上

功能 :返回给定条件的所有ResolveInfo对象(本质上是Service),集合对象

二、DEMO讲解

通过前面的介绍,相信您一定很了解了,本质上来讲,这些XXXInfo类不过是我们在AndroidManifest.XML文件中定义的信息,

知道到这点了,理解起来就不是很难了。

下面我透过两个简答的DEMO,来学以致用。

Demo 1: 通过queryIntentActivities()方法,查询Android系统的所有具备ACTION_MAIN和CATEGORY_LAUNCHER

的Intent的应用程序,点击后,能启动该应用,说白了就是做一个类似Home程序的简易Launcher 。

Demo 2:通过getInstalledApplications()方法获取应用,然后对其过滤,查找出我们需要的第三方应用,系统应用,安装在sdcard的应用。

Demo1 :

图:

Android中获取应用程序(包)的信息-----PackageManager的使用_第2张图片

1 、布局文件: 主要有两个:带listview的browse_app_list.xml文件 ;listview的项browse_app_item.xml

browse_app_list.xml

view plain print ?
  1. <?xmlversion="1.0"encoding="utf-8"?>
  2. <LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"
  3. android:orientation="vertical"android:layout_width="fill_parent"
  4. android:layout_height="fill_parent">>
  5. <ListViewandroid:id="@+id/listviewApp"android:layout_width="fill_parent"
  6. android:layout_height="fill_parent"></ListView>
  7. </LinearLayout>


browse_app_item.xmlbrowse_app_item.xml

view plain print ?
  1. <?xmlversion="1.0"encoding="utf-8"?>
  2. <LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="fill_parent"android:layout_height="50dip">
  4. <ImageViewandroid:id="@+id/imgApp"android:layout_width="wrap_content"
  5. android:layout_height="fill_parent"></ImageView>
  6. <RelativeLayoutandroid:layout_width="fill_parent"android:layout_marginLeft="10dip"
  7. android:layout_height="40dip">
  8. <TextViewandroid:id="@+id/tvLabel"android:layout_width="wrap_content"
  9. android:layout_height="wrap_content"android:text="AppLable:"></TextView>
  10. <TextViewandroid:id="@+id/tvAppLabel"android:layout_width="wrap_content"
  11. android:layout_toRightOf="@id/tvLabel"android:layout_height="wrap_content"
  12. android:layout_marginLeft="3dip"android:text="Label"android:textColor="#FFD700"></TextView>
  13. <TextViewandroid:id="@+id/tvName"android:layout_width="wrap_content"
  14. android:layout_height="wrap_content"android:layout_below="@id/tvLabel"
  15. android:text="包名:"></TextView>
  16. <TextViewandroid:id="@+id/tvPkgName"android:layout_width="wrap_content"
  17. android:layout_height="wrap_content"android:layout_below="@id/tvAppLabel"
  18. android:layout_alignLeft="@id/tvAppLabel"android:textColor="#FFD700"></TextView>
  19. </RelativeLayout>
  20. </LinearLayout>

2 、AppInfo.java : 保存应用程序信息的Model类

view plain print ?
  1. /Model类,用来存储应用程序信息
  2. publicclassAppInfo{
  3. privateStringappLabel;//应用程序标签
  4. privateDrawableappIcon;//应用程序图像
  5. privateIntentintent;//启动应用程序的Intent,一般是Action为Main和Category为Lancher的Activity
  6. privateStringpkgName;//应用程序所对应的包名
  7. publicAppInfo(){}
  8. publicStringgetAppLabel(){
  9. returnappLabel;
  10. }
  11. publicvoidsetAppLabel(StringappName){
  12. this.appLabel=appName;
  13. }
  14. publicDrawablegetAppIcon(){
  15. returnappIcon;
  16. }
  17. publicvoidsetAppIcon(DrawableappIcon){
  18. this.appIcon=appIcon;
  19. }
  20. publicIntentgetIntent(){
  21. returnintent;
  22. }
  23. publicvoidsetIntent(Intentintent){
  24. this.intent=intent;
  25. }
  26. publicStringgetPkgName(){
  27. returnpkgName;
  28. }
  29. publicvoidsetPkgName(StringpkgName){
  30. this.pkgName=pkgName;
  31. }
  32. }

3、 BrowseApplicationInfoAdapter.java : 自定义适配器类,为ListView提供视图

view plain print ?
  1. //自定义适配器类,提供给listView的自定义view
  2. publicclassBrowseApplicationInfoAdapterextendsBaseAdapter{
  3. privateList<AppInfo>mlistAppInfo=null;
  4. LayoutInflaterinfater=null;
  5. publicBrowseApplicationInfoAdapter(Contextcontext,List<AppInfo>apps){
  6. infater=(LayoutInflater)context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
  7. mlistAppInfo=apps;
  8. }
  9. @Override
  10. publicintgetCount(){
  11. //TODOAuto-generatedmethodstub
  12. System.out.println("size"+mlistAppInfo.size());
  13. returnmlistAppInfo.size();
  14. }
  15. @Override
  16. publicObjectgetItem(intposition){
  17. //TODOAuto-generatedmethodstub
  18. returnmlistAppInfo.get(position);
  19. }
  20. @Override
  21. publiclonggetItemId(intposition){
  22. //TODOAuto-generatedmethodstub
  23. return0;
  24. }
  25. @Override
  26. publicViewgetView(intposition,Viewconvertview,ViewGrouparg2){
  27. System.out.println("getViewat"+position);
  28. Viewview=null;
  29. ViewHolderholder=null;
  30. if(convertview==null||convertview.getTag()==null){
  31. view=infater.inflate(R.layout.browse_app_item,null);
  32. holder=newViewHolder(view);
  33. view.setTag(holder);
  34. }
  35. else{
  36. view=convertview;
  37. holder=(ViewHolder)convertview.getTag();
  38. }
  39. AppInfoappInfo=(AppInfo)getItem(position);
  40. holder.appIcon.setImageDrawable(appInfo.getAppIcon());
  41. holder.tvAppLabel.setText(appInfo.getAppLabel());
  42. holder.tvPkgName.setText(appInfo.getPkgName());
  43. returnview;
  44. }
  45. classViewHolder{
  46. ImageViewappIcon;
  47. TextViewtvAppLabel;
  48. TextViewtvPkgName;
  49. publicViewHolder(Viewview){
  50. this.appIcon=(ImageView)view.findViewById(R.id.imgApp);
  51. this.tvAppLabel=(TextView)view.findViewById(R.id.tvAppLabel);
  52. this.tvPkgName=(TextView)view.findViewById(R.id.tvPkgName);
  53. }
  54. }
  55. }

4 、MainActivity.java 主工程逻辑

请仔细体会queryIntentActivities()方法,并且注意到排序,它很重要。

view plain print ?
  1. <spanstyle="font-size:13px;">publicclassMainActivityextendsActivityimplementsOnItemClickListener{
  2. privateListViewlistview=null;
  3. privateList<AppInfo>mlistAppInfo=null;
  4. @Override
  5. publicvoidonCreate(BundlesavedInstanceState){
  6. super.onCreate(savedInstanceState);
  7. setContentView(R.layout.browse_app_list);
  8. listview=(ListView)findViewById(R.id.listviewApp);
  9. mlistAppInfo=newArrayList<AppInfo>();
  10. queryAppInfo();//查询所有应用程序信息
  11. BrowseApplicationInfoAdapterbrowseAppAdapter=newBrowseApplicationInfoAdapter(
  12. this,mlistAppInfo);
  13. listview.setAdapter(browseAppAdapter);
  14. listview.setOnItemClickListener(this);
  15. }
  16. //点击跳转至该应用程序
  17. publicvoidonItemClick(AdapterView<?>arg0,Viewview,intposition,
  18. longarg3){
  19. //TODOAuto-generatedmethodstub
  20. Intentintent=mlistAppInfo.get(position).getIntent();
  21. startActivity(intent);
  22. }
  23. //获得所有启动Activity的信息,类似于Launch界面
  24. publicvoidqueryAppInfo(){
  25. PackageManagerpm=this.getPackageManager();//获得PackageManager对象
  26. IntentmainIntent=newIntent(Intent.ACTION_MAIN,null);
  27. mainIntent.addCategory(Intent.CATEGORY_LAUNCHER);
  28. //通过查询,获得所有ResolveInfo对象.
  29. List<ResolveInfo>resolveInfos=pm
  30. .queryIntentActivities(mainIntent,PackageManager.MATCH_DEFAULT_ONLY);
  31. //调用系统排序,根据name排序
  32. //该排序很重要,否则只能显示系统应用,而不能列出第三方应用程序
  33. Collections.sort(resolveInfos,newResolveInfo.DisplayNameComparator(pm));
  34. if(mlistAppInfo!=null){
  35. mlistAppInfo.clear();
  36. for(ResolveInforeInfo:resolveInfos){
  37. StringactivityName=reInfo.activityInfo.name;//获得该应用程序的启动Activity的name
  38. StringpkgName=reInfo.activityInfo.packageName;//获得应用程序的包名
  39. StringappLabel=(String)reInfo.loadLabel(pm);//获得应用程序的Label
  40. Drawableicon=reInfo.loadIcon(pm);//获得应用程序图标
  41. //为应用程序的启动Activity准备Intent
  42. IntentlaunchIntent=newIntent();
  43. launchIntent.setComponent(newComponentName(pkgName,
  44. activityName));
  45. //创建一个AppInfo对象,并赋值
  46. AppInfoappInfo=newAppInfo();
  47. appInfo.setAppLabel(appLabel);
  48. appInfo.setPkgName(pkgName);
  49. appInfo.setAppIcon(icon);
  50. appInfo.setIntent(launchIntent);
  51. mlistAppInfo.add(appInfo);//添加至列表中
  52. System.out.println(appLabel+"activityName---"+activityName
  53. +"pkgName---"+pkgName);
  54. }
  55. }
  56. }
  57. }</span>


好了,第一个Demo完成 。。

Demo 2:

demo2在布局、适配器方面和Demo1一样。只是利用了getInstalledApplications()方法,继而通过ApplicationInfo.flags来挑选

我们希望的ApplicationInfo对象。

图:

Android中获取应用程序(包)的信息-----PackageManager的使用_第3张图片Android中获取应用程序(包)的信息-----PackageManager的使用_第4张图片

过滤应用程序如下:

view plain print ?
  1. packagecom.qiner.appinfo;
  2. importjava.util.ArrayList;
  3. importjava.util.Collections;
  4. importjava.util.List;
  5. importcom.qiner.appinfo.R;
  6. importandroid.app.Activity;
  7. importandroid.app.Application;
  8. importandroid.content.pm.ApplicationInfo;
  9. importandroid.content.pm.PackageManager;
  10. importandroid.os.Bundle;
  11. importandroid.view.View;
  12. importandroid.view.View.OnClickListener;
  13. importandroid.widget.Button;
  14. importandroid.widget.ListView;
  15. publicclassMainActivityextendsActivity{
  16. publicstaticfinalintFILTER_ALL_APP=0;//所有应用程序
  17. publicstaticfinalintFILTER_SYSTEM_APP=1;//系统程序
  18. publicstaticfinalintFILTER_THIRD_APP=2;//第三方应用程序
  19. publicstaticfinalintFILTER_SDCARD_APP=3;//安装在SDCard的应用程序
  20. privateListViewlistview=null;
  21. privatePackageManagerpm;
  22. privateintfilter=FILTER_ALL_APP;
  23. privateList<AppInfo>mlistAppInfo;
  24. privateBrowseApplicationInfoAdapterbrowseAppAdapter=null;
  25. /**Calledwhentheactivityisfirstcreated.*/
  26. @Override
  27. publicvoidonCreate(BundlesavedInstanceState){
  28. super.onCreate(savedInstanceState);
  29. setContentView(R.layout.browse_app_list);
  30. listview=(ListView)findViewById(R.id.listviewApp);
  31. if(getIntent()!=null){
  32. filter=getIntent().getIntExtra("filter",0);
  33. }
  34. mlistAppInfo=queryFilterAppInfo(filter);//查询所有应用程序信息
  35. //构建适配器,并且注册到listView
  36. browseAppAdapter=newBrowseApplicationInfoAdapter(this,mlistAppInfo);
  37. listview.setAdapter(browseAppAdapter);
  38. }
  39. //根据查询条件,查询特定的ApplicationInfo
  40. privateList<AppInfo>queryFilterAppInfo(intfilter){
  41. pm=this.getPackageManager();
  42. //查询所有已经安装的应用程序
  43. List<ApplicationInfo>listAppcations=pm
  44. .getInstalledApplications(PackageManager.GET_UNINSTALLED_PACKAGES);
  45. Collections.sort(listAppcations,
  46. newApplicationInfo.DisplayNameComparator(pm));//排序
  47. List<AppInfo>appInfos=newArrayList<AppInfo>();//保存过滤查到的AppInfo
  48. //根据条件来过滤
  49. switch(filter){
  50. caseFILTER_ALL_APP://所有应用程序
  51. appInfos.clear();
  52. for(ApplicationInfoapp:listAppcations){
  53. appInfos.add(getAppInfo(app));
  54. }
  55. returnappInfos;
  56. caseFILTER_SYSTEM_APP://系统程序
  57. appInfos.clear();
  58. for(ApplicationInfoapp:listAppcations){
  59. if((app.flags&ApplicationInfo.FLAG_SYSTEM)!=0){
  60. appInfos.add(getAppInfo(app));
  61. }
  62. }
  63. returnappInfos;
  64. caseFILTER_THIRD_APP://第三方应用程序
  65. appInfos.clear();
  66. for(ApplicationInfoapp:listAppcations){
  67. //非系统程序
  68. if((app.flags&ApplicationInfo.FLAG_SYSTEM)<=0){
  69. appInfos.add(getAppInfo(app));
  70. }
  71. //本来是系统程序,被用户手动更新后,该系统程序也成为第三方应用程序了
  72. elseif((app.flags&ApplicationInfo.FLAG_UPDATED_SYSTEM_APP)!=0){
  73. appInfos.add(getAppInfo(app));
  74. }
  75. }
  76. break;
  77. caseFILTER_SDCARD_APP://安装在SDCard的应用程序
  78. appInfos.clear();
  79. for(ApplicationInfoapp:listAppcations){
  80. if((app.flags&ApplicationInfo.FLAG_EXTERNAL_STORAGE)!=0){
  81. appInfos.add(getAppInfo(app));
  82. }
  83. }
  84. returnappInfos;
  85. default:
  86. returnnull;
  87. }
  88. returnappInfos;
  89. }
  90. //构造一个AppInfo对象,并赋值
  91. privateAppInfogetAppInfo(ApplicationInfoapp){
  92. AppInfoappInfo=newAppInfo();
  93. appInfo.setAppLabel((String)app.loadLabel(pm));
  94. appInfo.setAppIcon(app.loadIcon(pm));
  95. appInfo.setPkgName(app.packageName);
  96. returnappInfo;
  97. }
  98. }



你可以在此基础上,构建更多丰富的应用。比说说Settings模块中的卸载安装应用程序等。


本部分的内容是如何获取安装包得大小,包括缓存大小(cachesize)、数据大小(datasize)、应用程序大小(codesize)。

本部分的知识点涉及到AIDL、Java反射机制。理解起来也不是很难。

关于安装包得大小信息封装在PackageStats类中,该类很简单,只有几个字段:

PackageStats类:

常用字段:

public long cachesize 缓存大小

public long codesize 应用程序大小

public long datasize 数据大小

public String packageName 包名

PS:应用程序的总大小 = cachesize + codesize + datasize

也就是说只要获得了安装包所对应的PackageStats对象,就可以获得信息了。但是在AndroidSDK中并没有显示提供方法来

获得该对象,是不是很苦恼呢?但是,我们可以通过放射机制来调用系统中隐藏的函数(@hide)来获得每个安装包得信息。

具体方法如下:

第一步、 通过放射机制调用getPackageSizeInfo() 方法原型为:

view plain copy to clipboard print ?
  1. /*@parampackageName应用程序包名
  2. *@paramobserver当查询包得信息大小操作完成后,将回调给IPackageStatsObserver类中的onGetStatsCompleted()方法,
  3. *,并且我们需要的PackageStats对象也封装在其参数里.
  4. *@hide//隐藏函数的标记
  5. */
  6. publicabstractvoidgetPackageSizeInfo(StringpackageName,IPackageStatsObserverobserver);{
  7. //
  8. }

/*@param packageName 应用程序包名 *@param observer 当查询包得信息大小操作完成后,将回调给IPackageStatsObserver类中的onGetStatsCompleted()方法, * ,并且我们需要的PackageStats对象也封装在其参数里. * @hide //隐藏函数的标记 */ public abstract void getPackageSizeInfo(String packageName,IPackageStatsObserver observer);{ // }

内部调用流程如下,这个知识点较为复杂,知道即可,

getPackageSizeInfo方法内部调用getPackageSizeInfoLI(packageName, pStats)方法来完成包状态获取。

getPackageSizeInfoLI方法内部调用Installer.getSizeInfo(String pkgName, String apkPath,String fwdLockApkPath, PackageStats

pStats),继而将包状态信息返回给参数pStats。getSizeInfo这个方法内部是以本机Socket方式连接到Server,

然后向server发送一个文本字符串命令,格式:getsize apkPath fwdLockApkPath 给server。Server将结果返回,并解析到pStats

中。掌握这个调用知识链即可。

第二步、 由于需要获得系统级的服务或类,我们必须加入Android系统形成的AIDL文件,共两个:

IPackageStatsObserver.aidl 和 PackageStats.aidl文件。并将其放置在android.pm.content包路径下。

IPackageStatsObserver.aidl 文件

view plain copy to clipboard print ?
  1. packageandroid.content.pm;
  2. importandroid.content.pm.PackageStats;
  3. /**
  4. *APIforpackagedatachangerelatedcallbacksfromthePackageManager.
  5. *Someusagescenariosincludedeletionofcachedirectory,generate
  6. *statisticsrelatedtocode,data,cacheusage(TODO)
  7. *{@hide}
  8. */
  9. onewayinterfaceIPackageStatsObserver{
  10. voidonGetStatsCompleted(inPackageStatspStats,booleansucceeded);
  11. }

package android.content.pm; import android.content.pm.PackageStats; /** * API for package data change related callbacks from the Package Manager. * Some usage scenarios include deletion of cache directory, generate * statistics related to code, data, cache usage(TODO) * {@hide} */ oneway interface IPackageStatsObserver { void onGetStatsCompleted(in PackageStats pStats, boolean succeeded); }

PackageStats.aidl文件

view plain copy to clipboard print ?
  1. packageandroid.content.pm;
  2. parcelablePackageStats;

package android.content.pm; parcelable PackageStats;

第三步、 创建一个类继承至IPackageStatsObserver.Stub (桩,)它本质上实现了Binder机制。当我们把该类的一个实例通过getPackageSizeInfo()调用时,并该函数继而启动了启动中间流程去获取相关包得信息大小,当扫描完成后,最后将查询信息回调至该类的onGetStatsCompleted(in PackageStats pStats, boolean succeeded)方法,信息大小封装在此实例上。例如:

view plain copy to clipboard print ?
  1. //aidl文件形成的Bindler机制服务类
  2. publicclassPkgSizeObserverextendsIPackageStatsObserver.Stub{
  3. /***回调函数,
  4. *@parampStatus,返回数据封装在PackageStats对象中
  5. *@paramsucceeded代表回调成功
  6. */
  7. @Override
  8. publicvoidonGetStatsCompleted(PackageStatspStats,booleansucceeded)
  9. throwsRemoteException{
  10. //TODOAuto-generatedmethodstub
  11. cachesize=pStats.cacheSize;//缓存大小
  12. datasize=pStats.codeSize;//数据大小
  13. codesize=pStats.codeSize;//应用程序大小
  14. }
  15. }

//aidl文件形成的Bindler机制服务类 public class PkgSizeObserver extends IPackageStatsObserver.Stub{ /*** 回调函数, * @param pStatus ,返回数据封装在PackageStats对象中 * @param succeeded 代表回调成功 */ @Override public void onGetStatsCompleted(PackageStats pStats, boolean succeeded) throws RemoteException { // TODO Auto-generated method stub cachesize = pStats.cacheSize ; //缓存大小 datasize = pStats.codeSize ; //数据大小 codesize = pStats.codeSize ; //应用程序大小 } }

第四步、 最后我们可以获取 pStats的属性,获得它们的属性值,通过调用系统函数Formatter.formateFileSize(long size)转换

对应的以kb/mb为计量单位的字符串。

很重要的一点:为了能够通过反射获取应用程序大小,我们必须加入以下权限,否则,会出现警告并且得不到实际值。

view plain copy to clipboard print ?
  1. <uses-permissionandroid:name="android.permission.GET_PACKAGE_SIZE"></uses-permission>

<uses-permission android:name="android.permission.GET_PACKAGE_SIZE"></uses-permission>


流程图如下:

Android中获取应用程序(包)的信息-----PackageManager的使用_第5张图片

Demo说明

在第一部分应用得基础上,我们添加了一个新功能,点击任何一个应用后后,弹出显示该应用的包信息大小的对话框。

截图如下:

工程图: 程序效果图:

Android中获取应用程序(包)的信息-----PackageManager的使用_第6张图片Android中获取应用程序(包)的信息-----PackageManager的使用_第7张图片

1、dialg_app_size.xml 文件

view plain copy to clipboard print ?
  1. <?xmlversion="1.0"encoding="utf-8"?>
  2. <LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"
  3. android:orientation="vertical"android:layout_width="wrap_content"
  4. android:layout_height="wrap_content">
  5. <LinearLayoutandroid:layout_width="wrap_content"
  6. android:layout_height="wrap_content"android:orientation="horizontal">
  7. <TextViewandroid:layout_width="100dip"
  8. android:layout_height="wrap_content"android:text="缓存大小:"></TextView>
  9. <TextViewandroid:layout_width="100dip"android:id="@+id/tvcachesize"
  10. android:layout_height="wrap_content"></TextView>
  11. </LinearLayout>
  12. <LinearLayoutandroid:layout_width="wrap_content"
  13. android:layout_height="wrap_content"android:orientation="horizontal">
  14. <TextViewandroid:layout_width="100dip"
  15. android:layout_height="wrap_content"android:text="数据大小:"></TextView>
  16. <TextViewandroid:layout_width="100dip"android:id="@+id/tvdatasize"
  17. android:layout_height="wrap_content"></TextView>
  18. </LinearLayout>
  19. <LinearLayoutandroid:layout_width="wrap_content"
  20. android:layout_height="wrap_content"android:orientation="horizontal">
  21. <TextViewandroid:layout_width="100dip"
  22. android:layout_height="wrap_content"android:text="应用程序大小:"></TextView>
  23. <TextViewandroid:layout_width="100dip"android:id="@+id/tvcodesize"
  24. android:layout_height="wrap_content"></TextView>
  25. </LinearLayout>
  26. <LinearLayoutandroid:layout_width="wrap_content"
  27. android:layout_height="wrap_content"android:orientation="horizontal">
  28. <TextViewandroid:layout_width="100dip"
  29. android:layout_height="wrap_content"android:text="总大小:"></TextView>
  30. <TextViewandroid:layout_width="100dip"android:id="@+id/tvtotalsize"
  31. android:layout_height="wrap_content"></TextView>
  32. </LinearLayout>
  33. </LinearLayout>

<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="wrap_content" android:layout_height="wrap_content"> <LinearLayout android:layout_width="wrap_content" android:layout_height="wrap_content" android:orientation="horizontal"> <TextView android:layout_width="100dip" android:layout_height="wrap_content" android:text="缓存大小:"></TextView> <TextView android:layout_width="100dip" android:id="@+id/tvcachesize" android:layout_height="wrap_content"></TextView> </LinearLayout> <LinearLayout android:layout_width="wrap_content" android:layout_height="wrap_content" android:orientation="horizontal"> <TextView android:layout_width="100dip" android:layout_height="wrap_content" android:text="数据大小:"></TextView> <TextView android:layout_width="100dip" android:id="@+id/tvdatasize" android:layout_height="wrap_content"></TextView> </LinearLayout> <LinearLayout android:layout_width="wrap_content" android:layout_height="wrap_content" android:orientation="horizontal"> <TextView android:layout_width="100dip" android:layout_height="wrap_content" android:text="应用程序大小:"></TextView> <TextView android:layout_width="100dip" android:id="@+id/tvcodesize" android:layout_height="wrap_content"></TextView> </LinearLayout> <LinearLayout android:layout_width="wrap_content" android:layout_height="wrap_content" android:orientation="horizontal"> <TextView android:layout_width="100dip" android:layout_height="wrap_content" android:text="总大小:"></TextView> <TextView android:layout_width="100dip" android:id="@+id/tvtotalsize" android:layout_height="wrap_content"></TextView> </LinearLayout> </LinearLayout>

2、另外的资源文件或自定义适配器复用了第一部分,请知悉。

3、添加AIDL文件,如上。

4、主文件MainActivity.java如下:

view plain copy to clipboard print ?
  1. packagecom.qin.appsize;
  2. importjava.lang.reflect.Method;
  3. importjava.util.ArrayList;
  4. importjava.util.Collections;
  5. importjava.util.List;
  6. importcom.qin.appsize.AppInfo;
  7. importandroid.app.Activity;
  8. importandroid.app.AlertDialog;
  9. importandroid.content.ComponentName;
  10. importandroid.content.Context;
  11. importandroid.content.DialogInterface;
  12. importandroid.content.Intent;
  13. importandroid.content.pm.IPackageStatsObserver;
  14. importandroid.content.pm.PackageManager;
  15. importandroid.content.pm.PackageStats;
  16. importandroid.content.pm.ResolveInfo;
  17. importandroid.graphics.drawable.Drawable;
  18. importandroid.os.Bundle;
  19. importandroid.os.RemoteException;
  20. importandroid.text.format.Formatter;
  21. importandroid.util.Log;
  22. importandroid.view.LayoutInflater;
  23. importandroid.view.View;
  24. importandroid.widget.AdapterView;
  25. importandroid.widget.ListView;
  26. importandroid.widget.TextView;
  27. importandroid.widget.AdapterView.OnItemClickListener;
  28. publicclassMainActivityextendsActivityimplementsOnItemClickListener{
  29. privatestaticStringTAG="APP_SIZE";
  30. privateListViewlistview=null;
  31. privateList<AppInfo>mlistAppInfo=null;
  32. LayoutInflaterinfater=null;
  33. //全局变量,保存当前查询包得信息
  34. privatelongcachesize;//缓存大小
  35. privatelongdatasize;//数据大小
  36. privatelongcodesize;//应用程序大小
  37. privatelongtotalsize;//总大小
  38. @Override
  39. publicvoidonCreate(BundlesavedInstanceState){
  40. super.onCreate(savedInstanceState);
  41. setContentView(R.layout.browse_app_list);
  42. listview=(ListView)findViewById(R.id.listviewApp);
  43. mlistAppInfo=newArrayList<AppInfo>();
  44. queryAppInfo();//查询所有应用程序信息
  45. BrowseApplicationInfoAdapterbrowseAppAdapter=newBrowseApplicationInfoAdapter(
  46. this,mlistAppInfo);
  47. listview.setAdapter(browseAppAdapter);
  48. listview.setOnItemClickListener(this);
  49. }
  50. //点击弹出对话框,显示该包得大小
  51. publicvoidonItemClick(AdapterView<?>arg0,Viewview,intposition,longarg3){
  52. //更新显示当前包得大小信息
  53. queryPacakgeSize(mlistAppInfo.get(position).getPkgName());
  54. infater=(LayoutInflater)MainActivity.this.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
  55. Viewdialog=infater.inflate(R.layout.dialog_app_size,null);
  56. TextViewtvcachesize=(TextView)dialog.findViewById(R.id.tvcachesize);//缓存大小
  57. TextViewtvdatasize=(TextView)dialog.findViewById(R.id.tvdatasize);//数据大小
  58. TextViewtvcodesize=(TextView)dialog.findViewById(R.id.tvcodesize);//应用程序大小
  59. TextViewtvtotalsize=(TextView)dialog.findViewById(R.id.tvtotalsize);//总大小
  60. //类型转换并赋值
  61. tvcachesize.setText(formateFileSize(cachesize));
  62. tvdatasize.setText(formateFileSize(datasize));
  63. tvcodesize.setText(formateFileSize(codesize));
  64. tvtotalsize.setText(formateFileSize(totalsize));
  65. //显示自定义对话框
  66. AlertDialog.Builderbuilder=newAlertDialog.Builder(MainActivity.this);
  67. builder.setView(dialog);
  68. builder.setTitle(mlistAppInfo.get(position).getAppLabel()+"的大小信息为:");
  69. builder.setPositiveButton("确定",newDialogInterface.OnClickListener(){
  70. @Override
  71. publicvoidonClick(DialogInterfacedialog,intwhich){
  72. //TODOAuto-generatedmethodstub
  73. dialog.cancel();//取消显示对话框
  74. }
  75. });
  76. builder.create().show();
  77. }
  78. publicvoidqueryPacakgeSize(StringpkgName)throwsException{
  79. if(pkgName!=null){
  80. //使用放射机制得到PackageManager类的隐藏函数getPackageSizeInfo
  81. PackageManagerpm=getPackageManager();//得到pm对象
  82. try{
  83. //通过反射机制获得该隐藏函数
  84. MethodgetPackageSizeInfo=pm.getClass().getDeclaredMethod("getPackageSizeInfo",String.class,IPackageStatsObserver.class);
  85. //调用该函数,并且给其分配参数,待调用流程完成后会回调PkgSizeObserver类的函数
  86. getPackageSizeInfo.invoke(pm,pkgName,newPkgSizeObserver());
  87. }
  88. catch(Exceptionex){
  89. Log.e(TAG,"NoSuchMethodException");
  90. ex.printStackTrace();
  91. throwex;//抛出异常
  92. }
  93. }
  94. }
  95. //aidl文件形成的Bindler机制服务类
  96. publicclassPkgSizeObserverextendsIPackageStatsObserver.Stub{
  97. /***回调函数,
  98. *@parampStatus,返回数据封装在PackageStats对象中
  99. *@paramsucceeded代表回调成功
  100. */
  101. @Override
  102. publicvoidonGetStatsCompleted(PackageStatspStats,booleansucceeded)
  103. throwsRemoteException{
  104. //TODOAuto-generatedmethodstub
  105. cachesize=pStats.cacheSize;//缓存大小
  106. datasize=pStats.dataSize;//数据大小
  107. codesize=pStats.codeSize;//应用程序大小
  108. totalsize=cachesize+datasize+codesize;
  109. Log.i(TAG,"cachesize--->"+cachesize+"datasize---->"+datasize+"codeSize---->"+codesize);
  110. }
  111. }
  112. //系统函数,字符串转换long-String(kb)
  113. privateStringformateFileSize(longsize){
  114. returnFormatter.formatFileSize(MainActivity.this,size);
  115. }
  116. //获得所有启动Activity的信息,类似于Launch界面
  117. publicvoidqueryAppInfo(){
  118. PackageManagerpm=this.getPackageManager();//获得PackageManager对象
  119. IntentmainIntent=newIntent(Intent.ACTION_MAIN,null);
  120. mainIntent.addCategory(Intent.CATEGORY_LAUNCHER);
  121. //通过查询,获得所有ResolveInfo对象.
  122. List<ResolveInfo>resolveInfos=pm.queryIntentActivities(mainIntent,0);
  123. //调用系统排序,根据name排序
  124. //该排序很重要,否则只能显示系统应用,而不能列出第三方应用程序
  125. Collections.sort(resolveInfos,newResolveInfo.DisplayNameComparator(pm));
  126. if(mlistAppInfo!=null){
  127. mlistAppInfo.clear();
  128. for(ResolveInforeInfo:resolveInfos){
  129. StringactivityName=reInfo.activityInfo.name;//获得该应用程序的启动Activity的name
  130. StringpkgName=reInfo.activityInfo.packageName;//获得应用程序的包名
  131. StringappLabel=(String)reInfo.loadLabel(pm);//获得应用程序的Label
  132. Drawableicon=reInfo.loadIcon(pm);//获得应用程序图标
  133. //为应用程序的启动Activity准备Intent
  134. IntentlaunchIntent=newIntent();
  135. launchIntent.setComponent(newComponentName(pkgName,activityName));
  136. //创建一个AppInfo对象,并赋值
  137. AppInfoappInfo=newAppInfo();
  138. appInfo.setAppLabel(appLabel);
  139. appInfo.setPkgName(pkgName);
  140. appInfo.setAppIcon(icon);
  141. appInfo.setIntent(launchIntent);
  142. mlistAppInfo.add(appInfo);//添加至列表中
  143. }
  144. }
  145. }
  146. }

package com.qin.appsize; import java.lang.reflect.Method; import java.util.ArrayList; import java.util.Collections; import java.util.List; import com.qin.appsize.AppInfo; import android.app.Activity; import android.app.AlertDialog; import android.content.ComponentName; import android.content.Context; import android.content.DialogInterface; import android.content.Intent; import android.content.pm.IPackageStatsObserver; import android.content.pm.PackageManager; import android.content.pm.PackageStats; import android.content.pm.ResolveInfo; import android.graphics.drawable.Drawable; import android.os.Bundle; import android.os.RemoteException; import android.text.format.Formatter; import android.util.Log; import android.view.LayoutInflater; import android.view.View; import android.widget.AdapterView; import android.widget.ListView; import android.widget.TextView; import android.widget.AdapterView.OnItemClickListener; public class MainActivity extends Activity implements OnItemClickListener{ private static String TAG = "APP_SIZE"; private ListView listview = null; private List<AppInfo> mlistAppInfo = null; LayoutInflater infater = null ; //全局变量,保存当前查询包得信息 private long cachesize ; //缓存大小 private long datasize ; //数据大小 private long codesize ; //应用程序大小 private long totalsize ; //总大小 @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.browse_app_list); listview = (ListView) findViewById(R.id.listviewApp); mlistAppInfo = new ArrayList<AppInfo>(); queryAppInfo(); // 查询所有应用程序信息 BrowseApplicationInfoAdapter browseAppAdapter = new BrowseApplicationInfoAdapter( this, mlistAppInfo); listview.setAdapter(browseAppAdapter); listview.setOnItemClickListener(this); } // 点击弹出对话框,显示该包得大小 public void onItemClick(AdapterView<?> arg0, View view, int position,long arg3) { //更新显示当前包得大小信息 queryPacakgeSize(mlistAppInfo.get(position).getPkgName()); infater = (LayoutInflater) MainActivity.this.getSystemService(Context.LAYOUT_INFLATER_SERVICE); View dialog = infater.inflate(R.layout.dialog_app_size, null) ; TextView tvcachesize =(TextView) dialog.findViewById(R.id.tvcachesize) ; //缓存大小 TextView tvdatasize = (TextView) dialog.findViewById(R.id.tvdatasize) ; //数据大小 TextView tvcodesize = (TextView) dialog.findViewById(R.id.tvcodesize) ; // 应用程序大小 TextView tvtotalsize = (TextView) dialog.findViewById(R.id.tvtotalsize) ; //总大小 //类型转换并赋值 tvcachesize.setText(formateFileSize(cachesize)); tvdatasize.setText(formateFileSize(datasize)) ; tvcodesize.setText(formateFileSize(codesize)) ; tvtotalsize.setText(formateFileSize(totalsize)) ; //显示自定义对话框 AlertDialog.Builder builder =new AlertDialog.Builder(MainActivity.this) ; builder.setView(dialog) ; builder.setTitle(mlistAppInfo.get(position).getAppLabel()+"的大小信息为:") ; builder.setPositiveButton("确定", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { // TODO Auto-generated method stub dialog.cancel() ; // 取消显示对话框 } }); builder.create().show() ; } public void queryPacakgeSize(String pkgName) throws Exception{ if ( pkgName != null){ //使用放射机制得到PackageManager类的隐藏函数getPackageSizeInfo PackageManager pm = getPackageManager(); //得到pm对象 try { //通过反射机制获得该隐藏函数 Method getPackageSizeInfo = pm.getClass().getDeclaredMethod("getPackageSizeInfo", String.class,IPackageStatsObserver.class); //调用该函数,并且给其分配参数 ,待调用流程完成后会回调PkgSizeObserver类的函数 getPackageSizeInfo.invoke(pm, pkgName,new PkgSizeObserver()); } catch(Exception ex){ Log.e(TAG, "NoSuchMethodException") ; ex.printStackTrace() ; throw ex ; // 抛出异常 } } } //aidl文件形成的Bindler机制服务类 public class PkgSizeObserver extends IPackageStatsObserver.Stub{ /*** 回调函数, * @param pStatus ,返回数据封装在PackageStats对象中 * @param succeeded 代表回调成功 */ @Override public void onGetStatsCompleted(PackageStats pStats, boolean succeeded) throws RemoteException { // TODO Auto-generated method stub cachesize = pStats.cacheSize ; //缓存大小 datasize = pStats.dataSize ; //数据大小 codesize = pStats.codeSize ; //应用程序大小 totalsize = cachesize + datasize + codesize ; Log.i(TAG, "cachesize--->"+cachesize+" datasize---->"+datasize+ " codeSize---->"+codesize) ; } } //系统函数,字符串转换 long -String (kb) private String formateFileSize(long size){ return Formatter.formatFileSize(MainActivity.this, size); } // 获得所有启动Activity的信息,类似于Launch界面 public void queryAppInfo() { PackageManager pm = this.getPackageManager(); // 获得PackageManager对象 Intent mainIntent = new Intent(Intent.ACTION_MAIN, null); mainIntent.addCategory(Intent.CATEGORY_LAUNCHER); // 通过查询,获得所有ResolveInfo对象. List<ResolveInfo> resolveInfos = pm.queryIntentActivities(mainIntent, 0); // 调用系统排序 , 根据name排序 // 该排序很重要,否则只能显示系统应用,而不能列出第三方应用程序 Collections.sort(resolveInfos,new ResolveInfo.DisplayNameComparator(pm)); if (mlistAppInfo != null) { mlistAppInfo.clear(); for (ResolveInfo reInfo : resolveInfos) { String activityName = reInfo.activityInfo.name; // 获得该应用程序的启动Activity的name String pkgName = reInfo.activityInfo.packageName; // 获得应用程序的包名 String appLabel = (String) reInfo.loadLabel(pm); // 获得应用程序的Label Drawable icon = reInfo.loadIcon(pm); // 获得应用程序图标 // 为应用程序的启动Activity 准备Intent Intent launchIntent = new Intent(); launchIntent.setComponent(new ComponentName(pkgName,activityName)); // 创建一个AppInfo对象,并赋值 AppInfo appInfo = new AppInfo(); appInfo.setAppLabel(appLabel); appInfo.setPkgName(pkgName); appInfo.setAppIcon(icon); appInfo.setIntent(launchIntent); mlistAppInfo.add(appInfo); // 添加至列表中 } } } }


获取应用程序信息大小就是这么来的,整个过程相对而言还是挺简单的,比较难理解的是AIDL文件的使用和回调函数的处理。

仔细研究后,才有所理解



更多相关文章

  1. 开发 Android 应用程序 常见问题 FAQ
  2. Android:销毁所有的Activity退出应用程序几种方式
  3. android项目打包成apk应用程序后部署到虚拟机上测试
  4. Android应用程序获得root权限
  5. android app -- Picasso 二级缓存加载图片,可控制加载图片大小(附
  6. 第三部分:Android 应用程序接口指南---第四节:动画和图形---第五章
  7. Android开发之应用程序基础及组件

随机推荐

  1. 使用Android保护RESTful服务的最佳方式
  2. latch(用于等待异步响应)冻结WebView(和UI)
  3. Android自定义View流程
  4. android datepickerdialog 怎样在第一次
  5. 【android】两个view共用同一个animation
  6. Android异步双屏显示
  7. 【android】【Bootanimation】Bootanimat
  8. android基础画板的实现(surfaceView)
  9. 使用android的加速度计移动图像
  10. android.os.NetworkOnMainThreadExceptio