转载请注明出处:http://blog.csdn.net/qinjuning

     

本节内容是如何获取Android系统中应用程序的信息,主要包括packagename、label、icon、占用大小等。具体分为两个

部分,计划如下:

 第一部分: 获取应用程序的packagename、label、icon等 ;

            第二部分:获取应用程序的占用大小,包括:缓存大小(cachsize)、数据大小(datasize)。

每部分都为您准备了简单丰富的实例,您一定不会错过。

Android系统为我们提供了很多服务管理的类,包括ActivityManager、PowerManager(电源管理)、AudioManager(音频管理)

等。除此之外,还提供了一个PackageManger管理类,它的主要职责是管理应用程序包。 通过它,我们就可以获取应用程序信息。

引入: AnroidManifest.xml文件节点说明:

一、相关类的介绍

PackageItemInfo类

 说明: AndroidManifest.xml文件中所有节点的基类,提供了这些节点的基本信息:a label、icon、 meta-data。它并不

直接使用,而是由子类继承然后调用相应方法。

常用字段

public int icon           获得该资源图片在R文件中的值 (对应于android:icon属性)

public int labelRes     获得该label在R文件中的值(对应于android:label属性)

public String name   获得该节点的name值 (对应于android:name属性)

public String packagename   获得该应用程序的包名 (对应于android:packagename属性)

  常用方法

Drawable  loadIcon(PackageManager pm)               获得当前应用程序的图像

CharSequence  loadLabel(PackageManager pm)     获得当前应用程序的label

ActivityInfo类  继承自 PackageItemInfo

  说明: 获得应用程序中<activity/>或者 <receiver />节点的信息 。我们可以通过它来获取我们设置的任何属性,包括

theme 、launchMode、launchmode等

常用方法继承至PackageItemInfo类中的loadIcon()和loadLabel()

ServiceInfo 类

说明: 同ActivityInfo类似 ,同样继承自 PackageItemInfo,只不过它表示的是<service>节点信息。

ApplicationInfo类 继承自  PackageItemInfo

说明:获取一个特定引用程序中<application>节点的信息。

字段说明

      flags字段: FLAG_SYSTEM 系统应用程序

                   FLAG_EXTERNAL_STORAGE 表示该应用安装在sdcard中

 常用方法继承至PackageItemInfo类中的loadIcon()和loadLabel()

ResolveInfo类

 说明:根据<intent>节点来获取其上一层目录的信息,通常是<activity>、<receiver>、<service>节点信息。

  常用字段

public  ActivityInfo  activityInfo     获取 ActivityInfo对象,即<activity>或<receiver >节点信息

public ServiceInfo   serviceInfo     获取 ServiceInfo对象,即<service>节点信息

常用方法

Drawable loadIcon(PackageManager pm)             获得当前应用程序的图像

CharSequence loadLabel(PackageManager pm)  获得当前应用程序的label

PackageInfo类

 说明:手动获取AndroidManifest.xml文件的信息 。

常用字段

public String    packageName                   包名

public ActivityInfo[]     activities                   所有<activity>节点信息

public ApplicationInfo applicationInfo       <application>节点信息,只有一个

public ActivityInfo[]    receivers                  所有<receiver>节点信息,多个

public ServiceInfo[]    services                  所有<service>节点信息 ,多个

PackageManger 类

说明: 获得已安装的应用程序信息 。可以通过getPackageManager()方法获得。

常用方法

  public abstract PackageManager  getPackageManager()

功能:获得一个PackageManger对象

public abstrac  tDrawable    getApplicationIcon(StringpackageName)

参数: packageName 包名

功能:返回给定包名的图标,否则返回null

public abstract ApplicationInfo   getApplicationInfo(String packageName, int flags)

参数:packagename 包名

flags 该ApplicationInfo是此flags标记,通常可以直接赋予常数0即可

功能:返回该ApplicationInfo对象

     public abstract List<ApplicationInfo>  getInstalledApplications(int flags)

参数:flag为一般为GET_UNINSTALLED_PACKAGES,那么此时会返回所有ApplicationInfo。我们可以对ApplicationInfo

的flags过滤,得到我们需要的。

功能:返回给定条件的所有PackageInfo

    public abstract List<PackageInfo>  getInstalledPackages(int flags) 

