TShopping

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

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

[複製鏈接]
發表於 2014-3-11 00:54:11 | 顯示全部樓層 |閱讀模式
 
Push to Facebook Push to Plurk Push to Twitter 
                                                                                                                            轉載請註明出處:http://blog.csdn.net/qinjuning

          本節內容是如何獲取Android系統中應用程序的信息,主要包括packagename、label、icon、佔用大小等。具體分為兩個
分,計劃如下:
            第一部分: 獲取應用程序的packagename、label、icon等 ;
            第二部分:獲取應用程序的佔用大小,包括:緩存大小(cachsize)、數據大小(datasize)。

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

      Android系統為我們提供了很多服務管理的類,包括ActivityManager、PowerManager(電源管理)、AudioManager(音頻管理)
等。除此之外,還提供了一個PackageManger管理類,它的主要職責是管理應用程序包。 通過它,我們就可以獲取應用程序信息。

     引入: AnroidManifest.xml文件節點說明:
     
一、相關類的介紹
    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 abstrac  tDrawable    getApplicationIcon(StringpackageName)
               參數: packageName 包名
               功能:返回給定包名的圖標,否則返回null

       public abstract ApplicationInfo   getApplicationInfo(String packageName, 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 abstractResolveInfo  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 abstract List<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  :
         圖:
               
  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.xmlbrowse_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.   
  4.         private String appLabel;    //应用程序标签
  5.         private Drawable appIcon ;  //应用程序图像
  6.         private Intent intent ;     //启动应用程序的Intent ,一般是Action为Main和Category为Lancher的Activity
  7.         private String pkgName ;    //应用程序所对应的包名
  8.         
  9.         public AppInfo(){}
  10.         
  11.         public String getAppLabel() {
  12.                 return appLabel;
  13.         }
  14.         public void setAppLabel(String appName) {
  15.                 this.appLabel = appName;
  16.         }
  17.         public Drawable getAppIcon() {
  18.                 return appIcon;
  19.         }
  20.         public void setAppIcon(Drawable appIcon) {
  21.                 this.appIcon = appIcon;
  22.         }
  23.         public Intent getIntent() {
  24.                 return intent;
  25.         }
  26.         public void setIntent(Intent intent) {
  27.                 this.intent = intent;
  28.         }
  29.         public String getPkgName(){
  30.                 return pkgName ;
  31.         }
  32.         public void setPkgName(String pkgName){
  33.                 this.pkgName=pkgName ;
  34.         }
  35. }
複製代碼
3、 BrowseApplicationInfoAdapter.java : 自定義適配器類,為ListView提供視圖
  1. //自定义适配器类,提供给listView的自定义view
  2. public class BrowseApplicationInfoAdapter extends BaseAdapter {
  3.         
  4.         private List<AppInfo> mlistAppInfo = null;
  5.         
  6.         LayoutInflater infater = null;
  7.    
  8.         public BrowseApplicationInfoAdapter(Context context,  List<AppInfo> apps) {
  9.                 infater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
  10.                 mlistAppInfo = apps ;
  11.         }
  12.         @Override
  13.         public int getCount() {
  14.                 // TODO Auto-generated method stub
  15.                 System.out.println("size" + mlistAppInfo.size());
  16.                 return mlistAppInfo.size();
  17.         }
  18.         @Override
  19.         public Object getItem(int position) {
  20.                 // TODO Auto-generated method stub
  21.                 return mlistAppInfo.get(position);
  22.         }
  23.         @Override
  24.         public long getItemId(int position) {
  25.                 // TODO Auto-generated method stub
  26.                 return 0;
  27.         }
  28.         @Override
  29.         public View getView(int position, View convertview, ViewGroup arg2) {
  30.                 System.out.println("getView at " + position);
  31.                 View view = null;
  32.                 ViewHolder holder = null;
  33.                 if (convertview == null || convertview.getTag() == null) {
  34.                         view = infater.inflate(R.layout.browse_app_item, null);
  35.                         holder = new ViewHolder(view);
  36.                         view.setTag(holder);
  37.                 }
  38.                 else{
  39.                         view = convertview ;
  40.                         holder = (ViewHolder) convertview.getTag() ;
  41.                 }
  42.                 AppInfo appInfo = (AppInfo) getItem(position);
  43.                 holder.appIcon.setImageDrawable(appInfo.getAppIcon());
  44.                 holder.tvAppLabel.setText(appInfo.getAppLabel());
  45.                 holder.tvPkgName.setText(appInfo.getPkgName());
  46.                 return view;
  47.         }

  48.         class ViewHolder {
  49.                 ImageView appIcon;
  50.                 TextView tvAppLabel;
  51.                 TextView tvPkgName;

  52.                 public ViewHolder(View view) {
  53.                         this.appIcon = (ImageView) view.findViewById(R.id.imgApp);
  54.                         this.tvAppLabel = (TextView) view.findViewById(R.id.tvAppLabel);
  55.                         this.tvPkgName = (TextView) view.findViewById(R.id.tvPkgName);
  56.                 }
  57.         }
  58. }
