linux命令:java调用shell脚本与shell脚本调用java程序
admin
2024-03-17 20:01:03
0

一、java调用shell脚本

参考自:《如何使用JAVA调用SHELL》
1、参数说明
(1)RUNNING_SHELL_FILE:要运行的脚本。
(2)SHELL_FILE_DIR:要运行的脚本所在的目录; 当然你也可以把要运行的脚本写成全路径。
(3)runningStatus:运行状态,0标识正常。 详细可以看java文档。
(4)param1, param2, param3:可以在RUNNING_SHELL_FILE脚本中直接通过$1,$2,$3分别拿到的参数。
2、通过ProcessBuilder进行调度

ProcessBuilder builder = new ProcessBuilder("./" + RUNNING_SHELL_FILE, param1, param2, param3);
builder.directory(new File(SHELL_FILE_DIR));//更改我们的 shell 命令正在运行的工作目录
builder.environment();//将自定义键值映射设置为环境
builder.inheritIO();//将它们都继承到当前JVM进程的流中
int runningStatus = 0;
String s = null;
try {Process p = builder.start();try {runningStatus = p.waitFor();//因为shell进程是JAVA进程的子进程,JAVA作为父进程需要等待子进程执行完毕。} catch (InterruptedException e) {}
} catch (IOException e) {
}
if (runningStatus != 0) {
}
return;

优点:
  可以自定义一些细节。
(1)使用builder.directory()更改我们的 shell 命令正在运行的工作目录
(2)使用builder.environment()将自定义键值映射设置为环境
(3)使用builder.inheritIO()将它们都继承到当前JVM进程的流中
注意:
  在调用时需要执行waitFor()函数,因为shell进程是JAVA进程的子进程,JAVA作为父进程需要等待子进程执行完毕。
3、直接通过系统Runtime执行shell

int runningStatus = 0;
String s = null;
try {Process p = Runtime.getRuntime().exec(SHELL_FILE_DIR + RUNNING_SHELL_FILE + " "+param1+" "+param2+" "+param3);runningStatus = p.waitFor();//因为shell进程是JAVA进程的子进程,JAVA作为父进程需要等待子进程执行完毕。
} catch (IOException e) {
}
if (runningStatus != 0) {
}
return;

通过Runtime的方式并没有builder那么方便,特别是参数方面,必须自己加空格分开,因为exec会把整个字符串作为shell运行。
4、问题
(1)没权限运行
1)描述
  在做DTS迁移的过程中,要执行包里面的shell脚本, 解压出来了之后,发现执行不了。
2)解决

ProcessBuilder builder = new ProcessBuilder("/bin/chmod", "755", tempFile.getPath());
Process process = builder.start();
int rc = process.waitFor();

(2)java进行一直等待shell返回
1)描述
  shell脚本中有echo或者print输出, 导致缓冲区被用完了! 为了避免这种情况, 一定要把缓冲区读一下, 好处就是,可以对shell的具体运行状态进行log出来。
2)解决

ProcessBuilder pb = new ProcessBuilder("./" + RUNNING_SHELL_FILE, keyword.trim(), taskId.toString(), fileName);
pb.directory(new File(CASPERJS_FILE_DIR));
int runningStatus = 0;
String s = null;
try {Process p = pb.start();BufferedReader stdInput = new BufferedReader(new InputStreamReader(p.getInputStream()));BufferedReader stdError = new BufferedReader(new InputStreamReader(p.getErrorStream()));while ((s = stdInput.readLine()) != null) {LOG.error(s);}while ((s = stdError.readLine()) != null) {LOG.error(s);}try {runningStatus = p.waitFor();} catch (InterruptedException e) {}
} catch (IOException e) {
}

注意:
  在start()之后, waitFor()之前把缓冲区读出来打log, 就可以看到你的shell为什么会没有按照预期运行。 这个还有一个好处是,可以读shell里面输出的结果, 方便java代码进一步操作。

二、shell脚本调用java程序

参考自:《Linux 下使用Shell脚本启动Java应用》
1、项目部署

[root@xxx yyy]# unzip -oq Linux_Java_Shell.war -d /usr/local/Linux_Java_Shell

解析:
  不询问将war包传到/usr/local/Linux_Java_Shell进行解压。