参数如上

功能:返回给定条件的所有PackageInfo

public abstractResolveInfo  resolveActivity(Intent intent, int flags)

参数:  intent 查寻条件,Activity所配置的action和category

flags: MATCH_DEFAULT_ONLY    :Category必须带有CATEGORY_DEFAULT的Activity,才匹配

GET_INTENT_FILTERS         :匹配Intent条件即可

GET_RESOLVED_FILTER    :匹配Intent条件即可

功能 :返回给定条件的ResolveInfo对象(本质上是Activity)

public abstract  List<ResolveInfo>  queryIntentActivities(Intent intent, int flags)

参数同上

功能 :返回给定条件的所有ResolveInfo对象(本质上是Activity),集合对象

public abstract ResolveInfo  resolveService(Intent intent, int flags)

参数同上

功能 :返回给定条件的ResolveInfo对象(本质上是Service)

public abstract List<ResolveInfoqueryIntentServices(Intent intent, int flags)

参数同上

功能 :返回给定条件的所有ResolveInfo对象(本质上是Service),集合对象

二、DEMO讲解

通过前面的介绍,相信您一定很了解了,本质上来讲,这些XXXInfo类不过是我们在AndroidManifest.XML文件中定义的信息,

知道到这点了,理解起来就不是很难了。

下面我透过两个简答的DEMO,来学以致用。

   Demo 1: 通过queryIntentActivities()方法,查询Android系统的所有具备ACTION_MAIN和CATEGORY_LAUNCHER

      的Intent的应用程序,点击后,能启动该应用,说白了就是做一个类似Home程序的简易Launcher 。

           Demo 2 :通过getInstalledApplications()方法获取应用,然后对其过滤,查找出我们需要的第三方应用,系统应用,安装在sdcard的应用。

      Demo1  :

         图:

               

  1 、布局文件: 主要有两个:带listview的browse_app_list.xml文件 ;listview的项browse_app_item.xml

browse_app_list.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:orientation="vertical" android:layout_width="fill_parent"
  4. android:layout_height="fill_parent">>
  5. <ListView android:id="@+id/listviewApp" android:layout_width="fill_parent"
  6. android:layout_height="fill_parent" ></ListView>
  7. </LinearLayout>

browse_app_item.xmlbrowse_app_item.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="fill_parent" android:layout_height="50dip">
  4. <ImageView android:id="@+id/imgApp" android:layout_width="wrap_content"
  5. android:layout_height="fill_parent" ></ImageView>
  6. <RelativeLayout android:layout_width="fill_parent"  android:layout_marginLeft="10dip"
  7. android:layout_height="40dip">
  8. <TextView android:id="@+id/tvLabel" android:layout_width="wrap_content"
  9. android:layout_height="wrap_content" android:text="AppLable : "></TextView>
  10. <TextView android:id="@+id/tvAppLabel" android:layout_width="wrap_content"
  11. android:layout_toRightOf="@id/tvLabel" android:layout_height="wrap_content"
  12. android:layout_marginLeft="3dip" android:text="Label" android:textColor="#FFD700"></TextView>
  13. <TextView android:id="@+id/tvName" android:layout_width="wrap_content"
  14. android:layout_height="wrap_content" android:layout_below="@id/tvLabel"
  15. android:text="包名:"></TextView>
  16. <TextView android:id="@+id/tvPkgName" android:layout_width="wrap_content"
  17. android:layout_height="wrap_content" android:layout_below="@id/tvAppLabel"
  18. android:layout_alignLeft="@id/tvAppLabel" android:textColor="#FFD700"></TextView>
  19. </RelativeLayout>
  20. </LinearLayout>

