process 类 java_编写可执行jar——java的Process类的使用(二)

你知道怎么在控制台使用ping吗?那你知道怎么在java中使用ping吗?

1.批处理文件

批处理文件大家一定不陌生。接触最多的应该就是tomcat中的start.bat或者start.sh了。bat是在windows环境下运行的批处理文件,sh则是linux的shell脚本。

2.adb指令

安卓刷机的小伙伴们肯定对adb不陌生吧?为了使用adb,我们先安装一个夜神模拟器。比如我安装在C:\Program Files\Nox\bin,那么我们先打开cmd,并且cd到C:\Program Files\Nox\bin目录中。adb指令列表可以看https://www.yeshen.com/faqs/H15tDZ6YW。我们先启动夜神模拟器。

AAffA0nNPuCLAAAAAElFTkSuQmCC

启动完成后

AAffA0nNPuCLAAAAAElFTkSuQmCC

我们此时在控制台输入adb devices查看启动的夜神模拟器列表

AAffA0nNPuCLAAAAAElFTkSuQmCC

可以看到已经启动了一个模拟器,占用了62001的端口。

3.Process类

如何在java中执行上述的指令呢?java中有一个类叫做Process,用于执行shell之类的指令,而且可以获得控制台的输出内容。public class App {

public static void main(String...args)throws Exception{

//执行指令

Process process = Runtime.getRuntime().exec("adb devices");

System.out.println("【控制台执行信息】");

System.out.println(readInputstream(process.getInputStream()));

System.out.println("【控制台错误信息】");

System.out.println(readInputstream(process.getErrorStream()));

}

/**

* 读取控制台打印的文字

* @param in

* @return

* @throws Exception

*/

public static String readInputstream(InputStream in) throws Exception{

InputStream inputStream = in;

InputStreamReader inputStreamReader = new InputStreamReader(inputStream);

BufferedReader bufferedReader = new BufferedReader(inputStreamReader);

StringBuilder sb = new StringBuilder();

String line = null;

while((line=bufferedReader.readLine())!=null){

sb.append(line).append("\n");

}

return sb.toString();

}

}

执行结果如下

AAffA0nNPuCLAAAAAElFTkSuQmCC

这是因为在程序入口下找不到adb.exe这个文件。我们只要把【adb】换成【C:\Program Files\Nox\bin\adb】就可以了。

执行结果如下

AAffA0nNPuCLAAAAAElFTkSuQmCC

还有一种执行指令的方式就比上述的方式更加简单了。

下面要讲的是ProcessBuilder这个类。我们将main函数修改成以下样子public static void main(String...args)throws Exception{

//执行指令

//        Process process = Runtime.getRuntime().exec("C:\\Program Files\\Nox\\bin\\adb devices");

//此处利用ProcessBuilder的command方法,

//值得注意的是command方法支持可变参数,

//也可以使用List,只要按指令空格分开好每个指令项就OK

Process process = new ProcessBuilder().command("C:\\Program Files\\Nox\\bin\\adb","devices").start();

System.out.println("【控制台执行信息】");

System.out.println(readInputstream(process.getInputStream()));

System.out.println("【控制台错误信息】");

System.out.println(readInputstream(process.getErrorStream()));

}

执行结果也是一样的。接下来我们也建一个Builder。

4.编写指令词典

可以看到,adb指令实在太多了。作为一个java程序员,是没必要去记住这些东西的,因为实际上我们并不常用这个东西。所以我们只好建一个指令词典了。指令词典的demo如下(我使用了lombok插件,自动生成Getter方法,不知道的同志们,请百度搜一下):package com.zeemoo.nox.actuator.consts;

import lombok.Getter;

/**

* 夜神模拟器的adb指令词典

*

* @author zhang.shushan

* @date 2018/6/8

*/

@Getter

