java调用shell脚本想要error,info等信息的注意点

7 篇文章 0 订阅

情景1:

源头java调1.sh脚本,1.sh脚本中会启动java程序

A. java调用1.sh的脚本不管带不带“nohup”或者"nohup ... > 2>&1 ",java的error和info管道都是由1.sh中启动java的命令行的写法决定的

B. 基于A,1.sh中启动java的命令由下列注意点:

1. 想要通过process的errorStream和inputStream获取对应的调用信息,那么在目标sh命令中就不要带"nohup ... > 2>&1",也不要"nohup ... > nohup.out",因为这些都会把信息输出到指定位置(可以理解前者是特殊的一个位置而已)。

然而,"nohup ..."是可以的。

2. 但是基于1,1.sh脚本中的jar启动命令不写nohup了,jar启动后的所有logger输出却也会待会给源头java的info通道中,导致源头java处获取并输出(如果让输出了的话)一大片日志。

3. 为了2,就把“nohup... > /dev/null 2>&1”加上了,之后没有了2的大片日志。如果1.sh中的脚本层面有问题,error通道还是可以收到错误信息的。

4. 类似与1,如果在1.sh中启动jar的命令结尾追加了“&”(除了这个尾巴处由个'&',其它地方完全的java -jar a.jar的标准写法),info和error都是收不到信息的

乱序记录(仅仅个人看的):

1. java1启动1.sh后设置了两个futuretask

用于监听info和error通道并接受数据 -> 1.sh启动java2程 -> java2每隔2s输出一次info日志。于是java1程序每间隔2s输出一次收到的信息(说明info通道一直在输出),但是一直没有报错,所以error一直在阻塞;info在收到信息中匹配到关键字的话,会调用proces的destroyForcibly()方法取消process的wait,wait醒后下面的代码是调用info和erro通道的futureTask的interrupt函数用来停止这两个监听线程。首先,info和error中的while是进入下一次循环先执行还是wait下面的interrupt限制性的顺序是没有办法保证的(理由见2,当然除非使用sleep等手段),但是现象是:info监听的task可以停止,而error的没有停止。

原因:首先两个监听进程能不停的监听是因为有个while循环,循环条件是readLine和thread.isInterrupted,如下:

while ( !Thread.currentThread().isInterrupted() && (line = bufferedReader.readLine())!=null) {
...

 }

interrupted写在前边是为了能在每次进入while时先进行interrupted的判断,感知是否被futuTask调用过inteerrupted函数,避免了readLine在前出现的永久会阻塞等待没有机会判断interrupted而导致的不能退出while循环的问题。

bufferreader的readline函数阻塞,而info监听者的while循环中的一个条件:reader.readline()!=null这一行每隔2s就能往下运行,因此最迟2s就能进入下一次while循环,而此时while的另一个条件:Thread.currentThread().isInterrupted()函数就会使得线程结束。

而error由于一直阻塞在readLine上,因此一直没有机会去执行Thread.currentThread().isInterrupted()的判断,所以会一直占用线程,导致最后main线程和info线程退出了,可它却还在,所以java1程序一直在占用系统进程。