2 、AppInfo.Java : 保存应用程序信息的Model类

  1. /Model类 ,用来存储应用程序信息
  2. public class AppInfo {
  3. private String appLabel;    //应用程序标签
  4. private Drawable appIcon ;  //应用程序图像
  5. private Intent intent ;     //启动应用程序的Intent ,一般是Action为Main和Category为Lancher的Activity
  6. private String pkgName ;    //应用程序所对应的包名
  7. public AppInfo(){}
  8. public String getAppLabel() {
  9. return appLabel;
  10. }
  11. public void setAppLabel(String appName) {
  12. this.appLabel = appName;
  13. }
  14. public Drawable getAppIcon() {
  15. return appIcon;
  16. }
  17. public void setAppIcon(Drawable appIcon) {
  18. this.appIcon = appIcon;
  19. }
  20. public Intent getIntent() {
  21. return intent;
  22. }
  23. public void setIntent(Intent intent) {
  24. this.intent = intent;
  25. }
  26. public String getPkgName(){
  27. return pkgName ;
  28. }
  29. public void setPkgName(String pkgName){
  30. this.pkgName=pkgName ;
  31. }
  32. }

3、 BrowseApplicationInfoAdapter.java : 自定义适配器类,为ListView提供视图

  1. //自定义适配器类,提供给listView的自定义view
  2. public class BrowseApplicationInfoAdapter extends BaseAdapter {
  3. private List<AppInfo> mlistAppInfo = null;
  4. LayoutInflater infater = null;
  5. public BrowseApplicationInfoAdapter(Context context,  List<AppInfo> apps) {
  6. infater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
  7. mlistAppInfo = apps ;
  8. }
  9. @Override
  10. public int getCount() {
  11. // TODO Auto-generated method stub
  12. System.out.println("size" + mlistAppInfo.size());
  13. return mlistAppInfo.size();
  14. }
  15. @Override
  16. public Object getItem(int position) {
  17. // TODO Auto-generated method stub
  18. return mlistAppInfo.get(position);
  19. }
  20. @Override
  21. public long getItemId(int position) {
  22. // TODO Auto-generated method stub
  23. return 0;
  24. }
  25. @Override
  26. public View getView(int position, View convertview, ViewGroup arg2) {
  27. System.out.println("getView at " + position);
  28. View view = null;
  29. ViewHolder holder = null;
  30. if (convertview == null || convertview.getTag() == null) {
  31. view = infater.inflate(R.layout.browse_app_item, null);
  32. holder = new ViewHolder(view);
  33. view.setTag(holder);
  34. }
  35. else{
  36. view = convertview ;
  37. holder = (ViewHolder) convertview.getTag() ;
  38. }
  39. AppInfo appInfo = (AppInfo) getItem(position);
  40. holder.appIcon.setImageDrawable(appInfo.getAppIcon());
  41. holder.tvAppLabel.setText(appInfo.getAppLabel());
  42. holder.tvPkgName.setText(appInfo.getPkgName());
  43. return view;
  44. }
  45. class ViewHolder {
  46. ImageView appIcon;
  47. TextView tvAppLabel;
  48. TextView tvPkgName;
  49. public ViewHolder(View view) {
  50. this.appIcon = (ImageView) view.findViewById(R.id.imgApp);
  51. this.tvAppLabel = (TextView) view.findViewById(R.id.tvAppLabel);
  52. this.tvPkgName = (TextView) view.findViewById(R.id.tvPkgName);
  53. }
  54. }
  55. }

4 、MainActivity.java 主工程逻辑