public class NoxCmdDict {

/**

* adb脚本指令

*/

public final static String ADB = "adb";

/**

* 选择模拟器

*/

public final static String SERVER_HOST = "-s";

/**

* 模拟器列表

*/

public final static String DEVICES = "devices";

/**

* 安装apk

*/

public final static String INSTALL = "install";

/**

* 从电脑发送文件到模拟器,没什么卵用,还有权限限制

*/

@Deprecated

public final static String PUSH = "push";

/**

* 从模拟器拉取文件到电脑,没什么卵用,还有权限限制

*/

@Deprecated

public final static String PULL = "pull";

/**

* 卸载app

*/

public final static String UNINSTALL = "uninstall";

/**

* shell脚本指令,安卓内核为linux

*/

public final static String SHELL = "shell";

/**

* 列举进程

*/

public final static String SHELL_PS = "ps";

/**

* 包含某个字符串的进程信息

*/

public final static String SHELL_PS_MTH_FIND_STR ="|findStr";

/**

* 包指令

*/

public final static String SHELL_PM = "pm";

/**

* 包指令下的列举指令

*/

public final static String SHELL_PM_LIST = "list";

/**

* 包指令下的列举指令选项,表示列举所有的包名

*/

public final static String SHELL_PM_LIST_PACKAGES = "packages";

/**

* 包指令下的列举指令选项的附加选项,表示列举所有的包名和对应的路径

*/

public final static String SHELL_PM_LIST_PACKAGES_OP_NAME_AND_PATH = "-f";

/**

* 获取某个应用的路径,需要填写包名

*/

public final static String SHELL_PM_PATH = "path";

/**

* 清除应用缓存,后面接包名

*/

public final static String SHELL_PM_CLEAR_TEMP = "clear";

/**

* 应用管理指令(Activity Manager),启动或关闭应用

*/

public final static String SHELL_AM = "am";

/**

* 启动应用,最后接上包名+“/”+Activity类名

*/

public final static String SHELL_AM_START = "start";

/**

* 这个选项表示如果应用启动了就直接打开后台进程,如果没启动则启动。

* 不加此选项每次调用start的时候则每次都重启应用

*/

public final static String SHELL_AM_START_OP_XSTART = "-n";

/**

* 关闭应用,后接包名

*/

public final static String SHELL_AM_STOP = "force-stop";

/**

* 杀死进程,后接包名,似乎没什么用

*/

public final static String SHELL_AM_KILL = "kill";

/**

* 杀死后台所有进程,似乎没什么用

*/

public final static String SHELL_AM_KILL_ALL = "kill-all";

/**

* shell模拟输入

*/

public final static String SHELL_INPUT = "input";

/**

* 模拟输入文字(不支持中文)

*/

public final static String SHELL_INPUT_TEXT = "text";

/**

* 模拟点击按键

*/

public final static String SHELL_INPUT_KEYEVENT = "keyevent";

/**

* 模拟鼠标点击,后面接X,Y

*/

public final static String SHELL_INPUT_TAP = "tap";

/**

* 模拟鼠标滑动,后面接X1,Y1,X2,Y2

*/

public final static String SHELL_INPUT_SWIPE = "swipe";

/**

* 截屏

*/

public final static String SHELL_SCREEN_CAP = "screencap";

/**

* 截屏选项,后接截屏文件输出目标

*/

public final static String SHELL_SCREEN_CAP_OP_DEST = "-p";

/**

* 退出shell

*/

public final static String SHELL_EXIT = "exit";

/**

* 夜神模拟器的adb

*/

public final static String NOX_ADB = "nox_adb";

/**

* 夜神模拟器属性设置

*/

public final static String NOX_ADB_SHELL_SET_PROP = "setprop";

/**

* 设置手机的纬度

*/

public final static String NOX_ADB_SHELL_SET_PROP_LATITUDE = "persist.nox.gps.latitude";

/**

* 设置模拟器的经度

*/

public final static String NOX_ADB_SHELL_SET_PROP_LONGITUDE = "persist.nox.gps.longitude";

/**

* 设置模拟器的mac地址

*/

public final static String NOX_ADB_SHELL_SET_PROP_MAC = "setprop persist.nox.wifimac";

}

现在来创建一个Builder类,代码如下package com.zeemoo.nox.actuator.service;

import lombok.AllArgsConstructor;

import lombok.NoArgsConstructor;

import lombok.Setter;

import java.io.File;

import java.util.ArrayList;

import java.util.List;

//此处使用静态引入指令词典,可以试着删去这一行,看看有什么区别

import static com.zeemoo.nox.actuator.consts.NoxCmdDict.*;

/**

* 功能指令构建

*

* @author zhang.shushan

* @date 2018/6/8

*/

