Android之——获取手机安装的应用程序

Posted mfmdaoyou

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Android之——获取手机安装的应用程序相关的知识,希望对你有一定的参考价值。

转载请注明出处:http://blog.csdn.net/l1028386804/article/details/47114331

前几篇有关android的博文中。向大家介绍了几个项目中经常使用的有用功能。那么在这篇博文中。我将向大家介绍怎样获取Android手机中已经安装的手机应用程序,有用过金山手机卫士或者360手机卫士的童鞋都知道。这些软件都能够获取到当前手机中安装的应用程序。

那么,这些功能是怎样实现的呢?就让我们一起来看看这些功能详细是怎样实现的吧。

一、原理

原理非常easy,我们通过Android中提供的PackageManager类,来获取手机中安装的应用程序信息,将这些信息封装成一个对象,这个对象包括应用程序的图标、名称、版本、包名、是否是用户应用程序等信息。然后将这些对象封装成一个对象集合。再将这个集合显示到界面的listView上。形成一个应用程序列表。呈现给用户的便是一个手机中安装的应用程序列表了。

原理是不是非常easy呢?以下,就让我们一起来实现这些功能吧。

二、实现

1、创建应用程序的实体类AppInfo

为了更加面向对象化和体现面向对象的封装性。我将获取到的每个应用程序信息封装成了一个java对象,这个对象包括应用程序的图标、名称、版本、包名、是否是用户应用程序等信息。

详细实现代码例如以下:

package cn.lyz.mobilesafe.domain;

import android.graphics.drawable.Drawable;

/**
 * 获取的应用基本信息实体类
 * @author liuyazhuang
 *
 */
public class AppInfo {
	//图标
	private Drawable app_icon;
	//应用名称
	private String app_name;
	//应用版本
	private String app_version;
	//应用包名
	private String packagename;
	//是否是用户app
	private boolean isUserApp;
	
	
	public AppInfo() {
		super();
		// TODO Auto-generated constructor stub
	}
	public AppInfo(Drawable app_icon, String app_name, String app_version,
			String packagename) {
		super();
		this.app_icon = app_icon;
		this.app_name = app_name;
		this.app_version = app_version;
		this.packagename = packagename;
	}
	
	
	public AppInfo(Drawable app_icon, String app_name, String app_version,
			String packagename, boolean isUserApp) {
		super();
		this.app_icon = app_icon;
		this.app_name = app_name;
		this.app_version = app_version;
		this.packagename = packagename;
		this.isUserApp = isUserApp;
	}
	public Drawable getApp_icon() {
		return app_icon;
	}
	public void setApp_icon(Drawable app_icon) {
		this.app_icon = app_icon;
	}
	public String getApp_name() {
		return app_name;
	}
	public void setApp_name(String app_name) {
		this.app_name = app_name;
	}
	public String getApp_version() {
		return app_version;
	}
	public void setApp_version(String app_version) {
		this.app_version = app_version;
	}
	public String getPackagename() {
		return packagename;
	}
	public void setPackagename(String packagename) {
		this.packagename = packagename;
	}
	
	public boolean isUserApp() {
		return isUserApp;
	}
	public void setUserApp(boolean isUserApp) {
		this.isUserApp = isUserApp;
	}
	
	@Override
	public String toString() {
		return "AppInfo [app_icon=" + app_icon + ", app_name=" + app_name
				+ ", app_version=" + app_version + ", packagename="
				+ packagename + ", isUserApp=" + isUserApp + "]";
	}
}

2、获取手机应用程序的业务类AppInfoService

这个类主要实现获取手机中安装的应用程序的主要业务功能。封装了怎样获取手机安装的应用程序的方法。

详细实现代码例如以下:

package cn.lyz.mobilesafe.engine;

import java.util.ArrayList;
import java.util.List;

import android.content.Context;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.content.pm.PackageManager.NameNotFoundException;
import android.graphics.drawable.Drawable;
import cn.lyz.mobilesafe.domain.AppInfo;

/**
 * 获取手机应用程序
 * @author liuyazhuang
 *
 */
public class AppInfoService {