请仔细体会queryIntentActivities()方法,并且注意到排序,它很重要。

  1. <span style="font-size:13px;">public class MainActivity extends Activity implements OnItemClickListener {
  2. private ListView listview = null;
  3. private List<AppInfo> mlistAppInfo = null;
  4. @Override
  5. public void onCreate(Bundle savedInstanceState) {
  6. super.onCreate(savedInstanceState);
  7. setContentView(R.layout.browse_app_list);
  8. listview = (ListView) findViewById(R.id.listviewApp);
  9. mlistAppInfo = new ArrayList<AppInfo>();
  10. queryAppInfo(); // 查询所有应用程序信息
  11. BrowseApplicationInfoAdapter browseAppAdapter = new BrowseApplicationInfoAdapter(
  12. this, mlistAppInfo);
  13. listview.setAdapter(browseAppAdapter);
  14. listview.setOnItemClickListener(this);
  15. }
  16. // 点击跳转至该应用程序
  17. public void onItemClick(AdapterView<?> arg0, View view, int position,
  18. long arg3) {
  19. // TODO Auto-generated method stub
  20. Intent intent = mlistAppInfo.get(position).getIntent();
  21. startActivity(intent);
  22. }
  23. // 获得所有启动Activity的信息,类似于Launch界面
  24. public void queryAppInfo() {
  25. PackageManager pm = this.getPackageManager(); // 获得PackageManager对象
  26. Intent mainIntent = new Intent(Intent.ACTION_MAIN, null);
  27. mainIntent.addCategory(Intent.CATEGORY_LAUNCHER);
  28. // 通过查询,获得所有ResolveInfo对象.
  29. List<ResolveInfo> resolveInfos = pm
  30. .queryIntentActivities(mainIntent, PackageManager.MATCH_DEFAULT_ONLY);
  31. // 调用系统排序 , 根据name排序
  32. // 该排序很重要,否则只能显示系统应用,而不能列出第三方应用程序
  33. Collections.sort(resolveInfos,new ResolveInfo.DisplayNameComparator(pm));
  34. if (mlistAppInfo != null) {
  35. mlistAppInfo.clear();
  36. for (ResolveInfo reInfo : resolveInfos) {
  37. String activityName = reInfo.activityInfo.name; // 获得该应用程序的启动Activity的name
  38. String pkgName = reInfo.activityInfo.packageName; // 获得应用程序的包名
  39. String appLabel = (String) reInfo.loadLabel(pm); // 获得应用程序的Label
  40. Drawable icon = reInfo.loadIcon(pm); // 获得应用程序图标
  41. // 为应用程序的启动Activity 准备Intent
  42. Intent launchIntent = new Intent();
  43. launchIntent.setComponent(new ComponentName(pkgName,
  44. activityName));
  45. // 创建一个AppInfo对象,并赋值
  46. AppInfo appInfo = new AppInfo();
  47. appInfo.setAppLabel(appLabel);
  48. appInfo.setPkgName(pkgName);
  49. appInfo.setAppIcon(icon);
  50. appInfo.setIntent(launchIntent);
  51. mlistAppInfo.add(appInfo); // 添加至列表中
  52. System.out.println(appLabel + " activityName---" + activityName
  53. + " pkgName---" + pkgName);
  54. }
  55. }
  56. }
  57. }</span>

好了,第一个Demo完成 。。

 

 Demo 2:

demo2在布局、适配器方面和Demo1一样。只是利用了getInstalledApplications()方法,继而通过ApplicationInfo.flags来挑选

我们希望的ApplicationInfo对象。

图:

         