@Setter

@AllArgsConstructor

@NoArgsConstructor

public class NoxCmdBuilder {

/**

* 夜神模拟器的根目录

*/

private String noxPath;

/**

* 给指令添加指定模拟器

* @param cmds

* @param host

* @return

*/

public List addHost(List cmds,String host){

if(host!=null&&"".equals(host)){

cmds.add(1, SERVER_HOST);

cmds.add(2,host);

}

return cmds;

}

/**

* 筛选字符进程

* @param example

* @return

*/

public List listProcess(String example){

List cmds=listPackageName();

cmds.add(SHELL_PS_MTH_FIND_STR);

cmds.add(example);

return cmds;

}

/**

* 列举所有的进程

* @return

*/

public List listProcess(){

List cmds = this.shellModal();

cmds.add(SHELL_PS);

return cmds;

}

/**

* 列举所有模拟器

* @return

*/

public List devices(){

List cmds = new ArrayList<>();

cmds.add(noxPath+File.separator+ NOX_ADB);

cmds.add(DEVICES);

return cmds;

}

/**

* 启动夜神模拟器

* @return

*/

public String startNox(){

return noxPath+File.separator+"nox.exe";

}

/**

* 截屏

* @param vitualBoxFilePath

* @return

*/

public List screenCap(String vitualBoxFilePath){

List cmds = new ArrayList<>();

cmds.add(SHELL_SCREEN_CAP);

cmds.add(SHELL_SCREEN_CAP_OP_DEST);

cmds.add(vitualBoxFilePath);

return cmds;

}

/**

* 模拟滑动屏幕

* @param x1

* @param y1

* @param x2

* @param y2

* @return

*/

public List swipe(int x1,int y1,int x2,int y2){

List cmds = this.inputModal();

cmds.add(SHELL_INPUT_SWIPE);

cmds.add(String.valueOf(x1));

cmds.add(String.valueOf(y1));

cmds.add(String.valueOf(x2));

cmds.add(String.valueOf(y2));

return cmds;

}

/**

* 模拟点击

* @param x

* @param y

* @return

*/

public List click(int x,int y){

List cmds = this.inputModal();

cmds.add(SHELL_INPUT_TAP);

cmds.add(String.valueOf(x));

cmds.add(String.valueOf(y));

return cmds;

}

/**

* 模拟按键

* @param keys 按键

* @return

*/

public List pressKey(List keys){

List cmds = this.inputModal();

cmds.add(SHELL_INPUT_KEYEVENT);

cmds.addAll(keys);

return cmds;

}

/**

* 文本输入,不支持中文

* @param text

* @return

*/

public List inputText(String text){

List cmds = this.inputModal();

cmds.add(SHELL_INPUT_TEXT);

cmds.add(text);

return cmds;

}

/**

* 输入模式

* @return

*/

private List inputModal() {

List cmds = this.shellModal();

cmds.add(SHELL_INPUT);

return cmds;

}

/**

* 杀死后台所有进程

* @return

*/

public List killAllAppProcess(){

List cmds = this.activityManager();

cmds.add(SHELL_AM_KILL_ALL);

return cmds;

}

/**

* 杀死某个后台进程

* @param pkgNm

* @return

*/

public List killAppProcess(String pkgNm){

List cmds = this.activityManager();

cmds.add(SHELL_AM_KILL);

cmds.add(pkgNm);

return cmds;

}

/**

* 停止app

* @param pkgName

* @return

*/

public List stopApp(String pkgName){

List cmds = this.activityManager();

cmds.add(SHELL_AM_STOP);

cmds.add(pkgName);

return cmds;

}

/**

* 启动app

* @param pkgName

* @param activity

* @return

*/

public List startApp(String pkgName,String activity){

List cmds = this.activityManager();

cmds.add(SHELL_AM_START);

cmds.add(SHELL_AM_START_OP_XSTART);

cmds.add(pkgName+"/"+activity);

return cmds;

}

/**

* 获取某个应用的路径

* @param pkgName

* @return

*/

public List getAppPath(String pkgName){

List cmds = this.packageManager();

cmds.add(SHELL_PM_PATH);

cmds.add(pkgName);

return cmds;

}

/**

*

* @return

*/

public List listPackageNameAndPath(){

List cmds = this.listPackageName();

cmds.add(SHELL_PM_LIST_PACKAGES_OP_NAME_AND_PATH);

return cmds;

}

/**

* 列举所有的包名

* @return

*/

public List listPackageName(){

List cmds=this.packageManager();

cmds.add(SHELL_PM_LIST);

cmds.add(SHELL_PM_LIST_PACKAGES);

return cmds;

}

/**

* 包操作指令

* @return

*/

private List packageManager(){

List cmds = this.shellModal();

cmds.add(SHELL_PM);

return cmds;

}

/**

* activityManager模式

* @return

*/

private List activityManager(){

List cmds = this.shellModal();

cmds.add(SHELL_AM);

return cmds;

}

/**

* 从模拟器拉取文件到电脑,有权限限制

* @param virtualBoxFilePath

* @param pcFilePath

* @return

*/

@Deprecated

public List pull(String virtualBoxFilePath,String pcFilePath){

List cmds = new ArrayList<>();

cmds.add(noxPath+File.separator+virtualBoxFilePath);

cmds.add(PULL);

cmds.add(virtualBoxFilePath);

cmds.add(pcFilePath);

return cmds;

}

/**

* 推送文件到模拟器,有权限限制

* @param pcFilePath

* @param vitualBoxFilePath

* @return

*/

@Deprecated

public List push(String pcFilePath,String vitualBoxFilePath){

List cmds = new ArrayList<>();

cmds.add(noxPath+ File.separator+ NOX_ADB);

cmds.add(PUSH);

cmds.add(pcFilePath);

cmds.add(vitualBoxFilePath);

return cmds;

}

/**

* 卸载app

* @param packageName

* @return

*/

public List uninstall(String packageName){

List cmds = new ArrayList<>();

cmds.add(noxPath+ File.separator+ NOX_ADB);

cmds.add(UNINSTALL);

cmds.add(packageName);

return cmds;

}

/**

* 安装app

* @param apkPath

* @return

*/

public List install(String apkPath){

List cmds = new ArrayList<>();

cmds.add(noxPath+ File.separator+ NOX_ADB);

cmds.add(INSTALL);

cmds.add(apkPath);

return cmds;

}

/**

* 设置mac地址

* @param mac

* @return

*/

public List setMac(String mac){

List cmds = this.shellModal();

cmds.add(NOX_ADB_SHELL_SET_PROP);

cmds.add(NOX_ADB_SHELL_SET_PROP_MAC);

cmds.add(mac);

return cmds;

}

/**

* 设置经度

* @param longitude

* @return

*/

public List setLongitude(Double longitude){

List cmds = this.shellModal();

cmds.add(NOX_ADB_SHELL_SET_PROP);

cmds.add(NOX_ADB_SHELL_SET_PROP_LONGITUDE);

cmds.add(longitude.toString());

return cmds;

}

/**

* 设置模拟器纬度

* @param latitude

* @return

*/

public List setLatitude(Double latitude){

List cmds = this.shellModal();

cmds.add(NOX_ADB_SHELL_SET_PROP);

cmds.add(NOX_ADB_SHELL_SET_PROP_LATITUDE);

cmds.add(latitude.toString());

return cmds;

}

/**

* shell模式,设置模拟器属性之类的

* @return

*/

private List shellModal(){

List cmds = new ArrayList<>();

cmds.add(noxPath+ File.separator+ NOX_ADB);

cmds.add(SHELL);

return cmds;

}

/**

* shell模式,清除缓存

* @return

*/

public List clearTemp(String pckName) {

List cmds = this.shellModal();

cmds.add(SHELL_PM_CLEAR_TEMP);

cmds.add(pckName);

return cmds;

}

}