	private Context context;
	private PackageManager pm;
	public AppInfoService(Context context) {
		// TODO Auto-generated constructor stub
		this.context = context;
		pm = context.getPackageManager();
	}
	
	/**
	 * 得到手机中全部的应用程序信息
	 * @return
	 */
	public List<AppInfo> getAppInfos(){
		//创建要返回的集合对象
		List<AppInfo> appInfos = new ArrayList<AppInfo>();
		//获取手机中全部安装的应用集合
		List<ApplicationInfo> applicationInfos = pm.getInstalledApplications(PackageManager.GET_UNINSTALLED_PACKAGES);
		//遍历全部的应用集合
		for(ApplicationInfo info : applicationInfos){
			
			AppInfo appInfo = new AppInfo();
			
			//获取应用程序的图标
			Drawable app_icon = info.loadIcon(pm);
			appInfo.setApp_icon(app_icon);
			
			//获取应用的名称
			String app_name = info.loadLabel(pm).toString();
			appInfo.setApp_name(app_name);
			
			//获取应用的包名
			String packageName = info.packageName;
			appInfo.setPackagename(packageName);
			try {
				//获取应用的版本
				PackageInfo packageInfo = pm.getPackageInfo(packageName, 0);
				String app_version = packageInfo.versionName;
				appInfo.setApp_version(app_version);
			} catch (NameNotFoundException e) {
				e.printStackTrace();
			}
			//推断应用程序是否是用户程序
			boolean isUserApp = filterApp(info);
			appInfo.setUserApp(isUserApp);
			appInfos.add(appInfo);
		}
		return appInfos;
	}
	
	//推断应用程序是否是用户程序
    public boolean filterApp(ApplicationInfo info) {
    	//原来是系统应用,用户手动升级
        if ((info.flags & ApplicationInfo.FLAG_UPDATED_SYSTEM_APP) != 0) {
            return true;
            //用户自己安装的应用程序
        } else if ((info.flags & ApplicationInfo.FLAG_SYSTEM) == 0) {
            return true;
        }
        return false;
    }
}

3、样式文件styles.xml

在res/values文件夹下新建styles.xml文件来定义应用程序的样式信息。我在这个文件里主要定义个两个样式。

详细代码例如以下:

<style name="view_divide_line_style">
        <item name="android:layout_width">fill_parent</item>
        <item name="android:layout_height">1dip</item>
        <item name="android:layout_marginTop">5dip</item>
        <item name="android:background">@drawable/devide_line</item>
    </style>
<style name="text_title_style">
        <item name="android:layout_width">fill_parent</item>
        <item name="android:layout_height">wrap_content</item>
        <item name="android:gravity">center</item>
        <item name="android:textColor">#42E700</item>
        <item name="android:textSize">25sp</item>
    </style>

4、主布局文件applationinstall.xml

详细实现代码例如以下:

<?

xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" > <TextView android:id="@+id/tv_title" style="@style/text_title_style" android:text="所 有 程 序" /> <View style="@style/view_divide_line_style" /> <FrameLayout android:layout_width="fill_parent" android:layout_height="fill_parent" > <!-- android:cacheColorHint="#00000000" 缓存的颜色 默认是黄色 android:divider="#00ffffff" 切割线 android:dividerHeight="3.0dip" 切割线的宽度 --> <ListView android:id="@+id/lv_appmanage" android:layout_width="fill_parent" android:layout_height="fill_parent" android:cacheColorHint="#00000000" android:divider="#fff" android:fastScrollEnabled="true" android:dividerHeight="1.0dip" android:paddingLeft="3.0dip" android:paddingRight="3.0dip" /> <RelativeLayout android:id="@+id/rl_loading" android:layout_width="fill_parent" android:layout_height="fill_parent" > <LinearLayout android:layout_width="wrap_content" android:layout_height="wrap_content" android:orientation="vertical" android:layout_centerInParent="true"> <ProgressBar android:id="@+id/pb" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_gravity="center" /> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginTop="10dip" android:text="正在载入应用程序。。" android:textColor="#fff" android:textSize="22sp" /> </LinearLayout> </RelativeLayout> </FrameLayout> </LinearLayout>

5、ListView中每一项条目布局applationinstall_item.xml