複製代碼
4 、MainActivity.java 主工程邏輯  
          請仔細體會queryIntentActivities()方法,並且注意到排序,它很重要。
  1. <span style="font-size: 13px;">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(
  12.                                 this, mlistAppInfo);
  13.                 listview.setAdapter(browseAppAdapter);
  14.                 listview.setOnItemClickListener(this);
  15.         }
  16.         // 点击跳转至该应用程序
  17.         public void onItemClick(AdapterView<?> arg0, View view, int position,
  18.                         long arg3) {
  19.                 // TODO Auto-generated method stub
  20.                 Intent intent = mlistAppInfo.get(position).getIntent();
  21.                 startActivity(intent);
  22.         }
  23.         // 获得所有启动Activity的信息,类似于Launch界面
  24.         public void queryAppInfo() {
  25.                 PackageManager pm = this.getPackageManager(); // 获得PackageManager对象
  26.                 Intent mainIntent = new Intent(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,new ResolveInfo.DisplayNameComparator(pm));
  34.                 if (mlistAppInfo != null) {
  35.                         mlistAppInfo.clear();
  36.                         for (ResolveInfo reInfo : resolveInfos) {
  37.                                 String activityName = reInfo.activityInfo.name; // 获得该应用程序的启动Activity的name
  38.                                 String pkgName = reInfo.activityInfo.packageName; // 获得应用程序的包名
  39.                                 String appLabel = (String) reInfo.loadLabel(pm); // 获得应用程序的Label
  40.                                 Drawable icon = reInfo.loadIcon(pm); // 获得应用程序图标
  41.                                 // 为应用程序的启动Activity 准备Intent
  42.                                 Intent launchIntent = new Intent();
  43.                                 launchIntent.setComponent(new ComponentName(pkgName,
  44.                                                 activityName));
  45.                                 // 创建一个AppInfo对象,并赋值
  46.                                 AppInfo appInfo = new AppInfo();
  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對象。
       圖:
                       

過濾應用程序如下:
  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
  44.                                 .getInstalledApplications(PackageManager.GET_UNINSTALLED_PACKAGES);
  45.                 Collections.sort(listAppcations,
  46.                                 new ApplicationInfo.DisplayNameComparator(pm));// 排序
  47.                 List<AppInfo> appInfos = new ArrayList<AppInfo>(); // 保存过滤查到的AppInfo
  48.                 // 根据条件来过滤
  49.                 switch (filter) {
  50.                 case FILTER_ALL_APP: // 所有应用程序
  51.                         appInfos.clear();
  52.                         for (ApplicationInfo app : listAppcations) {
  53.                                 appInfos.add(getAppInfo(app));
  54.                         }
  55.                         return appInfos;
  56.                 case FILTER_SYSTEM_APP: // 系统程序
  57.                         appInfos.clear();
  58.                         for (ApplicationInfo app : listAppcations) {
  59.                                 if ((app.flags & ApplicationInfo.FLAG_SYSTEM) != 0) {
  60.                                         appInfos.add(getAppInfo(app));
  61.                                 }
  62.                         }
  63.                         return appInfos;
  64.                 case FILTER_THIRD_APP: // 第三方应用程序
  65.                         appInfos.clear();
  66.                         for (ApplicationInfo app : listAppcations) {
  67.                                 //非系统程序
  68.                                 if ((app.flags & ApplicationInfo.FLAG_SYSTEM) <= 0) {
  69.                                         appInfos.add(getAppInfo(app));
  70.                                 }
  71.                                 //本来是系统程序,被用户手动更新后,该系统程序也成为第三方应用程序了
  72.                                 else if ((app.flags & ApplicationInfo.FLAG_UPDATED_SYSTEM_APP) != 0){
  73.                                         appInfos.add(getAppInfo(app));
  74.                                 }
  75.                         }
  76.                         break;
  77.                 case FILTER_SDCARD_APP: // 安装在SDCard的应用程序
  78.                         appInfos.clear();
  79.                         for (ApplicationInfo app : listAppcations) {
  80.                                 if ((app.flags & ApplicationInfo.FLAG_EXTERNAL_STORAGE) != 0) {
  81.                                         appInfos.add(getAppInfo(app));
  82.                                 }
  83.                         }
  84.                         return appInfos;
  85.                 default:
  86.                         return null;
  87.                 }
  88.                 return appInfos;
  89.         }
  90.         // 构造一个AppInfo对象 ,并赋值
  91.         private AppInfo getAppInfo(ApplicationInfo app) {
  92.                 AppInfo appInfo = new AppInfo();
  93.                 appInfo.setAppLabel((String) app.loadLabel(pm));
  94.                 appInfo.setAppIcon(app.loadIcon(pm));
  95.                 appInfo.setPkgName(app.packageName);
  96.                 return appInfo;
  97.         }
  98. }
複製代碼
    你可以在此基礎上,構建更多豐富的應用。比說說Settings模塊中的卸載安裝應用程序等。


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


 

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

本版積分規則



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

GMT+8, 2016-12-8 20:03 , Processed in 0.060980 second(s), 22 queries .

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

Powered by Discuz! X3.2

© 2001-2013 Comsenz Inc.

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