我们来测试一下。修改main方法如下:public static void main(String...args)throws Exception{

NoxCmdBuilder noxCmdBuilder = new NoxCmdBuilder("C:\\Program Files\\Nox\\bin");

System.out.println(noxCmdBuilder.listPackageName());

}

输出结果[C:\Program Files\Nox\bin\nox_adb, shell, pm, list, packages]

直接将这个list放入ProcessBuilder中使用,就可以获得所有的包名了。我们来试一下。修改main如下public static void main(String... args) throws Exception {

NoxCmdBuilder noxCmdBuilder = new NoxCmdBuilder("C:\\Program Files\\Nox\\bin");

Process start = new ProcessBuilder().command(noxCmdBuilder.listPackageName()).start();

//此处注意的是,读取流信息的时候,有可能流对象太大,

//不能一次性读完,导致获取的字符串顺序错乱或缺失的问题,

//所以我们等程序执行完毕之后再去读取

//waitFor会让线程阻塞,直至process执行完毕

start.waitFor();

System.out.println(readInputstream(start.getInputStream()));

}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/573748.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python从爬虫到数据分析项目_零基础学习Python web开发、Python爬虫、Python数据分析,从基础到项目实战!...

随着大数据和人工智能的发展&#xff0c;目前Python语言的上升趋势比较明显&#xff0c;而且由于Python语言简单易学&#xff0c;所以不少初学者往往也会选择Python作为入门语言。Python语言目前是IT行业内应用最为广泛的编程语言之一&#xff0c;尤其是近几年来随着大数据和人…