2. java调用脚本后返回的proces对象,可以执行destroyForcibly()方法取消waitFor()。我的一个实验是:process在wait,同时新开了两个线程分别监听info和error通道,其中info通道收到关键字后,会想要打断process的wait,因此我给那两个线程中出传入了process引用,从而可以打断。不过,这里需要注意的就是,process被打断wait后,当前cpu不一定会立马执行waitFor下面的代码,还有可能执行info或者error通道线程中的代码的,因为这是三个线程嘛,一定要注意。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java调用Shell脚本并传参的步骤如下: 1. 使用Java的ProcessBuilder类创建一个进程,指定要执行的Shell脚本文件路径。 2. 通过ProcessBuilder类的command()方法设置Shell脚本的参数,可以使用数组或者List来传递参数。 3. 调用ProcessBuilder类的start()方法启动进程。 4. 使用Process类的getInputStream()、getErrorStream()和getOutputStream()方法获取进程的输入、错误和输出流。 5. 使用Java IO类来读取和写入进程的输入、错误和输出流。 6. 调用Process类的waitFor()方法等待进程执行完毕。 7. 使用Process类的exitValue()方法获取进程的退出值。 示例代码如下: ``` import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.util.ArrayList; import java.util.List; public class ShellExecutor { public static void main(String[] args) throws IOException, InterruptedException { String scriptPath = "/path/to/script.sh"; List<String> params = new ArrayList<>(); params.add("param1"); params.add("param2"); ProcessBuilder processBuilder = new ProcessBuilder(scriptPath); processBuilder.command().addAll(params); Process process = processBuilder.start(); BufferedReader inputReader = new BufferedReader(new InputStreamReader(process.getInputStream())); BufferedReader errorReader = new BufferedReader(new InputStreamReader(process.getErrorStream())); String line; while ((line = inputReader.readLine()) != null) { System.out.println(line); } while ((line = errorReader.readLine()) != null) { System.err.println(line); } int exitCode = process.waitFor(); System.out.println("Exit code: " + exitCode); } } ``` ### 回答2: Java调用Shell脚本并传参的过程可以分为以下几个步骤: 第一步:需要先定义一个Shell脚本,并在其中定义需要传入的参数。比如定义一个名为test.sh的Shell脚本: #!/bin/bash echo "接收到的参数为:$1" 第二步:在Java中使用ProcessBuilder调用Shell脚本,并设置传入的参数。示例代码如下: public class TestShell { public static void main(String[] args) throws IOException, InterruptedException { List<String> commands = new ArrayList<>(); commands.add("/bin/bash"); commands.add("./test.sh"); commands.add("hello"); ProcessBuilder builder = new ProcessBuilder(commands); builder.directory(new File("./")); Process process = builder.start(); InputStream is = process.getInputStream(); BufferedReader br = new BufferedReader(new InputStreamReader(is)); String line; while ((line = br.readLine()) != null) { System.out.println(line); } br.close(); process.waitFor(); } } 第三步:运行Java代码,控制台输出如下信息: 接收到的参数为:hello 程序运行结束。 可以看到,Java程序成功调用Shell脚本并传入了参数hello。实际开发中,需要按照实际需求编写相应的脚本Java代码。需要注意的是,在调用Shell脚本时,需要设置正确的权限以避免出现权限问题。 ### 回答3: Java是一种广泛应用于企业级应用程序开发的高级编程语言,而Shell脚本则是一种在Unix和Linux等操作系统中常用的脚本语言。当我们需要在Java应用程序中调用Shell脚本时,通常需要了解如何传递参数以及如何正确执行Shell脚本。 首先,Java调用Shell脚本的过程可以通过使用Runtime类中的exec()方法来实现,该方法可以执行系统命令并返回Process对象,从而使我们可以使用Java代码来控制该过程中的输入和输出。此外,使用ProcessBuilder类也是很常见的一种方法,它允许我们更加灵活地控制Shell脚本的命令行和参数。 在实际使用中,通过Java调用Shell脚本时,常常需要传递一些参数。这可以通过在exec()方法或ProcessBuilder对象中指定Shell脚本的命令行参数来实现。例如,对于以下Shell脚本: #!/bin/bash echo "Hello, $1!" 我们可以通过以下Java代码来调用脚本并传递参数: // 使用Runtime类调用Shell脚本并传递参数 String[] command = {"/bin/bash", "-c", "sh /path/to/script.sh John"}; Process process = Runtime.getRuntime().exec(command); // 使用ProcessBuilder类调用Shell脚本并传递参数 ProcessBuilder pb = new ProcessBuilder("/bin/bash", "-c", "sh /path/to/script.sh John"); Process process = pb.start(); 在这两种情况下,我们都通过命令行参数将"John"传递给了Shell脚本。 在执行Shell脚本时会返回一个Process对象,该对象提供了许多方法来控制执行过程中的输入和输出流,并可以在脚本执行完成后获得脚本的执行结果。例如,我们可以使用Process对象的getInputStream()方法来获得Shell脚本的标准输出流,并使用BufferedReader类来读取输出的内容。 总之,Java调用Shell脚本并传参是一个常见的需求,参考以上的代码和方法,我们可以轻松实现这个过程。值得注意的是,在传递参数时应该注意参数的正确性和安全性,以避免系统命令执行错误或命令注入等风险。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值