Android中獲得正在運(yùn)行的程序和系統(tǒng)服務(wù)的方法
ActivityManager.RunningAppProcessInfo類與獲取正在運(yùn)行的應(yīng)用程序
每一個(gè)應(yīng)用程序都會(huì)運(yùn)行在它獨(dú)立的進(jìn)程里,但是為了節(jié)省資源或者這些應(yīng)用程序是為了完成某一共同工作,它們
也可能會(huì)運(yùn)行在一個(gè)進(jìn)程里。
知識(shí)點(diǎn)介紹:
ActivityManager.RunningAppProcessInfo類
說(shuō)明: 封裝了正在運(yùn)行的進(jìn)程信息
常用字段:
int pid 進(jìn)程ID
int uid 進(jìn)程所在的用戶ID
String processName 進(jìn)程名,默認(rèn)是包名或者由android:process=””屬性指定
String [ ] pkgList 運(yùn)行在該進(jìn)程下的所有應(yīng)用程序包名
Demo說(shuō)明:
我們利用ActivityManager獲取所有正在運(yùn)行的進(jìn)程信息后,也就是獲取了每個(gè)進(jìn)程里正在運(yùn)行的應(yīng)用程序包名(pkgname),那么通過(guò)這些包名(pkgname),直接調(diào)用PackageManager類提供的方法,可以獲取這些應(yīng)用程序的信息了。
一些資源文件就不貼了,直接貼出了主工程邏輯。需要注意的在這兒我們一次性獲取了所有應(yīng)用程序信息,然后對(duì)這些應(yīng)用程序進(jìn)行過(guò)濾,得到我們需要的對(duì)象。 讀者可以使用PackageManager類提供的方法,進(jìn)行循環(huán)遍歷所有包名(pkgname),但是這樣效率會(huì)比較低。
截圖如下:
點(diǎn)擊某一進(jìn)程后
查看某一進(jìn)程運(yùn)行的應(yīng)用程序信息、所有正在運(yùn)行的進(jìn)程信息:
顯示正在運(yùn)行應(yīng)用程序的工程代碼如下:
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>(); // 查詢某一特定進(jìn)程的所有應(yīng)用程序 Intent intent = getIntent(); //是否查詢某一特定pid的應(yīng)用程序 int pid = intent.getIntExtra("EXTRA_PROCESS_ID", -1); if ( pid != -1) { //某一特定經(jīng)常里所有正在運(yùn)行的應(yīng)用程序 mlistAppInfo =querySpecailPIDRunningAppInfo(intent, pid); } else{ // 查詢所有正在運(yùn)行的應(yīng)用程序信息: 包括他們所在的進(jìn)程id和進(jìn)程名 tvInfo.setText("所有正在運(yùn)行的應(yīng)用程序有-------"); mlistAppInfo = queryAllRunningAppInfo(); } BrowseRunningAppAdapter browseAppAdapter = new BrowseRunningAppAdapter(this, mlistAppInfo); listview.setAdapter(browseAppAdapter); } // 查詢所有正在運(yùn)行的應(yīng)用程序信息: 包括他們所在的進(jìn)程id和進(jìn)程名 // 這兒我直接獲取了系統(tǒng)里安裝的所有應(yīng)用程序,然后根據(jù)報(bào)名pkgname過(guò)濾獲取所有真正運(yùn)行的應(yīng)用程序 private List<RunningAppInfo> queryAllRunningAppInfo() { pm = this.getPackageManager(); // 查詢所有已經(jīng)安裝的應(yīng)用程序 List<ApplicationInfo> listAppcations = pm.getInstalledApplications(PackageManager.GET_UNINSTALLED_PACKAGES); Collections.sort(listAppcations,new ApplicationInfo.DisplayNameComparator(pm));// 排序 // 保存所有正在運(yùn)行的包名 以及它所在的進(jìn)程信息 Map<String, ActivityManager.RunningAppProcessInfo> pgkProcessAppMap = new HashMap<String, ActivityManager.RunningAppProcessInfo>(); ActivityManager mActivityManager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE); // 通過(guò)調(diào)用ActivityManager的getRunningAppProcesses()方法獲得系統(tǒng)里所有正在運(yùn)行的進(jìn)程 List<ActivityManager.RunningAppProcessInfo> appProcessList = mActivityManager .getRunningAppProcesses(); for (ActivityManager.RunningAppProcessInfo appProcess : appProcessList) { int pid = appProcess.pid; // pid String processName = appProcess.processName; // 進(jìn)程名 Log.i(TAG, "processName: " + processName + " pid: " + pid); String[] pkgNameList = appProcess.pkgList; // 獲得運(yùn)行在該進(jìn)程里的所有應(yīng)用程序包 // 輸出所有應(yīng)用程序的包名 for (int i = 0; i < pkgNameList.length; i++) { String pkgName = pkgNameList[i]; Log.i(TAG, "packageName " + pkgName + " at index " + i+ " in process " + pid); // 加入至map對(duì)象里 pgkProcessAppMap.put(pkgName, appProcess); } } // 保存所有正在運(yùn)行的應(yīng)用程序信息 List<RunningAppInfo> runningAppInfos = new ArrayList<RunningAppInfo>(); // 保存過(guò)濾查到的AppInfo for (ApplicationInfo app : listAppcations) { // 如果該包名存在 則構(gòu)造一個(gè)RunningAppInfo對(duì)象 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)常里所有正在運(yùn)行的應(yī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("進(jìn)程id為"+pid +" 運(yùn)行的應(yīng)用程序共有 : "+pkgNameList.length); pm = this.getPackageManager(); // 保存所有正在運(yùn)行的應(yīng)用程序信息 List<RunningAppInfo> runningAppInfos = new ArrayList<RunningAppInfo>(); // 保存過(guò)濾查到的AppInfo for(int i = 0 ; i<pkgNameList.length ;i++){ //根據(jù)包名查詢特定的ApplicationInfo對(duì)象 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代表沒(méi)有任何標(biāo)記; } return runningAppInfos ; } // 構(gòu)造一個(gè)RunningAppInfo對(duì)象 ,并賦值 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類獲取正在運(yùn)行的服務(wù)
ActivityManager.RunningServiceInfo類: 封裝了正在運(yùn)行的服務(wù)信息
獲取系統(tǒng)里所有真正運(yùn)行的服務(wù)是通過(guò)調(diào)用ActivityManager方法來(lái)得到的,具體方法如下:
List<ActivityManager.RunningServiceInfo> getRunningServices (int maxNum)
功能:返回所有正在運(yùn)行的服務(wù)
參數(shù): maxNum 代表我們希望返回的服務(wù)數(shù)目大小,一般給個(gè)稍大的值即可, 例如,50 。
ActivityManager.RunningServiceInfo 類
常用字段:
long activeSince 服務(wù)第一次被激活的時(shí)間, 包括啟動(dòng)和綁定方式
int clientCount 如果該Service是通過(guò)Bind方法方式連接,則clientCount代表了service連接客戶端的數(shù)目
int crashCount 服務(wù)運(yùn)行期間,出現(xiàn)死機(jī)的次數(shù)
boolean foreground 若為true,則該服務(wù)在后臺(tái)執(zhí)行
int pid 如果不為0,表示該service所在的進(jìn)程ID號(hào)( PS:為0的話我也不清楚 - - 求指點(diǎn))
int uid 用戶ID 類似于Linux的用戶權(quán)限,例如root等
String process 進(jìn)程名,默認(rèn)是包名或者由屬性android:process指定
ComponentName service 獲得該Service的組件信息 包含了pkgname / servicename信息
PackageManger類
說(shuō)明: 封裝了對(duì)應(yīng)用程序信息的操作
獲得應(yīng)用程序信息的的方法如下:
public abstractApplicationInfo getApplicationInfo(String packageName, int flags)
參數(shù):packagename 包名
flags 該ApplicationInfo是此flags標(biāo)記,通??梢灾苯淤x予常數(shù)0即可
功能:返回ApplicationInfo對(duì)象
Demo說(shuō)明:
我們獲取了系統(tǒng)里正在運(yùn)行的服務(wù)信息,包括包名,圖標(biāo),service類名等。為了達(dá)到Settings下應(yīng)用程序模塊中的正在運(yùn)行服務(wù)的效果,我們點(diǎn)擊某一服務(wù)后,理論上來(lái)說(shuō)是可以停止該服務(wù)的,但是由于權(quán)限permissions不夠,可能報(bào)SecurityException異常,導(dǎo)致應(yīng)用程序發(fā)生異常。
關(guān)于權(quán)限不夠的問(wèn)題,可以分為兩種:
1、 在AndroidManifest.xml文件中,為<activity/>或<service/>節(jié)點(diǎn)指定android:permission屬性時(shí),在其他進(jìn)程中操作時(shí),需要聲明該permission權(quán)限 。
2、 系統(tǒng)權(quán)限,這個(gè)咱就沒(méi)什么話說(shuō)了。
截圖如下:
主工程邏輯如下:
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類 用來(lái)保存所有進(jìn)程信息 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服務(wù)的對(duì)象 mActivityManager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE); // 獲得正在運(yùn)行的Service信息 getRunningServiceInfo(); // 對(duì)集合排序 Collections.sort(serviceInfoList, new comparatorServiceLable()); System.out.println(serviceInfoList.size() + "-------------"); // 為L(zhǎng)istView構(gòu)建適配器對(duì)象 BrowseRunningServiceAdapter mServiceInfoAdapter = new BrowseRunningServiceAdapter(BrowseRunningServiceActivity.this, serviceInfoList); listviewService.setAdapter(mServiceInfoAdapter); tvTotalServiceNo.setText("當(dāng)前正在運(yùn)行的服務(wù)共有:" + serviceInfoList.size()); } // 獲得系統(tǒng)正在運(yùn)行的進(jìn)程信息 private void getRunningServiceInfo() { // 設(shè)置一個(gè)默認(rèn)Service的數(shù)量大小 int defaultNum = 20; // 通過(guò)調(diào)用ActivityManager的getRunningAppServicees()方法獲得系統(tǒng)里所有正在運(yùn)行的進(jìn)程 List<ActivityManager.RunningServiceInfo> runServiceList = mActivityManager .getRunningServices(defaultNum); System.out.println(runServiceList.size()); // ServiceInfo Model類 用來(lái)保存所有進(jìn)程信息 serviceInfoList = new ArrayList<RunSericeModel>(); for (ActivityManager.RunningServiceInfo runServiceInfo : runServiceList) { // 獲得Service所在的進(jìn)程的信息 int pid = runServiceInfo.pid; // service所在的進(jìn)程ID號(hào) int uid = runServiceInfo.uid; // 用戶ID 類似于Linux的權(quán)限不同,ID也就不同 比如 root等 // 進(jìn)程名,默認(rèn)是包名或者由屬性android:process指定 String processName = runServiceInfo.process; // 該Service啟動(dòng)時(shí)的時(shí)間值 long activeSince = runServiceInfo.activeSince; // 如果該Service是通過(guò)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, "所在進(jìn)程id :" + pid + " 所在進(jìn)程名:" + processName + " 所在進(jìn)程uid:" + uid + "\n" + " service啟動(dòng)的時(shí)間值:" + activeSince + " 客戶端綁定數(shù)目:" + clientCount + "\n" + "該service的組件信息:" + serviceName + " and " + pkgName); // 這兒我們通過(guò)service的組件信息,利用PackageManager獲取該service所在應(yīng)用程序的包名 ,圖標(biāo)等 PackageManager mPackageManager = this.getPackageManager(); // 獲取PackagerManager對(duì)象; 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); // 設(shè)置該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( "是否停止服務(wù)").setMessage( "服務(wù)只有在重新啟動(dòng)后,才可以繼續(xù)運(yùn)行。但這可能會(huì)給電子市場(chǎng)應(yīng)用程序帶來(lái)意想不到的結(jié)果。") .setPositiveButton("停止", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { // TODO Auto-generated method stub // 停止該Service //由于權(quán)限不夠的問(wèn)題,為了避免應(yīng)用程序出現(xiàn)異常,捕獲該SecurityException ,并彈出對(duì)話框 try { stopService(stopserviceIntent); } catch (SecurityException sEx) { //發(fā)生異常 說(shuō)明權(quán)限不夠 System.out.println(" deny the permission"); new AlertDialog.Builder(BrowseRunningServiceActivity.this).setTitle( "權(quán)限不夠").setMessage("對(duì)不起,您的權(quán)限不夠,無(wú)法停止該Service").create().show(); } // 刷新界面 // 獲得正在運(yùn)行的Service信息 getRunningServiceInfo(); // 對(duì)集合排序 Collections.sort(serviceInfoList, new comparatorServiceLable()); // 為L(zhǎng)istView構(gòu)建適配器對(duì)象 BrowseRunningServiceAdapter mServiceInfoAdapter = new BrowseRunningServiceAdapter( BrowseRunningServiceActivity.this, serviceInfoList); listviewService.setAdapter(mServiceInfoAdapter); tvTotalServiceNo.setText("當(dāng)前正在運(yùn)行的服務(wù)共有:" + serviceInfoList.size()); } }).setNegativeButton("取消", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { // TODO Auto-generated method stub dialog.dismiss(); // 取消對(duì)話框 } }).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()); } } }
相關(guān)文章
Android藍(lán)牙聊天開(kāi)源項(xiàng)目
這篇文章主要為大家詳細(xì)介紹了Android藍(lán)牙聊天開(kāi)源項(xiàng)目的開(kāi)發(fā),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-06-06Android仿天貓商品拋物線加入購(gòu)物車動(dòng)畫(huà)
這篇文章主要為大家詳細(xì)介紹了Android仿天貓商品拋物線加入購(gòu)物車動(dòng)畫(huà),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-06-06Android ScrollView滑動(dòng)實(shí)現(xiàn)仿QQ空間標(biāo)題欄漸變
這篇文章主要為大家詳細(xì)介紹了Android ScrollView滑動(dòng)實(shí)現(xiàn)仿QQ空間標(biāo)題欄漸變,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-08-08Android App的運(yùn)行環(huán)境及Android系統(tǒng)架構(gòu)概覽
這篇文章主要介紹了Android App的運(yùn)行環(huán)境及Android系統(tǒng)架構(gòu)概覽,并對(duì)應(yīng)用程序進(jìn)程間隔離機(jī)制等知識(shí)點(diǎn)作了介紹,需要的朋友可以參考下2016-03-03詳解Android的Splash啟動(dòng)圖的兩種動(dòng)態(tài)切換方式
本篇文章主要介紹了詳解Android的Splash啟動(dòng)圖的兩種動(dòng)態(tài)切換方式,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06Android實(shí)現(xiàn)帶頁(yè)面切換的鎖屏功能
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)帶頁(yè)面切換的鎖屏功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-06-06Android實(shí)現(xiàn)保存圖片到本地并在相冊(cè)中顯示
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)保存圖片到本地并在相冊(cè)中顯示的相關(guān)代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-03-03Android6.0開(kāi)發(fā)中屏幕旋轉(zhuǎn)原理與流程分析
這篇文章主要介紹了Android6.0開(kāi)發(fā)中屏幕旋轉(zhuǎn)原理與流程,結(jié)合實(shí)例形式詳細(xì)分析了Android6.0屏幕旋轉(zhuǎn)的原理與相關(guān)實(shí)現(xiàn)流程,并附帶了Android動(dòng)態(tài)開(kāi)啟與禁用屏幕旋轉(zhuǎn)的實(shí)現(xiàn)方法,需要的朋友可以參考下2017-11-11Android實(shí)現(xiàn)按鈕點(diǎn)擊事件的三種方法總結(jié)
Button是程序用于和用戶進(jìn)行交互的一個(gè)重要控件。既然有Button,那肯定有onClick方法,下面就教大家三種實(shí)現(xiàn)點(diǎn)擊事件的方法,感興趣的可以了解一下2022-04-04Android布局控件DrawerLayout實(shí)現(xiàn)完美側(cè)滑效果
這篇文章主要為大家詳細(xì)介紹了Android布局控件DrawerLayout實(shí)現(xiàn)完美側(cè)滑效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-08-08