-o:不必先询问用户,unzip执行后覆盖原有文件。
-q:执行时不显示任何信息。
-d <目录>:指定文件解压缩后所要存储的目录。
2、脚本启动

#!/bin/sh
#该脚本为Linux下启动java程序的通用脚本。既可以作为开机自启动service脚本被调用,
#也可以作为启动java程序的独立脚本来使用。
#
#警告!!!:该脚本stop部分使用系统kill命令来强制终止指定的java程序进程。
#在杀死进程前,未作任何条件检查。在某些情况下,如程序正在进行文件或数据库写操作,
#可能会造成数据丢失或数据不完整。如果必须要考虑到这类情况,则需要改写此脚本,
#增加在执行kill命令前的一系列检查。
#
#
###################################
#环境变量及程序执行参数
#需要根据实际环境以及Java程序名称来修改这些参数
###################################
#JDK所在路径
JAVA_HOME="/usr/local/jdk1.8.0_161/"#执行程序启动所使用的系统用户,考虑到安全,推荐不使用root帐号
RUNNING_USER=root#Java程序所在的目录(classes的上一级目录)
APP_HOME=/usr/local/Linux_Java_Shell/WEB-INF/#需要启动的Java主程序(main方法类)
APP_MAINCLASS=com.lpf.Test#拼凑完整的classpath参数,包括指定lib目录下所有的jar
CLASSPATH=$APP_HOME/classes
for i in "$APP_HOME"/lib/*.jar; doCLASSPATH="$CLASSPATH":"$i"
done#java虚拟机启动参数
JAVA_OPTS="-ms512m -mx512m -Xmn256m -Djava.awt.headless=true -XX:MaxPermSize=128m"###################################
#(函数)判断程序是否已启动
#
#说明:
#使用JDK自带的JPS命令及grep命令组合,准确查找pid
#jps 加 l 参数,表示显示java的完整包路径
#使用awk,分割出pid ($1部分),及Java程序名称($2部分)
###################################
#初始化psid变量(全局)
psid=0checkpid() {javaps=`$JAVA_HOME/bin/jps -l | grep $APP_MAINCLASS`if [ -n "$javaps" ]; thenpsid=`echo $javaps | awk '{print $1}'`elsepsid=0fi
}###################################
#(函数)启动程序
#
#说明:
#1. 首先调用checkpid函数,刷新$psid全局变量
#2. 如果程序已经启动($psid不等于0),则提示程序已启动
#3. 如果程序没有被启动,则执行启动命令行
#4. 启动命令执行后,再次调用checkpid函数
#5. 如果步骤4的结果能够确认程序的pid,则打印[OK],否则打印[Failed]
#注意:echo -n 表示打印字符后,不换行
#注意: "nohup 某命令 >/dev/null 2>&1 &" 的用法
###################################
start() {checkpidif [ $psid -ne 0 ]; thenecho "================================"echo "warn: $APP_MAINCLASS already started! (pid=$psid)"echo "================================"elseecho -n "Starting $APP_MAINCLASS ..."JAVA_CMD="nohup $JAVA_HOME/bin/java $JAVA_OPTS -classpath $CLASSPATH $APP_MAINCLASS >/dev/null 2>&1 &"su - $RUNNING_USER -c "$JAVA_CMD"checkpidif [ $psid -ne 0 ]; thenecho "(pid=$psid) [OK]"elseecho "[Failed]"fifi
}###################################
#(函数)停止程序
#
#说明:
#1. 首先调用checkpid函数,刷新$psid全局变量
#2. 如果程序已经启动($psid不等于0),则开始执行停止,否则,提示程序未运行
#3. 使用kill -9 pid命令进行强制杀死进程
#4. 执行kill命令行紧接其后,马上查看上一句命令的返回值: $?
#5. 如果步骤4的结果$?等于0,则打印[OK],否则打印[Failed]
#6. 为了防止java程序被启动多次,这里增加反复检查进程,反复杀死的处理(递归调用stop)。
#注意:echo -n 表示打印字符后,不换行
#注意: 在shell编程中,"$?" 表示上一句命令或者一个函数的返回值
###################################
stop() {checkpidif [ $psid -ne 0 ]; thenecho -n "Stopping $APP_MAINCLASS ...(pid=$psid) "su - $RUNNING_USER -c "kill -9 $psid"if [ $? -eq 0 ]; thenecho "[OK]"elseecho "[Failed]"ficheckpidif [ $psid -ne 0 ]; thenstopfielseecho "================================"echo "warn: $APP_MAINCLASS is not running"echo "================================"fi
}###################################
#(函数)检查程序运行状态
#
#说明:
#1. 首先调用checkpid函数,刷新$psid全局变量
#2. 如果程序已经启动($psid不等于0),则提示正在运行并表示出pid
#3. 否则,提示程序未运行
###################################
status() {checkpidif [ $psid -ne 0 ];  thenecho "$APP_MAINCLASS is running! (pid=$psid)"elseecho "$APP_MAINCLASS is not running"fi
}###################################
#(函数)打印系统环境参数
###################################
info() {echo "System Information:"echo "****************************"echo `head -n 1 /etc/issue`echo `uname -a`echoecho "JAVA_HOME=$JAVA_HOME"echo `$JAVA_HOME/bin/java -version`echoecho "APP_HOME=$APP_HOME"echo "APP_MAINCLASS=$APP_MAINCLASS"echo "****************************"
}###################################
#读取脚本的第一个参数($1),进行判断
#参数取值范围:{start|stop|restart|status|info}
#如参数不在指定范围之内,则打印帮助信息
###################################
case "$1" in'start')start;;'stop')stop;;'restart')stopstart;;'status')status;;'info')info;;
esacecho "Usage: $0 {start|stop|restart|status|info}"exit 1

注意:
nohup命令-不挂断地运行命令。

上一篇:go的基本数据类型

下一篇:redis教程

相关内容

热门资讯

【MySQL】锁 锁 文章目录锁全局锁表级锁表锁元数据锁(MDL)意向锁AUTO-INC锁...
【内网安全】 隧道搭建穿透上线... 文章目录内网穿透-Ngrok-入门-上线1、服务端配置:2、客户端连接服务端ÿ...
GCN的几种模型复现笔记 引言 本篇笔记紧接上文,主要是上一篇看写了快2w字,再去接入代码感觉有点...
数据分页展示逻辑 import java.util.Arrays;import java.util.List;impo...
Redis为什么选择单线程?R... 目录专栏导读一、Redis版本迭代二、Redis4.0之前为什么一直采用单线程?三、R...
【已解决】ERROR: Cou... 正确指令: pip install pyyaml
关于测试,我发现了哪些新大陆 关于测试 平常也只是听说过一些关于测试的术语,但并没有使用过测试工具。偶然看到编程老师...
Lock 接口解读 前置知识点Synchronized synchronized 是 Java 中的关键字,...
Win7 专业版安装中文包、汉... 参考资料:http://www.metsky.com/archives/350.htm...
3 ROS1通讯编程提高(1) 3 ROS1通讯编程提高3.1 使用VS Code编译ROS13.1.1 VS Code的安装和配置...
大模型未来趋势 大模型是人工智能领域的重要发展趋势之一,未来有着广阔的应用前景和发展空间。以下是大模型未来的趋势和展...
python实战应用讲解-【n... 目录 如何在Python中计算残余的平方和 方法1:使用其Base公式 方法2:使用statsmod...
学习u-boot 需要了解的m... 一、常用函数 1. origin 函数 origin 函数的返回值就是变量来源。使用格式如下...
常用python爬虫库介绍与简... 通用 urllib -网络库(stdlib)。 requests -网络库。 grab – 网络库&...
药品批准文号查询|药融云-中国... 药品批文是国家食品药品监督管理局(NMPA)对药品的审评和批准的证明文件...
【2023-03-22】SRS... 【2023-03-22】SRS推流搭配FFmpeg实现目标检测 说明: 外侧测试使用SRS播放器测...
有限元三角形单元的等效节点力 文章目录前言一、重新复习一下有限元三角形单元的理论1、三角形单元的形函数(Nÿ...
初级算法-哈希表 主要记录算法和数据结构学习笔记,新的一年更上一层楼! 初级算法-哈希表...
进程间通信【Linux】 1. 进程间通信 1.1 什么是进程间通信 在 Linux 系统中,进程间通信...
【Docker】P3 Dock... Docker数据卷、宿主机与挂载数据卷的概念及作用挂载宿主机配置数据卷挂载操作示例一个容器挂载多个目...