TShopping

 找回密碼
 註冊
搜索
查看: 371|回復: 0

[教學] Android中獲取應用程序(包)的信息-----PackageManager的使用(一)

[複製鏈接]
發表於 2014-11-29 00:32:04 | 顯示全部樓層 |閱讀模式
 
Push to Facebook Push to Plurk Push to Twitter 

本節內容是如何獲取Android系統中應用程序的信息,主要包括packagename、label、icon、佔用大小等。具體分為兩個

部分,計劃如下:

            第一部分:獲取應用程序的packagename、label、icon等;

            第二部分:獲取應用程序的佔用大小,包括:緩存大小(cachsize)、數據大小(datasize)。

       每部分都為您準備了簡單豐富的實例,您一定不會錯過。

      Android系統為我們提供了很多服務管理的類,包括ActivityManager、PowerManager(電源管理)、AudioManager(音頻管理)

等。除此之外,還提供了一個PackageManger管理類,它的主要職責是管理應用程序包。通過它,我們就可以獲取應用程序信息。

     引入: AnroidManifest.xml文件節點說明:

      1.gif

一、相關類的介紹
    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屬性)

               public String 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對象,即< service >節點信息

       常用方法

             Drawable loadIcon(PackageManager pm)              獲得當前應用程序的圖像

             CharSequence loadLabel(PackageManager pm)   獲得當前應用程序的label

PackageInfo類

       說明:手動獲取AndroidManifest.xml文件的信息。

       常用字段

           public String    packageName                    包名

           public ActivityInfo[] activities 所有<activity>節點信息

           public ApplicationInfo applicationInfo <application>節點信息,只有一個

           public ActivityInfo[]   receivers  所有<receiver>節點信息,多個

           public ServiceInfo[] services 所有<service>節點信息,多個

