本節内容是如何擷取Android系統中應用程式的資訊,主要包括packagename、label、icon、占用大小等。具體分為兩個
部分,計劃如下:
第一部分: 擷取應用程式的packagename、label、icon等 ;
每部分都為您準備了簡單豐富的執行個體,您一定不會錯過。
Android系統為我們提供了很多服務管理的類,包括ActivityManager、PowerManager(電源管理)、AudioManager(音頻管理)
等。除此之外,還提供了一個PackageManger管理類,它的主要職責是管理應用程式包。 通過它,我們就可以擷取應用程式資訊。
引入: AnroidManifest.xml檔案節點說明:

說明: 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
說明: 獲得應用程式中<activity/>或者 <receiver />節點的資訊 。我們可以通過它來擷取我們設定的任何屬性,包括
說明: 同ActivityInfo類似 ,同樣繼承自 PackageItemInfo,隻不過它表示的是<service>節點資訊。
說明:擷取一個特定引用程式中<application>節點的資訊。
字段說明:
常用方法繼承至PackageItemInfo類中的loadIcon()和loadLabel()
說明:根據<intent>節點來擷取其上一層目錄的資訊,通常是<activity>、<receiver>、<service>節點資訊。
常用字段:
public ActivityInfo activityInfo 擷取 ActivityInfo對象,即<activity>或<receiver >節點資訊
public ServiceInfo serviceInfo 擷取 ServiceInfo對象,即<activity>節點資訊
常用方法:
Drawable loadIcon(PackageManager pm) 獲得目前應用程式的圖像
CharSequence loadLabel(PackageManager pm) 獲得目前應用程式的label
說明:手動擷取AndroidManifest.xml檔案的資訊 。
public ActivityInfo[] activities 所有<activity>節點資訊
public ApplicationInfo applicationInfo <application>節點資訊,隻有一個
public ServiceInfo[] services 所有<service>節點資訊 ,多個
說明: 獲得已安裝的應用程式資訊 。可以通過getPackageManager()方法獲得。
常用方法:
public abstract PackageManager getPackageManager()
功能:獲得一個PackageManger對象
參數: packageName 包名
功能:傳回給定包名的圖示,否則傳回null
參數:packagename 包名
flags 該ApplicationInfo是此flags标記,通常可以直接賦予常數0即可
功能:傳回該ApplicationInfo對象
的flags過濾,得到我們需要的。
功能:傳回給定條件的所有PackageInfo
參數如上
功能:傳回給定條件的所有PackageInfo
參數: intent 查尋條件,Activity所配置的action和category
參數同上
參數同上
參數同上
通過前面的介紹,相信您一定很了解了,本質上來講,這些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
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical" android:layout_width="fill_parent"
android:layout_height="fill_parent">>
<ListView android:id="@+id/listviewApp" android:layout_width="fill_parent"
android:layout_height="fill_parent" ></ListView>
</LinearLayout>
browse_app_item.xmlbrowse_app_item.xml
android:layout_width="fill_parent" android:layout_height="50dip">
<ImageView android:id="@+id/imgApp" android:layout_width="wrap_content"
android:layout_height="fill_parent" ></ImageView>
<RelativeLayout android:layout_width="fill_parent" android:layout_marginLeft="10dip"
android:layout_height="40dip">
<TextView android:id="@+id/tvLabel" android:layout_width="wrap_content"
android:layout_height="wrap_content" android:text="AppLable : "></TextView>
<TextView android:id="@+id/tvAppLabel" android:layout_width="wrap_content"
android:layout_toRightOf="@id/tvLabel" android:layout_height="wrap_content"
android:layout_marginLeft="3dip" android:text="Label" android:textColor="#FFD700"></TextView>
<TextView android:id="@+id/tvName" android:layout_width="wrap_content"
android:layout_height="wrap_content" android:layout_below="@id/tvLabel"
android:text="包名:"></TextView>
<TextView android:id="@+id/tvPkgName" android:layout_width="wrap_content"
android:layout_height="wrap_content" android:layout_below="@id/tvAppLabel"
android:layout_alignLeft="@id/tvAppLabel" android:textColor="#FFD700"></TextView>
</RelativeLayout>
2 、AppInfo.java : 儲存應用程式資訊的Model類
/Model類 ,用來存儲應用程式資訊
public class AppInfo {
private String appLabel; //應用程式标簽
private Drawable appIcon ; //應用程式圖像
private Intent intent ; //啟動應用程式的Intent ,一般是Action為Main和Category為Lancher的Activity
private String pkgName ; //應用程式所對應的包名
public AppInfo(){}
public String getAppLabel() {
return appLabel;
}
public void setAppLabel(String appName) {
this.appLabel = appName;
public Drawable getAppIcon() {
return appIcon;
public void setAppIcon(Drawable appIcon) {
this.appIcon = appIcon;
public Intent getIntent() {
return intent;
public void setIntent(Intent intent) {
this.intent = intent;
public String getPkgName(){
return pkgName ;
public void setPkgName(String pkgName){
this.pkgName=pkgName ;
}
3、 BrowseApplicationInfoAdapter.java : 自定義擴充卡類,為ListView提供視圖
//自定義擴充卡類,提供給listView的自定義view
public class BrowseApplicationInfoAdapter extends BaseAdapter {
private List<AppInfo> mlistAppInfo = null;
LayoutInflater infater = null;
public BrowseApplicationInfoAdapter(Context context, List<AppInfo> apps) {
infater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
mlistAppInfo = apps ;
@Override
public int getCount() {
// TODO Auto-generated method stub
System.out.println("size" + mlistAppInfo.size());
return mlistAppInfo.size();
public Object getItem(int position) {
return mlistAppInfo.get(position);
public long getItemId(int position) {
return 0;
public View getView(int position, View convertview, ViewGroup arg2) {
System.out.println("getView at " + position);
View view = null;
ViewHolder holder = null;
if (convertview == null || convertview.getTag() == null) {
view = infater.inflate(R.layout.browse_app_item, null);
holder = new ViewHolder(view);
view.setTag(holder);
}
else{
view = convertview ;
holder = (ViewHolder) convertview.getTag() ;
}
AppInfo appInfo = (AppInfo) getItem(position);
holder.appIcon.setImageDrawable(appInfo.getAppIcon());
holder.tvAppLabel.setText(appInfo.getAppLabel());
holder.tvPkgName.setText(appInfo.getPkgName());
return view;
class ViewHolder {
ImageView appIcon;
TextView tvAppLabel;
TextView tvPkgName;
public ViewHolder(View view) {
this.appIcon = (ImageView) view.findViewById(R.id.imgApp);
this.tvAppLabel = (TextView) view.findViewById(R.id.tvAppLabel);
this.tvPkgName = (TextView) view.findViewById(R.id.tvPkgName);
4 、MainActivity.java 主工程邏輯
請仔細體會queryIntentActivities()方法,并且注意到排序,它很重要。
<span style="font-size:13px;">public class MainActivity extends Activity implements OnItemClickListener {
private ListView listview = null;
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) {
Intent intent = mlistAppInfo.get(position).getIntent();
startActivity(intent);
// 獲得所有啟動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, PackageManager.MATCH_DEFAULT_ONLY);
// 調用系統排序 , 根據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); // 添加至清單中
System.out.println(appLabel + " activityName---" + activityName
+ " pkgName---" + pkgName);
}
}</span>
好了,第一個Demo完成 。。
Demo 2:
我們希望的ApplicationInfo對象。
圖:
過濾應用程式如下:
package com.qiner.appinfo;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import com.qiner.appinfo.R;
import android.app.Activity;
import android.app.Application;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.ListView;
public class MainActivity extends Activity {
public static final int FILTER_ALL_APP = 0; // 所有應用程式
public static final int FILTER_SYSTEM_APP = 1; // 系統程式
public static final int FILTER_THIRD_APP = 2; // 第三方應用程式
public static final int FILTER_SDCARD_APP = 3; // 安裝在SDCard的應用程式
private PackageManager pm;
private int filter = FILTER_ALL_APP;
private List<AppInfo> mlistAppInfo ;
private BrowseApplicationInfoAdapter browseAppAdapter = null ;
/** Called when the activity is first created. */
if(getIntent()!=null){
filter = getIntent().getIntExtra("filter", 0) ;
mlistAppInfo = queryFilterAppInfo(filter); // 查詢所有應用程式資訊
// 建構擴充卡,并且注冊到listView
browseAppAdapter = new BrowseApplicationInfoAdapter(this, mlistAppInfo);
// 根據查詢條件,查詢特定的ApplicationInfo
private List<AppInfo> queryFilterAppInfo(int filter) {
pm = this.getPackageManager();
// 查詢所有已經安裝的應用程式
List<ApplicationInfo> listAppcations = pm
.getInstalledApplications(PackageManager.GET_UNINSTALLED_PACKAGES);
Collections.sort(listAppcations,
new ApplicationInfo.DisplayNameComparator(pm));// 排序
List<AppInfo> appInfos = new ArrayList<AppInfo>(); // 儲存過濾查到的AppInfo
// 根據條件來過濾
switch (filter) {
case FILTER_ALL_APP: // 所有應用程式
appInfos.clear();
for (ApplicationInfo app : listAppcations) {
appInfos.add(getAppInfo(app));
return appInfos;
case FILTER_SYSTEM_APP: // 系統程式
if ((app.flags & ApplicationInfo.FLAG_SYSTEM) != 0) {
appInfos.add(getAppInfo(app));
}
case FILTER_THIRD_APP: // 第三方應用程式
//非系統程式
if ((app.flags & ApplicationInfo.FLAG_SYSTEM) <= 0) {
}
//本來是系統程式,被使用者手動更新後,該系統程式也成為第三方應用程式了
else if ((app.flags & ApplicationInfo.FLAG_UPDATED_SYSTEM_APP) != 0){
break;
case FILTER_SDCARD_APP: // 安裝在SDCard的應用程式
if ((app.flags & ApplicationInfo.FLAG_EXTERNAL_STORAGE) != 0) {
default:
return null;
return appInfos;
// 構造一個AppInfo對象 ,并指派
private AppInfo getAppInfo(ApplicationInfo app) {
AppInfo appInfo = new AppInfo();
appInfo.setAppLabel((String) app.loadLabel(pm));
appInfo.setAppIcon(app.loadIcon(pm));
appInfo.setPkgName(app.packageName);
return appInfo;
你可以在此基礎上,建構更多豐富的應用。比說說Settings子產品中的解除安裝安裝應用程式等。