热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

Android中获得正在运行的程序和系统服务的方法

这篇文章主要介绍了Android中获得正在运行的程序和系统服务的方法,分别是对ActivityManager.RunningAppProcessInfo类和ActivityManager.RunningServiceInfo类的使用,需要的朋友可以参考下

ActivityManager.RunningAppProcessInfo类与获取正在运行的应用程序
每一个应用程序都会运行在它独立的进程里,但是为了节省资源或者这些应用程序是为了完成某一共同工作,它们
也可能会运行在一个进程里。

2016228174335726.gif (295×68)

 知识点介绍:
ActivityManager.RunningAppProcessInfo类
说明: 封装了正在运行的进程信息
常用字段:
int   pid    进程ID
int   uid    进程所在的用户ID
String   processName 进程名,默认是包名或者由android:process=””属性指定
String [ ]   pkgList      运行在该进程下的所有应用程序包名

Demo说明:
我们利用ActivityManager获取所有正在运行的进程信息后,也就是获取了每个进程里正在运行的应用程序包名(pkgname),那么通过这些包名(pkgname),直接调用PackageManager类提供的方法,可以获取这些应用程序的信息了。
一些资源文件就不贴了,直接贴出了主工程逻辑。需要注意的在这儿我们一次性获取了所有应用程序信息,然后对这些应用程序进行过滤,得到我们需要的对象。 读者可以使用PackageManager类提供的方法,进行循环遍历所有包名(pkgname),但是这样效率会比较低。
截图如下:
点击某一进程后

2016228174354207.gif (323×347)

查看某一进程运行的应用程序信息、所有正在运行的进程信息:

2016228174412674.gif (329×297)

2016228174430158.gif (345×425)

显示正在运行应用程序的工程代码如下:

 
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 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(); 
 
    // 查询某一特定进程的所有应用程序 
    Intent intent = getIntent(); 
    //是否查询某一特定pid的应用程序 
    int pid = intent.getIntExtra("EXTRA_PROCESS_ID", -1); 
     
    if ( pid != -1) { 
      //某一特定经常里所有正在运行的应用程序 
      mlistAppInfo =querySpecailPIDRunningAppInfo(intent, pid); 
    } 
    else{ 
      // 查询所有正在运行的应用程序信息: 包括他们所在的进程id和进程名 
      tvInfo.setText("所有正在运行的应用程序有-------"); 
      mlistAppInfo = queryAllRunningAppInfo();  
    } 
    BrowseRunningAppAdapter browseAppAdapter = new BrowseRunningAppAdapter(this, mlistAppInfo); 
    listview.setAdapter(browseAppAdapter); 
  } 
 
  // 查询所有正在运行的应用程序信息: 包括他们所在的进程id和进程名 
  // 这儿我直接获取了系统里安装的所有应用程序,然后根据报名pkgname过滤获取所有真正运行的应用程序 
  private List queryAllRunningAppInfo() { 
    pm = this.getPackageManager(); 
    // 查询所有已经安装的应用程序 
    List listAppcatiOns= pm.getInstalledApplications(PackageManager.GET_UNINSTALLED_PACKAGES); 
    Collections.sort(listAppcations,new ApplicationInfo.DisplayNameComparator(pm));// 排序 
 
    // 保存所有正在运行的包名 以及它所在的进程信息 
    Map pgkProcessAppMap = new HashMap(); 
 
    ActivityManager mActivityManager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE); 
    // 通过调用ActivityManager的getRunningAppProcesses()方法获得系统里所有正在运行的进程 
    List 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  runningAppInfos = new ArrayList(); // 保存过滤查到的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; 
 
  } 
  // 某一特定经常里所有正在运行的应用程序 
  private List 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 runningAppInfos = new ArrayList(); // 保存过滤查到的AppInfo 
 
    for(int i = 0 ; i

ActivityManager.RunningServiceInfo类获取正在运行的服务
ActivityManager.RunningServiceInfo类:  封装了正在运行的服务信息
 
获取系统里所有真正运行的服务是通过调用ActivityManager方法来得到的,具体方法如下:
 
List getRunningServices (int maxNum)
功能:返回所有正在运行的服务
参数:   maxNum 代表我们希望返回的服务数目大小,一般给个稍大的值即可, 例如,50 。
                              
ActivityManager.RunningServiceInfo 类
  常用字段:
 
long   activeSince        服务第一次被激活的时间, 包括启动和绑定方式
int      clientCount          如果该Service是通过Bind方法方式连接,则clientCount代表了service连接客户端的数目
int      crashCount          服务运行期间,出现死机的次数
boolean   foreground   若为true,则该服务在后台执行
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)
参数:packagename 包名
flags 该ApplicationInfo是此flags标记,通常可以直接赋予常数0即可
功能:返回ApplicationInfo对象
Demo说明:
我们获取了系统里正在运行的服务信息,包括包名,图标,service类名等。为了达到Settings下应用程序模块中的正在运行服务的效果,我们点击某一服务后,理论上来说是可以停止该服务的,但是由于权限permissions不够,可能报SecurityException异常,导致应用程序发生异常。
 
关于权限不够的问题,可以分为两种:
1、 在AndroidManifest.xml文件中,为节点指定android:permission属性时,在其他进程中操作时,需要声明该permission权限 。
2、 系统权限,这个咱就没什么话说了。
截图如下:

2016228174507748.gif (320×480)