PackageManger 類

      說明:獲得已安裝的應用程序信息。可以通過getPackageManager()方法獲得。

      常用方法

          public abstract PackageManager getPackageManager()   

               功能:獲得一個PackageManger對象

         public abstract Drawable getApplicationIcon( String packageName)

               參數: packageName 包名

               功能:返回給定包名的圖標,否則返回null

       public abstract ApplicationInfo   getApplicationInfo ( StringpackageName, int flags)

               參數:packagename 包名

                          flags該ApplicationInfo是此flags標記,通常可以直接賦予常數0即可

               功能:返回該ApplicationInfo對象

          public abstract List <ApplicationInfo> getInstalledApplications(int flags)

               參數:flag為一般為GET_UNINSTALLED_PACKAGES,那麼此時會返回所有ApplicationInfo。我們可以對ApplicationInfo的flags過濾,得到我們需要的。

               功能:返回給定條件的所有PackageInfo

         public abstract List < PackageInfo > getInstalledPackages(int flags)

               參數如上

               功能:返回給定條件的所有PackageInfo

       public abstract ResolveInfo   resolveActivity( Intent intent, 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 abstract   List < ResolveInfo > queryIntentActivities( Intent intent, int flags)

            參數同上

            功能:返回給定條件的所有ResolveInfo對象(本質上是Activity),集合對象

      public abstract ResolveInfo   resolveService( Intent intent, int flags)

           參數同上

           功能:返回給定條件的ResolveInfo對象(本質上是Service)

     public abstractList < ResolveInfo >  queryIntentServices ( Intent intent, 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 :

          圖:

2.gif

        1 、佈局文件: 主要有兩個:帶listview的browse_app_list.xml文件;listview的項browse_app_item.xml

browse_app_list.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3.         android:orientation="vertical" android:layout_width="fill_parent"
  4.         android:layout_height="fill_parent">
  5.         <ListView android:id="@+id/listviewApp" android:layout_width="fill_parent"
  6.                 android:layout_height="fill_parent" ></ListView>
  7. </LinearLayout>
複製代碼

        browse_app_item.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3.         android:layout_width="fill_parent" android:layout_height="50dip">

  4.         <ImageView android:id="@+id/imgApp" android:layout_width="wrap_content"
  5.                 android:layout_height="fill_parent" ></ImageView>
  6.         <RelativeLayout android:layout_width="fill_parent"  android:layout_marginLeft="10dip"
  7.                 android:layout_height="40dip">
  8.                 <TextView android:id="@+id/tvLabel" android:layout_width="wrap_content"
  9.                         android:layout_height="wrap_content" android:text="AppLable : "></TextView>
  10.                 <TextView android: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.                 <TextView android:id="@+id/tvName" android:layout_width="wrap_content"
  14.                         android:layout_height="wrap_content" android:layout_below="@id/tvLabel"
  15.                         android:text="包名:"></TextView>
  16.                 <TextView android: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類

  1. //Model类 ,用来存储应用程序信息
  2. public class AppInfo {
  3.         private String appLabel;    //应用程序标签
  4.         private Drawable appIcon ;  //应用程序图像
  5.         private Intent intent ;     //启动应用程序的Intent ,一般是Action为Main和Category为Lancher的Activity
  6.         private String pkgName ;    //应用程序所对应的包名
  7.         
  8.         public AppInfo(){}
  9.    
  10.         public String getAppLabel() {
  11.                 return appLabel;
  12.         }
  13.         public void setAppLabel(String appName) {
  14.                 this.appLabel = appName;
  15.         }
  16.         public Drawable getAppIcon() {
  17.                 return appIcon;
  18.         }
  19.         public void setAppIcon(Drawable appIcon) {
  20.                 this.appIcon = appIcon;
  21.         }
  22.         public Intent getIntent() {
  23.                 return intent;
  24.         }
  25.         public void setIntent(Intent intent) {
  26.                 this.intent = intent;
  27.         }
  28.         public String getPkgName(){
  29.                 return pkgName ;
  30.         }
  31.         public void setPkgName(String pkgName){
  32.                 this.pkgName=pkgName ;
  33.         }
  34. }
複製代碼

          3、 BrowseApplicationInfoAdapter.java : 自定義適配器類,為ListView提供視圖

  1. //自定义适配器类,提供给listView的自定义view
  2. public class BrowseApplicationInfoAdapter extends BaseAdapter {
  3.         private List<AppInfo> mlistAppInfo = null;
  4.         LayoutInflater infater = null;
  5.         public BrowseApplicationInfoAdapter(Context context,  List<AppInfo> apps) {
  6.                 infater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
  7.                 mlistAppInfo = apps ;
  8.         }
  9.         @Override
  10.         public int getCount() {
  11.                 // TODO Auto-generated method stub
  12.                 System.out.println("size" + mlistAppInfo.size());
  13.                 return mlistAppInfo.size();
  14.         }
  15.         @Override
  16.         public Object getItem(int position) {
  17.                 // TODO Auto-generated method stub
  18.                 return mlistAppInfo.get(position);
  19.         }
  20.         @Override
  21.         public long getItemId(int position) {
  22.                 // TODO Auto-generated method stub
  23.                 return 0;
  24.         }
  25.         @Override
  26.         public View getView(int position, View convertview, ViewGroup arg2) {
  27.                 System.out.println("getView at " + position);
  28.                 View view = null;
  29.                 ViewHolder holder = null;
  30.                 if (convertview == null || convertview.getTag() == null) {
  31.                         view = infater.inflate(R.layout.browse_app_item, null);
  32.                         holder = new ViewHolder(view);
  33.                         view.setTag(holder);
  34.                 }
  35.                 else{
  36.                         view = convertview ;
  37.                         holder = (ViewHolder) convertview.getTag() ;
  38.                 }
  39.                 AppInfo appInfo = (AppInfo) getItem(position);
  40.                 holder.appIcon.setImageDrawable(appInfo.getAppIcon());
  41.                 holder.tvAppLabel.setText(appInfo.getAppLabel());
  42.                 holder.tvPkgName.setText(appInfo.getPkgName());
  43.                 return view;
  44.         }

  45.         class ViewHolder {
  46.                 ImageView appIcon;
  47.                 TextView tvAppLabel;
  48.                 TextView tvPkgName;
  49.                 public ViewHolder(View view) {
  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()方法,並且注意到排序,它很重要。

  1. public class MainActivity extends Activity implements OnItemClickListener {
  2.         private ListView listview = null;
  3.         private List<AppInfo> mlistAppInfo = null;
  4.         @Override
  5.         public void onCreate(Bundle savedInstanceState) {
  6.                 super.onCreate(savedInstanceState);
  7.                 setContentView(R.layout.browse_app_list);
  8.                 listview = (ListView) findViewById(R.id.listviewApp);
  9.                 mlistAppInfo = new ArrayList<AppInfo>();
  10.                 queryAppInfo(); // 查询所有应用程序信息
  11.                 BrowseApplicationInfoAdapter browseAppAdapter = new BrowseApplicationInfoAdapter(this, mlistAppInfo);
  12.                 listview.setAdapter(browseAppAdapter);
  13.                 listview.setOnItemClickListener(this);
  14.         }
  15.         // 点击跳转至该应用程序
  16.         public void onItemClick(AdapterView<?> arg0, View view, int position,long arg3) {
  17.                 // TODO Auto-generated method stub
  18.                 Intent intent = mlistAppInfo.get(position).getIntent();
  19.                 startActivity(intent);
  20.         }
  21.         // 获得所有启动Activity的信息,类似于Launch界面
  22.         public void queryAppInfo() {
  23.                 PackageManager pm = this.getPackageManager(); // 获得PackageManager对象
  24.                 Intent mainIntent = new Intent(Intent.ACTION_MAIN, null);
  25.                 mainIntent.addCategory(Intent.CATEGORY_LAUNCHER);
  26.                 // 通过查询,获得所有ResolveInfo对象.
  27.                 List<ResolveInfo> resolveInfos = pm.queryIntentActivities(mainIntent, PackageManager.MATCH_DEFAULT_ONLY);
  28.                 // 调用系统排序,根据name排序
  29.                 // 该排序很重要,否则只能显示系统应用,而不能列出第三方应用程序
  30.                 Collections.sort(resolveInfos,new ResolveInfo.DisplayNameComparator(pm));
  31.                 if (mlistAppInfo != null) {
  32.                         mlistAppInfo.clear();
  33.                         for (ResolveInfo reInfo : resolveInfos) {
  34.                                 String activityName = reInfo.activityInfo.name; // 获得该应用程序的启动Activity的name
  35.                                 String pkgName = reInfo.activityInfo.packageName; // 获得应用程序的包名
  36.                                 String appLabel = (String) reInfo.loadLabel(pm); // 获得应用程序的Label
  37.                                 Drawable icon = reInfo.loadIcon(pm); // 获得应用程序图标
  38.                                 // 为应用程序的启动Activity 准备Intent
  39.                                 Intent launchIntent = new Intent();
  40.                                 launchIntent.setComponent(new ComponentName(pkgName,activityName));
  41.                                 // 创建一个AppInfo对象,并赋值
  42.                                 AppInfo appInfo = new AppInfo();
  43.                                 appInfo.setAppLabel(appLabel);
  44.                                 appInfo.setPkgName(pkgName);
  45.                                 appInfo.setAppIcon(icon);
  46.                                 appInfo.setIntent(launchIntent);
  47.                                 mlistAppInfo.add(appInfo); // 添加至列表中
  48.                                 System.out.println(appLabel + " activityName---" + activityName + " pkgName---" + pkgName);
  49.                         }
  50.                 }
  51.         }
  52. }
複製代碼

        好了,第一個Demo完成。。


Demo 2:

        demo2在佈局、適配器方面和Demo1一樣。只是利用了getInstalledApplications ()方法,繼而通過ApplicationInfo.flags來挑選我們希望的ApplicationInfo對象。

       圖:

               3.gif           4.gif

過濾應用程序如下:

  1. package com.qiner.appinfo;

  2. import java.util.ArrayList;
  3. import java.util.Collections;
  4. import java.util.List;
  5. import com.qiner.appinfo.R;
  6. import android.app.Activity;
  7. import android.app.Application;
  8. import android.content.pm.ApplicationInfo;
  9. import android.content.pm.PackageManager;
  10. import android.os.Bundle;
  11. import android.view.View;
  12. import android.view.View.OnClickListener;
  13. import android.widget.Button;
  14. import android.widget.ListView;

  15. public class MainActivity extends Activity  {
  16.         public static final int FILTER_ALL_APP = 0; // 所有应用程序
  17.         public static final int FILTER_SYSTEM_APP = 1; // 系统程序
  18.         public static final int FILTER_THIRD_APP = 2; // 第三方应用程序
  19.         public static final int FILTER_SDCARD_APP = 3; // 安装在SDCard的应用程序
  20.         private ListView listview = null;
  21.         private PackageManager pm;
  22.         private int filter = FILTER_ALL_APP;
  23.         private List<AppInfo> mlistAppInfo ;
  24.         private BrowseApplicationInfoAdapter browseAppAdapter = null ;
  25.         /** Called when the activity is first created. */
  26.         @Override
  27.         public void onCreate(Bundle savedInstanceState) {
  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 = new BrowseApplicationInfoAdapter(this, mlistAppInfo);
  37.                 listview.setAdapter(browseAppAdapter);
  38.         }
  39.         // 根据查询条件,查询特定的ApplicationInfo
  40.         private List<AppInfo> queryFilterAppInfo(int filter) {
  41.                 pm = this.getPackageManager();
  42.                 // 查询所有已经安装的应用程序
  43.                 List<ApplicationInfo> listAppcations = pm.getInstalledApplications(PackageManager.GET_UNINSTALLED_PACKAGES);
  44.                 Collections.sort(listAppcations,new ApplicationInfo.DisplayNameComparator(pm));// 排序
  45.                 List<AppInfo> appInfos = new ArrayList<AppInfo>(); // 保存过滤查到的AppInfo
  46.                 // 根据条件来过滤
  47.                 switch (filter) {
  48.                   case FILTER_ALL_APP: // 所有应用程序
  49.                         appInfos.clear();
  50.                         for (ApplicationInfo app : listAppcations) {
  51.                                 appInfos.add(getAppInfo(app));
  52.                         }
  53.                         return appInfos;
  54.                   case FILTER_SYSTEM_APP: // 系统程序
  55.                         appInfos.clear();
  56.                         for (ApplicationInfo app : listAppcations) {
  57.                                 if ((app.flags & ApplicationInfo.FLAG_SYSTEM) != 0) {
  58.                                         appInfos.add(getAppInfo(app));
  59.                                 }
  60.                         }
  61.                         return appInfos;
  62.                   case FILTER_THIRD_APP: // 第三方应用程序
  63.                         appInfos.clear();
  64.                         for (ApplicationInfo app : listAppcations) {
  65.                                 //非系统程序
  66.                                 if ((app.flags & ApplicationInfo.FLAG_SYSTEM) <= 0) {
  67.                                         appInfos.add(getAppInfo(app));
  68.                                 }
  69.                                 //本来是系统程序,被用户手动更新后,该系统程序也成为第三方应用程序了
  70.                                 else if ((app.flags & ApplicationInfo.FLAG_UPDATED_SYSTEM_APP) != 0){
  71.                                         appInfos.add(getAppInfo(app));
  72.                                 }
  73.                         }
  74.                         break;
  75.                   case FILTER_SDCARD_APP: // 安装在SDCard的应用程序
  76.                         appInfos.clear();
  77.                         for (ApplicationInfo app : listAppcations) {
  78.                                 if ((app.flags & ApplicationInfo.FLAG_EXTERNAL_STORAGE) != 0) {
  79.                                         appInfos.add(getAppInfo(app));
  80.                                 }
  81.                         }
  82.                         return appInfos;
  83.                   default:
  84.                         return null;
  85.                 }
  86.                 return appInfos;
  87.         }
  88.         // 构造一个AppInfo对象 ,并赋值
  89.         private AppInfo getAppInfo(ApplicationInfo app) {
  90.                 AppInfo appInfo = new AppInfo();
  91.                 appInfo.setAppLabel((String) app.loadLabel(pm));
  92.                 appInfo.setAppIcon(app.loadIcon(pm));
  93.                 appInfo.setPkgName(app.packageName);
  94.                 return appInfo;
  95.         }
  96. }
複製代碼

你可以在此基礎上,構建更多豐富的應用。比說說Settings模塊中的卸載安裝應用程序等。

本節的源代碼已上傳,下載地址:http://download.csdn.net/detail/qinjuning/3775869


轉帖:http://blog.csdn.net/qinjuning/article/details/6867806


 

臉書網友討論
您需要登錄後才可以回帖 登錄 | 註冊 |

本版積分規則



Archiver|手機版|小黑屋|免責聲明|TShopping

GMT+8, 2016-12-3 20:48 , Processed in 0.065165 second(s), 25 queries .

本論壇言論純屬發表者個人意見,與 TShopping綜合論壇 立場無關 如有意見侵犯了您的權益 請寫信聯絡我們。

Powered by Discuz! X3.2

© 2001-2013 Comsenz Inc.

快速回復 返回頂部 返回列表