详细实现代码例如以下

<?xml version="1.0" encoding="utf-8"?

> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="wrap_content" android:orientation="horizontal" android:paddingBottom="8.0dip" android:paddingLeft="6.0dip" android:paddingRight="5.0dip" android:paddingTop="8.0dip" > <ImageView android:id="@+id/iv_appicon" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerVertical="true" android:src="@drawable/ic_launcher" /> <TextView android:id="@+id/tv_appname" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerVertical="true" android:layout_marginLeft="4.0dip" android:layout_toRightOf="@id/iv_appicon" android:textColor="#fff" android:text="我最摇摆" android:textSize="16.0dip" /> <TextView android:id="@+id/tv_appversion" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentRight="true" android:layout_centerVertical="true" android:layout_marginLeft="4.0dip" android:textColor="#fff" android:layout_marginRight="10dp" android:text="1.0" android:textSize="16.0dip" /> </RelativeLayout>

6、自己定义ListView适配器AppManagerAdapter

这个类继承自BaseAdapter主要作为List显示数据的适配器,在这个类中通过布局载入器LayoutInflater来载入条目布局。找到布局上的控件来设置对应的信息。

详细实现代码例如以下:

package cn.lyz.mobilesafe.adapter;

import java.util.List;

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.TextView;
import cn.lyz.mobilesafe.R;
import cn.lyz.mobilesafe.domain.AppInfo;

/**
 * App管理的Adapter类
 * @author liuyazhuang
 *
 */
public class AppManagerAdapter extends BaseAdapter {

	private Context context;
	//布局载入器
	private LayoutInflater mInflater;
	private List<AppInfo> appInfos;
	
	//动态改变appInfos
	public void setAppInfos(List<AppInfo> appInfos) {
		this.appInfos = appInfos;
	}

	public AppManagerAdapter(Context context,List<AppInfo> appInfos) {
		this.context = context;
		this.appInfos = appInfos;
		mInflater = LayoutInflater.from(context);
	}
	
	public int getCount() {
		return appInfos.size();
	}

	public Object getItem(int position) {
		return appInfos.get(position);
	}

	public long getItemId(int position) {
		return position;
	}

	public View getView(int position, View convertView, ViewGroup parent) {
		//1 得到控件
		//2 得到数据
		//3 绑定数据
		View view = null;
		if(convertView != null){
			view = convertView;
		}else{
			view = mInflater.inflate(R.layout.applationinstall_item, null);
		}
		
		//获取布局控件
		ImageView iv_appicon = (ImageView) view.findViewById(R.id.iv_appicon);
		TextView tv_appname = (TextView) view.findViewById(R.id.tv_appname);
		TextView tv_appversion = (TextView) view.findViewById(R.id.tv_appversion);
		
		//获取position位置上的AppInfo对象
		AppInfo appInfo = appInfos.get(position);
		
		iv_appicon.setImageDrawable(appInfo.getApp_icon());
		tv_appname.setText(appInfo.getApp_name());
		tv_appversion.setText(appInfo.getApp_version());
		return view;
	}

}

7、程序显示界面AppManagerActivity

这个类实现的功能非常easy,调用其它类的方法,将获取的信息显示到ListView上。

详细的实现是在onCreate方法中找到布局上的控件。并在一个线程程序中调用AppInfoService中的方法获取手机中安装的应用程序,并将获取的结果通过Handler与Message机制传递到主线程,主线程将这些数据显示到UI视图上。

详细实现代码例如以下:

package cn.lyz.mobilesafe.activity;

import java.util.ArrayList;
import java.util.List;

import android.app.Activity;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.widget.ListView;
import android.widget.PopupWindow;
import android.widget.RelativeLayout;
import android.widget.TextView;
import cn.lyz.mobilesafe.R;
import cn.lyz.mobilesafe.adapter.AppManagerAdapter;
import cn.lyz.mobilesafe.domain.AppInfo;
import cn.lyz.mobilesafe.engine.AppInfoService;

/**
 * APP管理的Manager类
 * @author liuyazhuang
 *
 */
public class AppManagerActivity extends Activity{

	protected static final int SUCCESS_GET_APPLICAITON = 0;
	