主工程逻辑如下:

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 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()); 
  } 
  // 获得系统正在运行的进程信息 
  private void getRunningServiceInfo() { 
 
    // 设置一个默认Service的数量大小 
    int defaultNum = 20; 
    // 通过调用ActivityManager的getRunningAppServicees()方法获得系统里所有正在运行的进程 
    List runServiceList = mActivityManager 
        .getRunningServices(defaultNum); 
 
    System.out.println(runServiceList.size()); 
 
    // ServiceInfo Model类 用来保存所有进程信息 
    serviceInfoList = new ArrayList(); 
 
    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连接客户端的数目 
      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 
          + " 客户端绑定数目:" + 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<&#63;> 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( 
        "服务只有在重新启动后,才可以继续运行。但这可能会给电子市场应用程序带来意想不到的结果。") 
        .setPositiveButton("停止", new DialogInterface.OnClickListener() { 
 
          @Override 
          public void onClick(DialogInterface dialog, int which) { 
            // TODO Auto-generated method stub 
            // 停止该Service 
            //由于权限不够的问题,为了避免应用程序出现异常,捕获该SecurityException ,并弹出对话框 
            try { 
              stopService(stopserviceIntent); 
            } catch (SecurityException sEx) { 
              //发生异常 说明权限不够  
              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(); 
  } 
 
  // 自定义排序 根据AppLabel排序 
  private class comparatorServiceLable implements Comparator { 
 
    @Override 
    public int compare(RunSericeModel object1, RunSericeModel object2) { 
      // TODO Auto-generated method stub 
      return object1.getAppLabel().compareTo(object2.getAppLabel()); 
    } 
 
  } 
 
} 


推荐阅读
  • 本文介绍了ADB(Android Debug Bridge)的基本概念、安装方法、环境配置、连接真机步骤以及常用命令和高级技巧。ADB是一个强大的工具,适用于Android设备的开发和调试。 ... [详细]
  • Eclipse 中 Maven 的基础配置指南
    本文详细介绍了如何在 Eclipse 环境中配置 Maven,包括环境变量的设置、Maven 插件的安装与配置等关键步骤,旨在帮助开发者顺利搭建开发环境。 ... [详细]
  • 解决Ant编译时出现的非法字符错误
    在进行Java项目的Ant构建过程中,有时会遇到由平台差异引发的编译错误。本文将详细探讨一种常见的错误——'error: illegal character'及其解决方案。 ... [详细]
  • 目录介绍01.CoordinatorLayout滑动抖动问题描述02.滑动抖动问题分析03.自定义AppBarLayout.Behavior说明04.CoordinatorLayo ... [详细]
  • 本文介绍了两种有效的方法来检查Android应用是否拥有特定权限,如媒体读写权限(media_rw)。通过这些方法,开发者和安全人员可以更好地了解应用的行为,确保其不会滥用权限。 ... [详细]
  • 本文详细介绍了ejabberd中的验证码服务、接收器以及服务器间通信的监督者和工作进程,包括它们的启动方式和主要功能。 ... [详细]
  • nsitionalENhttp:www.w3.orgTRxhtml1DTDxhtml1-transitional.dtd ... [详细]
  • 本文介绍了ThinkPHP框架的基本概念及其主要特性。作为一款遵循Apache许可证的开源框架,ThinkPHP不仅支持多种平台和Web服务器,还提供了丰富的功能以适应不同的开发需求。 ... [详细]
  • C# 对象转 JSON 字符串的方法与应用
    本文介绍如何在 C# 中使用一般处理程序(ASHX)将对象转换为 JSON 字符串,并通过设置响应类型为 application/json 来确保客户端能够正确解析返回的数据。同时,文章还提供了 HTML 页面中不依赖 jQuery 的 AJAX 方法来接收和处理这些 JSON 数据的具体实现。 ... [详细]
  • 利用Java与Tesseract-OCR实现数字识别
    本文深入探讨了如何利用Java语言结合Tesseract-OCR技术来实现图像中的数字识别功能,旨在为开发者提供详细的指导和实践案例。 ... [详细]
  • 作为一名计算机科学专业的大三学生,我在过去的一年里自学了Visual Basic (VB),但感觉进展缓慢。VB的学习是否仅仅局限于控件的使用?如何有效地学习API?此外,有人认为Basic语言已经过时,这对VB的未来意味着什么? ... [详细]
  • 本文详细介绍了如何通过修改 Jenkins 的配置文件来解决因权限设置不当导致的登录后页面为空的问题,包括多种权限配置策略的选择与应用。 ... [详细]
  • Linux环境下生成指定范围随机数的方法
    本文详细介绍了如何在Linux系统中利用内置功能生成不同范围内的随机数,包括基本的使用方法和一些高级技巧。 ... [详细]
  • 本文详细介绍了Python编程语言中的集合(Set)数据类型,包括其基本概念、定义方法、常用操作以及应用场景,旨在帮助开发者更好地理解和利用这一高效的数据结构。 ... [详细]
  • 使用Solr从MySQL导入数据构建全量索引
    为了更好地掌握Solr的各项功能,本文档将在本地Windows环境中演示如何从MySQL数据库中导入数据至Solr,并构建全量索引。这将有助于开发者熟悉Solr的数据处理流程,尤其是在无法直接在生产服务器上进行实践的情况下。 ... [详细]
author-avatar
mobiledu2502852625
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有