Android中獲得正在運行的程序和系統(tǒng)服務的方法
ActivityManager.RunningAppProcessInfo類與獲取正在運行的應用程序
每一個應用程序都會運行在它獨立的進程里,但是為了節(jié)省資源或者這些應用程序是為了完成某一共同工作,它們
也可能會運行在一個進程里。

知識點介紹:
ActivityManager.RunningAppProcessInfo類
說明: 封裝了正在運行的進程信息
常用字段:
int pid 進程ID
int uid 進程所在的用戶ID
String processName 進程名,默認是包名或者由android:process=””屬性指定
String [ ] pkgList 運行在該進程下的所有應用程序包名
Demo說明:
我們利用ActivityManager獲取所有正在運行的進程信息后,也就是獲取了每個進程里正在運行的應用程序包名(pkgname),那么通過這些包名(pkgname),直接調(diào)用PackageManager類提供的方法,可以獲取這些應用程序的信息了。
一些資源文件就不貼了,直接貼出了主工程邏輯。需要注意的在這兒我們一次性獲取了所有應用程序信息,然后對這些應用程序進行過濾,得到我們需要的對象。 讀者可以使用PackageManager類提供的方法,進行循環(huán)遍歷所有包名(pkgname),但是這樣效率會比較低。
截圖如下:
點擊某一進程后

查看某一進程運行的應用程序信息、所有正在運行的進程信息:


顯示正在運行應用程序的工程代碼如下:
package com.qin.ammp;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import android.app.Activity;
import android.app.ActivityManager;
import android.content.Context;
import android.content.Intent;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.content.pm.PackageManager.NameNotFoundException;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.AdapterView;
import android.widget.ListView;
import android.widget.TextView;
import android.widget.AdapterView.OnItemClickListener;
public class BrowseRunningAppActivity extends Activity {
private static String TAG = "BrowseRunningAppActivity";
private ListView listview = null;
private List<RunningAppInfo> mlistAppInfo = null;
private TextView tvInfo = null ;
private PackageManager pm;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.browse_app_list);
listview = (ListView) findViewById(R.id.listviewApp);
tvInfo = (TextView)findViewById(R.id.tvInfo) ;
mlistAppInfo = new ArrayList<RunningAppInfo>();
// 查詢某一特定進程的所有應用程序
Intent intent = getIntent();
//是否查詢某一特定pid的應用程序
int pid = intent.getIntExtra("EXTRA_PROCESS_ID", -1);
if ( pid != -1) {
//某一特定經(jīng)常里所有正在運行的應用程序
mlistAppInfo =querySpecailPIDRunningAppInfo(intent, pid);
}
else{
// 查詢所有正在運行的應用程序信息: 包括他們所在的進程id和進程名
tvInfo.setText("所有正在運行的應用程序有-------");
mlistAppInfo = queryAllRunningAppInfo();
}
BrowseRunningAppAdapter browseAppAdapter = new BrowseRunningAppAdapter(this, mlistAppInfo);
listview.setAdapter(browseAppAdapter);
}
// 查詢所有正在運行的應用程序信息: 包括他們所在的進程id和進程名
// 這兒我直接獲取了系統(tǒng)里安裝的所有應用程序,然后根據(jù)報名pkgname過濾獲取所有真正運行的應用程序
private List<RunningAppInfo> queryAllRunningAppInfo() {
pm = this.getPackageManager();
// 查詢所有已經(jīng)安裝的應用程序
List<ApplicationInfo> listAppcations = pm.getInstalledApplications(PackageManager.GET_UNINSTALLED_PACKAGES);
Collections.sort(listAppcations,new ApplicationInfo.DisplayNameComparator(pm));// 排序
// 保存所有正在運行的包名 以及它所在的進程信息
Map<String, ActivityManager.RunningAppProcessInfo> pgkProcessAppMap = new HashMap<String, ActivityManager.RunningAppProcessInfo>();
ActivityManager mActivityManager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);
// 通過調(diào)用ActivityManager的getRunningAppProcesses()方法獲得系統(tǒng)里所有正在運行的進程
List<ActivityManager.RunningAppProcessInfo> appProcessList = mActivityManager
.getRunningAppProcesses();
for (ActivityManager.RunningAppProcessInfo appProcess : appProcessList) {
int pid = appProcess.pid; // pid
String processName = appProcess.processName; // 進程名
Log.i(TAG, "processName: " + processName + " pid: " + pid);
String[] pkgNameList = appProcess.pkgList; // 獲得運行在該進程里的所有應用程序包
// 輸出所有應用程序的包名
for (int i = 0; i < pkgNameList.length; i++) {
String pkgName = pkgNameList[i];
Log.i(TAG, "packageName " + pkgName + " at index " + i+ " in process " + pid);
// 加入至map對象里
pgkProcessAppMap.put(pkgName, appProcess);
}
}
// 保存所有正在運行的應用程序信息
List<RunningAppInfo> runningAppInfos = new ArrayList<RunningAppInfo>(); // 保存過濾查到的AppInfo
for (ApplicationInfo app : listAppcations) {
// 如果該包名存在 則構造一個RunningAppInfo對象
if (pgkProcessAppMap.containsKey(app.packageName)) {
// 獲得該packageName的 pid 和 processName
int pid = pgkProcessAppMap.get(app.packageName).pid;
String processName = pgkProcessAppMap.get(app.packageName).processName;
runningAppInfos.add(getAppInfo(app, pid, processName));
}
}
return runningAppInfos;
}
// 某一特定經(jīng)常里所有正在運行的應用程序
private List<RunningAppInfo> querySpecailPIDRunningAppInfo(Intent intent , int pid) {
String[] pkgNameList = intent.getStringArrayExtra("EXTRA_PKGNAMELIST");
String processName = intent.getStringExtra("EXTRA_PROCESS_NAME");
//update ui
tvInfo.setText("進程id為"+pid +" 運行的應用程序共有 : "+pkgNameList.length);
pm = this.getPackageManager();
// 保存所有正在運行的應用程序信息
List<RunningAppInfo> runningAppInfos = new ArrayList<RunningAppInfo>(); // 保存過濾查到的AppInfo
for(int i = 0 ; i<pkgNameList.length ;i++){
//根據(jù)包名查詢特定的ApplicationInfo對象
ApplicationInfo appInfo;
try {
appInfo = pm.getApplicationInfo(pkgNameList[i], 0);
runningAppInfos.add(getAppInfo(appInfo, pid, processName));
}
catch (NameNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} // 0代表沒有任何標記;
}
return runningAppInfos ;
}
// 構造一個RunningAppInfo對象 ,并賦值
private RunningAppInfo getAppInfo(ApplicationInfo app, int pid, String processName) {
RunningAppInfo appInfo = new RunningAppInfo();
appInfo.setAppLabel((String) app.loadLabel(pm));
appInfo.setAppIcon(app.loadIcon(pm));
appInfo.setPkgName(app.packageName);
appInfo.setPid(pid);
appInfo.setProcessName(processName);
return appInfo;
}
}
ActivityManager.RunningServiceInfo類獲取正在運行的服務
ActivityManager.RunningServiceInfo類: 封裝了正在運行的服務信息
獲取系統(tǒng)里所有真正運行的服務是通過調(diào)用ActivityManager方法來得到的,具體方法如下:
List<ActivityManager.RunningServiceInfo> getRunningServices (int maxNum)
功能:返回所有正在運行的服務
參數(shù): maxNum 代表我們希望返回的服務數(shù)目大小,一般給個稍大的值即可, 例如,50 。
ActivityManager.RunningServiceInfo 類
常用字段:
long activeSince 服務第一次被激活的時間, 包括啟動和綁定方式
int clientCount 如果該Service是通過Bind方法方式連接,則clientCount代表了service連接客戶端的數(shù)目
int crashCount 服務運行期間,出現(xiàn)死機的次數(shù)
boolean foreground 若為true,則該服務在后臺執(zhí)行
int pid 如果不為0,表示該service所在的進程ID號( PS:為0的話我也不清楚 - - 求指點)
int uid 用戶ID 類似于Linux的用戶權限,例如root等
String process 進程名,默認是包名或者由屬性android:process指定
ComponentName service 獲得該Service的組件信息 包含了pkgname / servicename信息
PackageManger類
說明: 封裝了對應用程序信息的操作
獲得應用程序信息的的方法如下:
public abstractApplicationInfo getApplicationInfo(String packageName, int flags)
參數(shù):packagename 包名
flags 該ApplicationInfo是此flags標記,通常可以直接賦予常數(shù)0即可
功能:返回ApplicationInfo對象
Demo說明:
我們獲取了系統(tǒng)里正在運行的服務信息,包括包名,圖標,service類名等。為了達到Settings下應用程序模塊中的正在運行服務的效果,我們點擊某一服務后,理論上來說是可以停止該服務的,但是由于權限permissions不夠,可能報SecurityException異常,導致應用程序發(fā)生異常。
關于權限不夠的問題,可以分為兩種:
1、 在AndroidManifest.xml文件中,為<activity/>或<service/>節(jié)點指定android:permission屬性時,在其他進程中操作時,需要聲明該permission權限 。
2、 系統(tǒng)權限,這個咱就沒什么話說了。
截圖如下:

主工程邏輯如下:
package com.qin.runservice;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;
import android.app.Activity;
import android.app.ActivityManager;
import android.app.AlertDialog;
import android.app.Dialog;
import android.content.ComponentName;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager;
import android.content.pm.PackageManager.NameNotFoundException;
import android.os.Bundle;
import android.os.Debug;
import android.util.Log;
import android.view.ContextMenu;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.view.ContextMenu.ContextMenuInfo;
import android.widget.AdapterView;
import android.widget.ListView;
import android.widget.TextView;
import android.widget.AdapterView.OnItemClickListener;
public class BrowseRunningServiceActivity extends Activity implements
OnItemClickListener {
private static String TAG = "RunServiceInfo";
private ActivityManager mActivityManager = null;
// ProcessInfo Model類 用來保存所有進程信息
private List<RunSericeModel> serviceInfoList = null;
private ListView listviewService;
private TextView tvTotalServiceNo;
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.browse_service_list);
listviewService = (ListView) findViewById(R.id.listviewService);
listviewService.setOnItemClickListener(this);
tvTotalServiceNo = (TextView) findViewById(R.id.tvTotalServiceNo);
// 獲得ActivityManager服務的對象
mActivityManager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);
// 獲得正在運行的Service信息
getRunningServiceInfo();
// 對集合排序
Collections.sort(serviceInfoList, new comparatorServiceLable());
System.out.println(serviceInfoList.size() + "-------------");
// 為ListView構建適配器對象
BrowseRunningServiceAdapter mServiceInfoAdapter = new
BrowseRunningServiceAdapter(BrowseRunningServiceActivity.this, serviceInfoList);
listviewService.setAdapter(mServiceInfoAdapter);
tvTotalServiceNo.setText("當前正在運行的服務共有:" + serviceInfoList.size());
}
// 獲得系統(tǒng)正在運行的進程信息
private void getRunningServiceInfo() {
// 設置一個默認Service的數(shù)量大小
int defaultNum = 20;
// 通過調(diào)用ActivityManager的getRunningAppServicees()方法獲得系統(tǒng)里所有正在運行的進程
List<ActivityManager.RunningServiceInfo> runServiceList = mActivityManager
.getRunningServices(defaultNum);
System.out.println(runServiceList.size());
// ServiceInfo Model類 用來保存所有進程信息
serviceInfoList = new ArrayList<RunSericeModel>();
for (ActivityManager.RunningServiceInfo runServiceInfo : runServiceList) {
// 獲得Service所在的進程的信息
int pid = runServiceInfo.pid; // service所在的進程ID號
int uid = runServiceInfo.uid; // 用戶ID 類似于Linux的權限不同,ID也就不同 比如 root等
// 進程名,默認是包名或者由屬性android:process指定
String processName = runServiceInfo.process;
// 該Service啟動時的時間值
long activeSince = runServiceInfo.activeSince;
// 如果該Service是通過Bind方法方式連接,則clientCount代表了service連接客戶端的數(shù)目
int clientCount = runServiceInfo.clientCount;
// 獲得該Service的組件信息 可能是pkgname/servicename
ComponentName serviceCMP = runServiceInfo.service;
String serviceName = serviceCMP.getShortClassName(); // service 的類名
String pkgName = serviceCMP.getPackageName(); // 包名
// 打印Log
Log.i(TAG, "所在進程id :" + pid + " 所在進程名:" + processName + " 所在進程uid:"
+ uid + "\n" + " service啟動的時間值:" + activeSince
+ " 客戶端綁定數(shù)目:" + clientCount + "\n" + "該service的組件信息:"
+ serviceName + " and " + pkgName);
// 這兒我們通過service的組件信息,利用PackageManager獲取該service所在應用程序的包名 ,圖標等
PackageManager mPackageManager = this.getPackageManager(); // 獲取PackagerManager對象;
try {
// 獲取該pkgName的信息
ApplicationInfo appInfo = mPackageManager.getApplicationInfo(
pkgName, 0);
RunSericeModel runService = new RunSericeModel();
runService.setAppIcon(appInfo.loadIcon(mPackageManager));
runService.setAppLabel(appInfo.loadLabel(mPackageManager) + "");
runService.setServiceName(serviceName);
runService.setPkgName(pkgName);
// 設置該service的組件信息
Intent intent = new Intent();
intent.setComponent(serviceCMP);
runService.setIntent(intent);
runService.setPid(pid);
runService.setProcessName(processName);
// 添加至集合中
serviceInfoList.add(runService);
} catch (NameNotFoundException e) {
// TODO Auto-generated catch block
System.out.println("--------------------- error -------------");
e.printStackTrace();
}
}
}
// 觸摸可停止
@Override
public void onItemClick(AdapterView<?> arg0, View arg1, int position,
long arg3) {
// TODO Auto-generated method stub
final Intent stopserviceIntent = serviceInfoList.get(position)
.getIntent();
new AlertDialog.Builder(BrowseRunningServiceActivity.this).setTitle(
"是否停止服務").setMessage(
"服務只有在重新啟動后,才可以繼續(xù)運行。但這可能會給電子市場應用程序帶來意想不到的結果。")
.setPositiveButton("停止", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
// TODO Auto-generated method stub
// 停止該Service
//由于權限不夠的問題,為了避免應用程序出現(xiàn)異常,捕獲該SecurityException ,并彈出對話框
try {
stopService(stopserviceIntent);
} catch (SecurityException sEx) {
//發(fā)生異常 說明權限不夠
System.out.println(" deny the permission");
new AlertDialog.Builder(BrowseRunningServiceActivity.this).setTitle(
"權限不夠").setMessage("對不起,您的權限不夠,無法停止該Service").create().show();
}
// 刷新界面
// 獲得正在運行的Service信息
getRunningServiceInfo();
// 對集合排序
Collections.sort(serviceInfoList, new comparatorServiceLable());
// 為ListView構建適配器對象
BrowseRunningServiceAdapter mServiceInfoAdapter = new BrowseRunningServiceAdapter(
BrowseRunningServiceActivity.this,
serviceInfoList);
listviewService.setAdapter(mServiceInfoAdapter);
tvTotalServiceNo.setText("當前正在運行的服務共有:"
+ serviceInfoList.size());
}
}).setNegativeButton("取消",
new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog,
int which) {
// TODO Auto-generated method stub
dialog.dismiss(); // 取消對話框
}
}).create().show();
}
// 自定義排序 根據(jù)AppLabel排序
private class comparatorServiceLable implements Comparator<RunSericeModel> {
@Override
public int compare(RunSericeModel object1, RunSericeModel object2) {
// TODO Auto-generated method stub
return object1.getAppLabel().compareTo(object2.getAppLabel());
}
}
}
相關文章
Android ScrollView滑動實現(xiàn)仿QQ空間標題欄漸變
這篇文章主要為大家詳細介紹了Android ScrollView滑動實現(xiàn)仿QQ空間標題欄漸變,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-08-08
Android App的運行環(huán)境及Android系統(tǒng)架構概覽
這篇文章主要介紹了Android App的運行環(huán)境及Android系統(tǒng)架構概覽,并對應用程序進程間隔離機制等知識點作了介紹,需要的朋友可以參考下2016-03-03
詳解Android的Splash啟動圖的兩種動態(tài)切換方式
本篇文章主要介紹了詳解Android的Splash啟動圖的兩種動態(tài)切換方式,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-06-06
Android布局控件DrawerLayout實現(xiàn)完美側滑效果
這篇文章主要為大家詳細介紹了Android布局控件DrawerLayout實現(xiàn)完美側滑效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-08-08