	//布局中的各个控件
	private RelativeLayout rl_loading;
	private ListView lv_appmanage;
	private TextView tv_title;
	//包管理器
	private PackageManager pm;
	//获取手机应用信息的业务类
	private AppInfoService appInfoService;
	//手机应用app信息集合
	private List<AppInfo> appInfos;
	//用户应用程序信息集合
	private List<AppInfo> userAppInfos;
	//是否是全部的app程序,默觉得true
	private boolean isAllApp = true;
	//AppManagerAdapter适配器对象
	private AppManagerAdapter mAdapter;
	
	private PopupWindow mPopupWindow;
	//mHandler方法
	private Handler mHandler = new Handler(){
		public void handleMessage(android.os.Message msg) {
			switch (msg.what) {
			case SUCCESS_GET_APPLICAITON:
				//给listview去绑定数据。隐藏载入的控件
				mAdapter = new AppManagerAdapter(getApplicationContext(), appInfos);
				//设置数据
				lv_appmanage.setAdapter(mAdapter);
				//隐藏RelativeLayout
				rl_loading.setVisibility(View.GONE);
				//View.VISIBLE (控件显示)View.INVISIBLE(控件隐藏  但占领空间)  View.GONE(控件隐藏  不占领空间)
				break;

			default:
				break;
			}
		};
	};
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		// TODO Auto-generated method stub
		super.onCreate(savedInstanceState);
		setContentView(R.layout.applationinstall);
		//获取布局中的控件
		rl_loading = (RelativeLayout) findViewById(R.id.rl_loading);
		lv_appmanage = (ListView) findViewById(R.id.lv_appmanage);
		tv_title = (TextView) findViewById(R.id.tv_title);
		//实例化AppInfoService对象
		appInfoService = new AppInfoService(this);
		//包管理器
		pm = getPackageManager();
		
		//在子线程中获取手机安装的应用程序信息
		new Thread(){
			public void run() {
				appInfos = appInfoService.getAppInfos();
				
				userAppInfos = new ArrayList<AppInfo>();
				for(AppInfo appInfo:appInfos){
					if(appInfo.isUserApp()){
						userAppInfos.add(appInfo);
					}
				}
				Message msg = new Message();
				msg.what = SUCCESS_GET_APPLICAITON;
				mHandler.sendMessage(msg);
			};
		}.start();
		
	}
}

补充:推断应用程序是否是用户程序

//推断应用程序是否是用户程序
    public boolean filterApp(ApplicationInfo info) {
    	//原来是系统应用。用户手动升级
        if ((info.flags & ApplicationInfo.FLAG_UPDATED_SYSTEM_APP) != 0) {
            return true;
            //用户自己安装的应用程序
        } else if ((info.flags & ApplicationInfo.FLAG_SYSTEM) == 0) {
            return true;
        }
        return false;
    }

三、执行效果

正在载入应用程序

技术分享

获取到手机中安装的应用程序

技术分享

四、温馨提示

本实例中。为了方面,我把一些文字直接写在了布局文件里和相关的类中。大家在真实的项目中要把这些文字写在strings.xml文件里,在外部引用这些资源,切记,这是作为一个Android程序猿最主要的开发常识和规范,我在这里仅仅是为了方便直接写在了类和布局文件里。




以上是关于Android之——获取手机安装的应用程序的主要内容,如果未能解决你的问题,请参考以下文章

Android开发之通过包管理器获取安装应用信息

Android 插件化VirtualApp 源码分析 ( 目前的 API 现状 | 安装应用源码分析 | 安装按钮执行的操作 | 返回到 HomeActivity 执行的操作 )(代码片段

Android - 片段中的联系人选择器

Android 逆向获取安装在手机中的应用的 APK 包 ( 进入 adb shell | 获取 root 权限 | 进入 /data/app/ 目录 | 拷贝 base.apk 到外置存储 )(代码

Android 逆向获取安装在手机中的应用的 APK 包 ( 进入 adb shell | 获取 root 权限 | 进入 /data/app/ 目录 | 拷贝 base.apk 到外置存储 )(代码

Android获取各个应用程序的缓存文件代码小片段(使用AIDL)