过滤应用程序如下:

  1. package com.qiner.appinfo;
  2. import java.util.ArrayList;
  3. import java.util.Collections;
  4. import java.util.List;
  5. import com.qiner.appinfo.R;
  6. import android.app.Activity;
  7. import android.app.Application;
  8. import android.content.pm.ApplicationInfo;
  9. import android.content.pm.PackageManager;
  10. import android.os.Bundle;
  11. import android.view.View;
  12. import android.view.View.OnClickListener;
  13. import android.widget.Button;
  14. import android.widget.ListView;
  15. public class MainActivity extends Activity  {
  16. public static final int FILTER_ALL_APP = 0; // 所有应用程序
  17. public static final int FILTER_SYSTEM_APP = 1; // 系统程序
  18. public static final int FILTER_THIRD_APP = 2; // 第三方应用程序
  19. public static final int FILTER_SDCARD_APP = 3; // 安装在SDCard的应用程序
  20. private ListView listview = null;
  21. private PackageManager pm;
  22. private int filter = FILTER_ALL_APP;
  23. private List<AppInfo> mlistAppInfo ;
  24. private BrowseApplicationInfoAdapter browseAppAdapter = null ;
  25. /** Called when the activity is first created. */
  26. @Override
  27. public void onCreate(Bundle savedInstanceState) {
  28. super.onCreate(savedInstanceState);
  29. setContentView(R.layout.browse_app_list);
  30. listview = (ListView) findViewById(R.id.listviewApp);
  31. if(getIntent()!=null){
  32. filter = getIntent().getIntExtra("filter", 0) ;
  33. }
  34. mlistAppInfo = queryFilterAppInfo(filter); // 查询所有应用程序信息
  35. // 构建适配器,并且注册到listView
  36. browseAppAdapter = new BrowseApplicationInfoAdapter(this, mlistAppInfo);
  37. listview.setAdapter(browseAppAdapter);
  38. }
  39. // 根据查询条件,查询特定的ApplicationInfo
  40. private List<AppInfo> queryFilterAppInfo(int filter) {
  41. pm = this.getPackageManager();
  42. // 查询所有已经安装的应用程序
  43. List<ApplicationInfo> listAppcations = pm
  44. .getInstalledApplications(PackageManager.GET_UNINSTALLED_PACKAGES);
  45. Collections.sort(listAppcations,
  46. new ApplicationInfo.DisplayNameComparator(pm));// 排序
  47. List<AppInfo> appInfos = new ArrayList<AppInfo>(); // 保存过滤查到的AppInfo
  48. // 根据条件来过滤
  49. switch (filter) {
  50. case FILTER_ALL_APP: // 所有应用程序
  51. appInfos.clear();
  52. for (ApplicationInfo app : listAppcations) {
  53. appInfos.add(getAppInfo(app));
  54. }
  55. return appInfos;
  56. case FILTER_SYSTEM_APP: // 系统程序
  57. appInfos.clear();
  58. for (ApplicationInfo app : listAppcations) {
  59. if ((app.flags & ApplicationInfo.FLAG_SYSTEM) != 0) {
  60. appInfos.add(getAppInfo(app));
  61. }
  62. }
  63. return appInfos;
  64. case FILTER_THIRD_APP: // 第三方应用程序
  65. appInfos.clear();
  66. for (ApplicationInfo app : listAppcations) {
  67. //非系统程序
  68. if ((app.flags & ApplicationInfo.FLAG_SYSTEM) <= 0) {
  69. appInfos.add(getAppInfo(app));
  70. }
  71. //本来是系统程序,被用户手动更新后,该系统程序也成为第三方应用程序了
  72. else if ((app.flags & ApplicationInfo.FLAG_UPDATED_SYSTEM_APP) != 0){
  73. appInfos.add(getAppInfo(app));
  74. }
  75. }
  76. break;
  77. case FILTER_SDCARD_APP: // 安装在SDCard的应用程序
  78. appInfos.clear();
  79. for (ApplicationInfo app : listAppcations) {
  80. if ((app.flags & ApplicationInfo.FLAG_EXTERNAL_STORAGE) != 0) {
  81. appInfos.add(getAppInfo(app));
  82. }
  83. }
  84. return appInfos;
  85. default:
  86. return null;
  87. }
  88. return appInfos;
  89. }
  90. // 构造一个AppInfo对象 ,并赋值
  91. private AppInfo getAppInfo(ApplicationInfo app) {
  92. AppInfo appInfo = new AppInfo();
  93. appInfo.setAppLabel((String) app.loadLabel(pm));
  94. appInfo.setAppIcon(app.loadIcon(pm));
  95. appInfo.setPkgName(app.packageName);
  96. return appInfo;
  97. }
  98. }

你可以在此基础上,构建更多丰富的应用。比说说Settings模块中的卸载安装应用程序等。

   本节的源代码已上传,下载地址:http://download.csdn.net/detail/qinjuning/3775869

from:http://blog.csdn.net/qinjuning/article/details/6867806