java filedialog 打开文件_java 用文件对话框打开文件

//文件的打开import java.awt.FileDialog;import java.awt.event.*;import java.io.*;import java.io.File;import java.io.FileReader;public class FileOpen {private FileDialog filedialog_open;private String fileopen null, filename null;// 用于存放打开文件地址 和…

2782: [HNOI2006]最短母串

2782: [HNOI2006]最短母串 Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 3 Solved: 2[Submit][Status][Web Board]Description 给定n个字符串&#xff08;S1,S2,„,Sn&#xff09;&#xff0c;要求找到一个最短的字符串T&#xff0c;使得这n个字符串&#xff08;S1,S2,„,…

c java 内部类_java程序中能否在内部类当中再定义一个内部类?

展开全部我被你的想62616964757a686964616fe78988e69d8331333363386664法震撼了,哈哈.亏你想的出来...这么弄代码不好理解,Java看起来醒目,也是Java中的一个规范!可以吗?必须可以..看代码演示...声明下,我也第一次,多次嵌套,看你想法后去试验下是可行的我用的两种办法!不多说看…

pythongui做计算器_python 实现简单的计算器(gui界面)

运行效果&#xff1a;完整代码from tkinter import *def click(num):global opopopstr(num)iptext.set(op)def evaluate():global opoutputstr(eval(op))iptext.set(output)def clearDisplay():global opop""iptext.set(op)calcTk()calc.title("TechVidvan Calc…

ios学习笔记——RunTime

Objective—C是面向运行时的语言&#xff0c;就是说它会尽可能的把编译和链接时要执行的逻辑延迟到运行时。这就给你很大的灵活性&#xff0c;你可以按需要把消息重定向给合适的对象&#xff0c;你甚至可以交换方法的实现。最重要的还是消息机制。 C语言使用“静态绑定”&#…

exhaustion java_Java Player.setExhaustion方法代碼示例

import org.bukkit.entity.Player; //導入方法依賴的package包/類/*** Set SamaGamesAPI*/Overridepublic void startGame(){this.getInGamePlayers().forEach(((uuid1, survivalPlayer) -> survivalPlayer.getPlayerIfOnline().closeInventory()));super.startGame();Objec…

python内存池机制_python的内存管理机制

一、python是一个什么样类型的语言1、python是一种动态解释性强类型定义的高级、通用性编程语言。解释型&#xff1a;执行的时候&#xff0c;才一条一条的解释成机器语言给计算机来执行。如&#xff1a;python、js、ruby、PHP等编译型&#xff1a;把源程序的每一条语句都编译成…

MATLAB GUI不同控件函数间变量传递方法

在GUI中一个控件函数结束后&#xff0c;会将在这个函数中使用的变量全部删除&#xff0c;如果想在另一个控件中使用&#xff0c;则必须想办法将这个变量保存传递&#xff0c;方法有以下几种&#xff1a;1、使用globe定义全局变量&#xff0c;这种方法不是很好&#xff0c;一方面…

java变量数据类型_java变量与数据类型

第二章 变量与数据类型变量变量的概述变量是指内存中的一个存储区域&#xff0c;该区域要有自己的名称(变量名)、类型(数据类型)&#xff0c;该区域的数据可以在同一数据类型的范围内不断变化值。每个变量都有自己的作用范围&#xff0c;叫作用域变量的注意事项1.Java中的变量必…

