热门标签 | 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()); 
    } 
 
  } 
 
} 


推荐阅读
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 本文讲述了如何通过代码在Android中更改Recycler视图项的背景颜色。通过在onBindViewHolder方法中设置条件判断,可以实现根据条件改变背景颜色的效果。同时,还介绍了如何修改底部边框颜色以及提供了RecyclerView Fragment layout.xml和项目布局文件的示例代码。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • 本文详细介绍了Linux中进程控制块PCBtask_struct结构体的结构和作用,包括进程状态、进程号、待处理信号、进程地址空间、调度标志、锁深度、基本时间片、调度策略以及内存管理信息等方面的内容。阅读本文可以更加深入地了解Linux进程管理的原理和机制。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • 本文介绍了使用AJAX的POST请求实现数据修改功能的方法。通过ajax-post技术,可以实现在输入某个id后,通过ajax技术调用post.jsp修改具有该id记录的姓名的值。文章还提到了AJAX的概念和作用,以及使用async参数和open()方法的注意事项。同时强调了不推荐使用async=false的情况,并解释了JavaScript等待服务器响应的机制。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • Webmin远程命令执行漏洞复现及防护方法
    本文介绍了Webmin远程命令执行漏洞CVE-2019-15107的漏洞详情和复现方法,同时提供了防护方法。漏洞存在于Webmin的找回密码页面中,攻击者无需权限即可注入命令并执行任意系统命令。文章还提供了相关参考链接和搭建靶场的步骤。此外,还指出了参考链接中的数据包不准确的问题,并解释了漏洞触发的条件。最后,给出了防护方法以避免受到该漏洞的攻击。 ... [详细]
  • Linux磁盘的分区、格式化的观察和操作步骤
    本文介绍了如何观察Linux磁盘的分区状态,使用lsblk命令列出系统上的所有磁盘列表,并解释了列表中各个字段的含义。同时,还介绍了使用parted命令列出磁盘的分区表类型和分区信息的方法。在进行磁盘分区操作时,根据分区表类型选择使用fdisk或gdisk命令,并提供了具体的分区步骤。通过本文,读者可以了解到Linux磁盘分区和格式化的基本知识和操作步骤。 ... [详细]
  • 本文介绍了Linux系统中正则表达式的基础知识,包括正则表达式的简介、字符分类、普通字符和元字符的区别,以及在学习过程中需要注意的事项。同时提醒读者要注意正则表达式与通配符的区别,并给出了使用正则表达式时的一些建议。本文适合初学者了解Linux系统中的正则表达式,并提供了学习的参考资料。 ... [详细]
  • Ubuntu 9.04中安装谷歌Chromium浏览器及使用体验[图文]
    nsitionalENhttp:www.w3.orgTRxhtml1DTDxhtml1-transitional.dtd ... [详细]
  • 如何在服务器主机上实现文件共享的方法和工具
    本文介绍了在服务器主机上实现文件共享的方法和工具,包括Linux主机和Windows主机的文件传输方式,Web运维和FTP/SFTP客户端运维两种方式,以及使用WinSCP工具将文件上传至Linux云服务器的操作方法。此外,还介绍了在迁移过程中需要安装迁移Agent并输入目的端服务器所在华为云的AK/SK,以及主机迁移服务会收集的源端服务器信息。 ... [详细]
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社区 版权所有