linux sh 脚本调用外部命令

bash shell的命令分为两类:外部命令和内部命令。外部命令是通过系统调用或独立的程序实现的,如sed、awk等等。内部命令是由特殊的文件格式(.def)所实现,如cd、history、exec等等。

fork ( /directory/script.sh)

首先,我們所執行的任何程式,都是由父行程(parent process)所產生出來的一個子行程(child process),子行程在結束後,將返回到父行程去。此一現像在 Linux 系統中被稱為   fork 。

當子行程被產生的時候,將會從父行程那裡獲得一定的資源分配、及(更重要的是)繼承父行程的環境。


* 所謂環境變量其實就是那些會傳給子行程的變量。
簡單而言,"遺傳性"就是區分本地變量與環境變量的決定性指標。
然而,從遺傳的角度來看,我們也不難發現環境變量的另一個重要特徵:
* 環境變量只能從父行程到子行程單向繼承。換句話說:在子行程中的環境如何變更,均不會影響父行程的環境。

fork是最普通的, 就是直接在脚本里面用/directory/script.sh来调用script.sh这个脚本.

运行的时候开一个sub-shell执行调用的脚本,sub-shell执行的时候, parent-shell还在。

sub-shell执行完毕后返回parent-shell. sub-shell从parent-shell继承环境变量.但是sub-shell中的环境变量不会带回parent-shell。子进程与父进程唯一不同的地方在于pid(process id)。

exec (exec /directory/script.sh)

exec 也是讓 script 在同一個行程上執行,但是原有行程則被結束了。


也就是簡而言之:原有行程會否終止,就是 exec 與 source/fork 的最大差異了。

exec命令在执行时会把当前的shell process关闭,然后换到后面的命令继续执行(原线程被干掉,建立一个新线程???right?)

source (source /directory/script.sh)

* 所謂 source 就是讓 script 在當前 shell 內執行、而不是產生一個 sub-shell 來執行。


由於所有執行結果均於當前 shell 內完成,若 script 的環境有所改變,當然也會改變當前環境了﹗

 

与fork的区别是不新开一个sub-shell来执行被调用的脚本,而是在同一个shell中执行. 所以被调用的脚本中声明的变量和环境变量, 都可以在主脚本中得到和使用.


可以通过下面这两个脚本来体会三种调用方式的不同:

1.sh 

  1. #!/bin/bash
  2. A=B
  3. echo "PID for 1.sh before exec/source/fork:$$"

export A

  1. echo "1.sh: /$A is $A"
  2. case $1 in
  3.        exec)
  4.             echo "using exec..."
  5.             exec ./2.sh ;;
  6.        source)
  7.             echo "using source..."
  8.             . ./2.sh ;;
  9.        *)
  10.             echo "using fork by default..."
  11.             ./2.sh ;;
  12. esac
  13. echo "PID for 1.sh after exec/source/fork:$$"
  14. echo "1.sh: /$A is $A"

2.sh

  1.  
    1. #!/bin/bash
    2. echo "PID for 2.sh: $$"
    3. echo "2.sh get /$A=$A from 1.sh"
    4. A=C
    5. export A
    6. echo "2.sh: /$A is $A"

 

运行以下3个脚本来观察结果:

1。$ ./1.sh fork
2。$ ./1.sh source
3。$ ./1.sh exec

Linux系统中部署WAR包并让其调用外部Shell脚本是一种常见的需求,特别是在需要执行复杂的系统级操作或与现有基础设施集成时。以下是实现这一过程的关键步骤: ### 1. 确保权限适当 首先,确认运行Web应用程序的用户有足够的权限去读取、执行目标shell脚本,并且能够访问相关的文件夹及依赖项。 ```bash chmod +x /path/to/script.sh # 给予脚本可执行权限 ``` ### 2. 使用ProcessBuilder 或 Runtime 类调用 Shell 脚本 #### (a) ProcessBuilder 方式: 这是推荐的方式之一因为它更灵活也更容易管理错误流等信息。 ```java import java.io.BufferedReader; import java.io.InputStreamReader; public class ExecuteScript { public static void main(String[] args){ try{ String command = "/bin/bash /path/to/your-script.sh"; // 替换为您的实际路径 // 创建一个新的进程生成器实例 ProcessBuilder processBuilder = new ProcessBuilder(); processBuilder.command("sh", "-c", command); // 启动进程并获取输入输出流用于监控结果 Process process = processBuilder.start(); BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream())); StringBuilder builder = new StringBuilder(); String line; while ((line = reader.readLine()) != null) { builder.append(line).append("\n"); } int exitCode = process.waitFor(); System.out.println("Exit Code: " + exitCode); }catch(Exception e){ e.printStackTrace(); } } } ``` #### (b) Runtime.exec() 方法: 虽然也可以使用`Runtime.getRuntime().exec()`方法来启动另一个进程,但它相对简单而且灵活性较差。 ```java String cmd ="/path/to/your-script.sh"; Process proc= Runtime.getRuntime().exec(cmd); // ... similar handling as above for reading outputs etc. proc.waitFor(); System.out.println(proc.exitValue()); ``` 注意:无论是哪种方式都应妥善处理可能产生的异常以及考虑超时机制以防某些长期未完成的任务阻塞主线程。 --- 如果需要传递参数给脚本只需修改命令字符串部分即可,例如 `command="/bin/bash -s arg1 arg2 < scriptfile"` 这种形式传参过去。 最后别忘了测试各种边界条件包括但不限于脚本失败返回值检查等等!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值