python实现推荐系统代码_推荐系统之矩阵分解及其Python代码实现

有如下R(5,4)的打分矩阵&#xff1a;(“-”表示用户没有打分)其中打分矩阵R(n,m)是n行和m列&#xff0c;n表示user个数&#xff0c;m行表示item个数那么&#xff0c;如何根据目前的矩阵R(5,4)如何对未打分的商品进行评分的预测(如何得到分值为0的用户的打分值)&#xff1f;——…

数据库分区分表以及读写分离

谈谈怎么实现Oracle数据库分区表 Oracle数据库分区是作为Oracle数据库性能优化的一种重要的手段和方法&#xff0c;做手头的项目以前&#xff0c;只聆听过分区的大名&#xff0c;感觉特神秘&#xff0c;看见某某高手在讨论会上夸夸其谈时&#xff0c;真是骂自己学艺不精&#x…

java第二章_JAVA第二章知识点

JAVA第二章知识点本章知识梳理2.1 关键字2.2 标识符2.3 变 量2.4运算符2.5 程序流程控制2.6 方法2.1 关键字关键字(keyword)的定义和特点定义&#xff1a;被java语言赋予了特殊含义&#xff0c;用做专门用途的字符串(单词)特点&#xff1a;关键字中所有字母都为小写保留字(rese…

JSP Workshop

http://www.cnblogs.com/ITtangtang/p/4126395.html 发现http://www.tutorialspoint.com/里的资料很全也很不错啊&#xff01; 资料&#xff1a;http://www.tutorialspoint.com/jsp/jsp_tutorial.pdf 另外&#xff0c;http://www.runoob.com/jsp/jsp-tutorial.html 中关于JSP…

崇天老师python123测验6_嵩天老师python123测验1: Python基本语法元素 (第1周)

选择题**Guido van Rossum正式对外发布Python版本的年份是&#xff1a;‪‬‪‬‪‬‪‬‪‬‮‬‪‬‫‬‪‬‪‬‪‬‪‬‪‬‮‬‪‬‮‬‪‬‪‬‪‬‪‬‪‬‮‬‫‬‫‬‪‬‪‬‪‬‪‬‪‬‮‬‪‬‪‬‪‬‪‬‪‬‪‬‪‬‮‬‭‬‫‬‪‬‪‬‪‬‪‬‪‬‮‬‫‬‪…

从链接中获取文件名及扩展名

exeStr [filePath lastPathCompoment];   从链接中读取完整的带后缀文件名称 exeStr [filePath stringByDeleteExtension]; 从链接中读取不带后缀名的文件名称 fileType [filePath pathExtension];       获得链接中文件扩展名&#xff0c;不带"." …

mysql的学习要点_MySQL中的联合索引的学习要点总结

MySQL中的联合索引的学习要点总结联合索引又叫复合索引。对于复合索引:Mysql从左到右的使用索引中的字段&#xff0c;一个查询可以只使用索引中的一部份&#xff0c;但只能是最左侧部分。例如索引是key index (a,b,c). 可以支持a | a,b| a,b,c 3种组合进行查找&#xff0c;但不…

h5页面不可 移动_H5营销|为什么H5适合于微信营销

随着互联网技术的不断发展&#xff0c;更新在移动互联网时代&#xff0c;网络营销也开始越来越新颖化&#xff0c;而微信H5就是其中的佼佼者。它的出现满足了用户视觉上的审美要求&#xff0c;并且可以使营销方式变得更加的美观整洁&#xff0c;那么这里就有一个问题。为什么微…

input框

input: 输入时与输入框有点距离 padding-left: 5px; 点击时没有浅蓝色的框&#xff1a; outline: none; 设置输入框背景颜色&#xff1a;background: seagreen; 提示语&#xff1a; placeholder"请输入搜索内容" 放在HTML 输入框有图像&#xff1a; background-image…

python3 image_python3 ImageTk 安装方法

标签&#xff1a;使用命令&#xff1a;$ sudo yum search PIL | grep python3可显示得知&#xff1a;python3-dogpile-cache.noarch : A caching front-end based on the Dogpile lock.python3-dogpile-core.noarch : A ‘dogpile‘ lock, typically used as a component ofpyt…