【转】Android中获取应用程序(包)的信息-----PackageManager的使用(一)的更多相关文章

  1. Android中获取应用程序(包)的信息----PackageManager

    本节内容是如何获取Android系统中应用程序的信息,主要包括packagename.label.icon.占用大小等.具体分为两个 部分,计划如下:   第一部分: 获取应用程序的packagena ...

  2. Android中获取应用程序(包)的信息-----PackageManager的使用(一)

    本节内容是如何获取Android系统中应用程序的信息,主要包括packagename.label.icon.占用大小等.具体分为两个 部分,计划如下:  第一部分: 获取应用程序的packagenam ...

  3. Android中获取应用程序(包)的信息-----PackageManager的使用

    本节内容是如何获取Android系统中应用程序的信息,主要包括packagename.label.icon.占用大小等.具体分为两个 部分,计划如下: 第一部分: 获取应用程序的packagename ...

  4. Android中获取应用程序(包)的大小-----PackageManager的使用(二)

    通过第一部分<<Android中获取应用程序(包)的信息-----PackageManager的使用(一)>>的介绍,对PackageManager以及 AndroidMani ...

  5. Android中获取正在运行的应用程序-----ActivityManager.RunningAppProcessInfo类详解

    今天继续讲解关于ActivityManager的使用,通过前面一节的学习,我们学会了如何利用ActivityManager获取系统里 正在运行的进程.本文要讲解的知识点是利用这些进程信息获取系统里正在 ...

  6. Android中获取正在运行的服务-------ActivityManager.RunningServiceInfo的使用

    关于PackageManager和ActivityManager的使用 ,自己也写了一些DEMO 了,基本上写的线路参考了Settings模块下的 应用程序,大家如果真正的有所兴趣,建议大家看看源码, ...

  7. Android实现获取应用程序相关信息列表的方法

    本文所述为Androdi获取手机应用列表的方法,比如获取到Android应用的软件属性.大小和应用程序路径.应用名称等,获取所有已安装的Android应用列表,包括那些卸载了的,但没有清除数据的应用程 ...

  8. Android中获取系统上安装的APP信息

    Version:0.9 StartHTML:-1 EndHTML:-1 StartFragment:00000099 EndFragment:00003259 Android中获取系统上安装的APP信 ...

  9. URL转Drawable之 Android中获取网络图片的三种方法

    转载自: http://doinone.iteye.com/blog/1074283 Android中获取网络图片是一件耗时的操作,如果直接获取有可能会出现应用程序无响应(ANR:Applicatio ...

随机推荐

  1. Java基础教程(12)--深入理解类

    一.方法的返回值   当我们在程序中调用方法时,虚拟机将会跳转到对应的方法中去执行.当以下几种情况发生时,虚拟机将会回到调用方法的语句并继续向下执行: 执行完方法中所有的语句: 遇到return语句: ...

  2. 手把手教你写一个java的orm(五)

    生成sql:where 上一篇里我们实现了生成insert的sql,下面要开始实现update,delete,select的sql语句了.但是这些语句有一个比较麻烦的地方是:它们一般后面都会有wher ...

  3. PL/SQL Developer图形化窗口创建数据库(表空间和用户)以及相关查询sql

    前言:上一篇安装好oracle和pl/sql后,这篇主要讲如何创建数据库,因为接下来我的项目会连接数据库进行开发. 第一步.先用系统管理员登录pl/sql 我这里系统管理员用户名为system,密码为 ...

  4. QT的信号和槽机制简介

    信号与槽作为QT的核心机制在QT编程中有着广泛的应用,本文介绍了信号与槽的一些基本概念.元对象工具以及在实际使用过程中应注意的一些问题. QT是一个跨平台的C++ GUI应用构架,它提供了丰富的窗口部 ...

  5. 重温jQuery

    Write Less, Do More! ——John Resig(jQuery设计者) 目录 基础知识 概况 编程访问DOM节点 基础知识 Web网页是有结构的HTML文档.浏览器分析HTML文档, ...

  6. 【 js 片段 】如何组织表单的默认提交?【亲测有效】

    最近做的一个项目,我分到的部分有表单验证.点击了提交按钮,但我并不想让他跳转页面去提交.于是经过各种百度,各种 stackoverflow,各种抱大神腿之后,有了以下解决办法: 重点就是阻止 form ...

  7. Jquery使用Id获取焦点和失去焦点

    Jquery使用Id获取焦点和失去焦点有2种方法 先用第一种(val()=="空"): <div> <input type="text" id ...

  8. jedis、jedisPool、jedisCluster的使用方法

    jedis 连接redis(单机): 使用jedis如何操作redis,但是其实方法是跟redis的操作大部分是相对应的. 所有的redis命令都对应jedis的一个方法     1.在macen工程 ...

  9. FFmpeg实现将图片转换为视频

    ##名称:ffmpeg实现将图片转换为视频 ##平台:ubuntu(已经安装好了ffmpeg工具) ##日期:2017年12月10日 简介: 因为学习需要,需要将连续图片转换成视频,昨天和今天早上用o ...

  10. 安装部署 OpenPAI Install OpenPAI on Ubuntu

      介绍 不管是机器学习的老手,还是入门的新人,都应该装备上尽可能强大的算力.除此之外,还要压榨出硬件的所有潜力来加快模型训练.OpenPAI作为GPU管理的利器,不管是一块GPU,还是上千块GPU, ...