1. 程式人生 > >Java呼叫Shell命令和指令碼

Java呼叫Shell命令和指令碼

1.介紹

有時候我們在Linux中執行Java程式時,需要呼叫一些Shell命令和指令碼。而Runtime.getRuntime().exec()方法給我們提供了這個功能,而且Runtime.getRuntime()給我們提供了以下幾種exec()方法:

Process exec(String command) 
在單獨的程序中執行指定的字串命令。 

Process exec(String[] cmdarray) 
在單獨的程序中執行指定命令和變數。 

Process exec(String[] cmdarray, String[] envp) 
在指定環境的獨立程序中執行指定命令和變數。 

Process exec(String[] cmdarray, String[] envp, File dir) 
在指定環境和工作目錄的獨立程序中執行指定的命令和變數。 

Process exec(String command, String[] envp) 
在指定環境的單獨程序中執行指定的字串命令。 

Process exec(String command, String[] envp, File dir) 
在有指定環境和工作目錄的獨立程序中執行指定的字串命令。 

其中,其實cmdarray和command差不多,同時如果引數中如果沒有envp引數或設為null,表示呼叫命令將在當前程式執行的環境中執行;如果沒有dir引數或設為null,表示呼叫命令將在當前程式執行的目錄中執行,因此呼叫到其他目錄中的檔案和指令碼最好使用絕對路徑。各個引數的含義:
  1. cmdarray: 包含所呼叫命令及其引數的陣列。 
  2. command: 一條指定的系統命令。
  3. envp: 字串陣列,其中每個元素的環境變數的設定格式為name=value;如果子程序應該繼承當前程序的環境,則該引數為 null。
  4. dir: 子程序的工作目錄;如果子程序應該繼承當前程序的工作目錄,則該引數為 null。 

細心的讀者會發現,為了執行呼叫操作,JVM會啟一個Process,所以我們可以通過呼叫Process類的以下方法,得知呼叫操作是否正確執行:

abstract  int waitFor() 
導致當前執行緒等待,如有必要,一直要等到由該 Process 物件表示的程序已經終止。 

程序的出口值。根據慣例,0 表示正常終止;否則,就表示異常失敗。

另外,呼叫某些Shell命令或指令碼時,會有返回值,那麼我們如果捕獲這些返回值或輸出呢?為了解決這個問題,Process類提供了:

abstract InputStream  getInputStream() 
獲取子程序的輸入流。 最好對輸入流進行緩衝。

2.呼叫Shell命令

這裡為了說明問題,我僅用tar命令進行演示。tar命令是一個打包而不進行壓縮的命令。同時,為了檢查tar的呼叫是否被正常執行,我將呼叫waitFor()方法。

private void callCMD(String tarName, String fileName, String... workspace){
	try {
		String cmd = "tar -cf" + tarName + " " + fileName;
//            String[] cmd = {"tar", "-cf", tarName, fileName};
		File dir = null;
		if(workspace[0] != null){
			dir = new File(workspace[0]);
			System.out.println(workspace[0]);
		}
		process = Runtime.getRuntime().exec(cmd, null, dir);
//          process = Runtime.getRuntime().exec(cmd);
		int status = process.waitFor();
		if(status != 0){
			System.err.println("Failed to call shell's command and the return status's is: " + status);
		}
	}
	catch (Exception e){
		e.printStackTrace();
	}
}

注意:如果把命令放到一個String[]中時,必須把命令中每個部分作為一個元素存在String[]中,或者是把命令按照空格符分割得到的String[]。
String[] cmd = {"tar", "-cf", tarName, fileName};		//right
String[] cmd = {"tar -cf", tarName, fileName};			//error

為了說明dir引數的作用,我特地把該Java程式和要打包的目錄hive/放在不同的目錄:
/root/workspace/eclipse/Test/src/edu/wzm/CallShell.java
/root/experiment/hive

如果我不設定dir或設dir為null,那麼fileName不得不是相對路徑,最好是絕對路徑:
call.callCMD("/root/experiment/hive.tar", "/root/experiment/hive", null);
// OR
call.callCMD("/root/experiment/hive.tar", "/root/experiment/hive");

如果我設定了dir指向了hive所在的父目錄就好辦多了:
call.callCMD("hive.tar", "hive", "/root/experiment/");

3.呼叫Shell指令碼

Java呼叫Shell命令和呼叫Shell指令碼的操作一模一樣。我這裡介紹另外幾個方面:

  1. 給指令碼傳遞引數;
  2. 捕獲呼叫的輸出結果;
  3. envp的使用。

給指令碼傳遞引數,這個操作很簡單,無非是把引數加到呼叫命令後面組成String,或String[]。

捕獲呼叫輸出資訊,前面也提到過用Process.getInputStream()。不過,建議最好對輸入流進行緩衝:

BufferedReader input = new BufferedReader(new InputStreamReader(process.getInputStream()));

另外,envp是一個String[],並且String[]中的每一個元素的形式是:name=value。如:我的Linux系統中沒有以下環境變數,但是我把它們寫在Java程式碼中,作為envp:
val=2
call=Bash Shell

我要呼叫的Shell指令碼是:/root/experiment/test.sh。
#!/usr/bin/env bash

args=1
if [ $# -eq 1 ];then
	args=$1
	echo "The argument is: $args"
fi

echo "This is a $call"
start=`date +%s`
sleep 3s
end=`date +%s`
cost=$((($end - $start) * $args * $val))
echo "Cost Time: $cost"

Java呼叫程式碼是:
private void callScript(String script, String args, String... workspace){
	try {
		String cmd = "sh " + script + " " + args;
//        	String[] cmd = {"sh", script, "4"};
		File dir = null;
		if(workspace[0] != null){
			dir = new File(workspace[0]);
			System.out.println(workspace[0]);
		}
		String[] evnp = {"val=2", "call=Bash Shell"};
		process = Runtime.getRuntime().exec(cmd, evnp, dir);
//            process = Runtime.getRuntime().exec(cmd);
		BufferedReader input = new BufferedReader(new InputStreamReader(process.getInputStream()));
		String line = "";
		while ((line = input.readLine()) != null) {
			System.out.println(line);
		}
		input.close();
	}
	catch (Exception e){
		e.printStackTrace();
	}
}

public static void main(String[] args) {
	// TODO Auto-generated method stub
	CallShell call = new CallShell();
	call.callScript("test.sh", "4", "/root/experiment/");
}

輸出:
/root/experiment/
The argument is: 4
This is a Bash Shell
Cost Time: 24

相關推薦

Java呼叫Shell命令指令碼

1.介紹 有時候我們在Linux中執行Java程式時,需要呼叫一些Shell命令和指令碼。而Runtime.getRuntime().exec()方法給我們提供了這個功能,而且Runtime.getRuntime()給我們提供了以下幾種exec()方法: Process e

java呼叫shell命令並獲取執行結果

原文地址:http://blog.csdn.net/arkblue/article/details/7897396 使用到Process和Runtime兩個類,返回值通過Process類的getInputStream()方法獲取 package ark;

Java呼叫shell命令解壓.7z壓縮包

Linux下解壓.7z壓縮包,java呼叫shell命令執行,解壓速度是java程式解壓的幾十倍乃至更多,首先確認主機上已安裝.7z命令。 package com.annet.upload.core.utils; import java.io.Buffe

java呼叫shell指令碼,解決傳參許可權問題

1. java 執行shell java 通過 Runtime.getRuntime().exec() 方法執行 shell 的命令或 指令碼,exec()方法的引數可以是指令碼的路徑也可以是直接的 shell命令 程式碼如下(此程式碼是存在問題的。完整程式

java呼叫shell指令碼、windows命令

public class ShellUtil {  private static Logger logger = Logger.getLogger(ShellUtil.class);  /**   *   * @param shPath  需要執行的命令或指令碼路徑   * @return   */  pu

java呼叫shell命令

import java.io.BufferedReader; import java.io.InputStreamReader; public class Shell { public static void main(String[] args) { String command =

awk呼叫shell命令與getlinesystem函式

total14 -rw-r--r-- 1 Administrator None 2 Mar 3 20:20 4 -rw-r--r-- 1 Administrator None 0 Feb 28 19:49 4]] -rw-r--r-- 1 Administrator None 87 Mar 5 20:22 a

java實現shell命令--小程式(通過RuntimeProcess實現)

先上程式碼(功能是要列出目錄): package testProcess; import java.io.BufferedReader; import java.io.InputStreamReader; import java.io.LineNumberReader; p

常用shell命令vi編輯器

css lin str 目錄 用戶輸入 重定向 move 處理 操作 Shell 介紹 Shell:殼(區別於:核) 普通意義上的shell就是可以接受用戶輸入命令的程序。它之所以被稱作shell是因為它隱藏了操作系統低層的細節。 Bourne-Again shell(ba

java執行shell命令中有空格的處理方法

java執行shell命令中有空格的處理?Runtime.getRuntime().exec(cmdstring);如果此時cmdstring中的參數(例如cp文件時文件名)含有特殊符號空格,此時就會出現錯誤,因為源碼會按照一些特殊字符(" \t\n\r\f",註意到其中含有空格)去切分cmdstring

每天一個shell命令腳本

col text 家目錄 mage home 返回 絕對路徑 數據文件 定位 常見linux 目錄名稱/ 虛擬目錄的根目錄。/bin 二進制目錄/boot 啟動目錄。存放啟動文件/dev 設備目錄/etc 系統配置目錄/home 主目錄,linux在這裏創建用戶目錄/li

Java 呼叫Linux 命令,並獲取命令執行結果

1.工具類 public class ExcuteLinux { public static String exeCmd(String commandStr) { String result = null; try { St

java呼叫cmd命令實現各種操作

public static void main(String[] args) { /*獲取cmd命令*/ try { Process pro = Runtime.getRuntime().exec("cmd /c calc"); //新增要進行的命令,"cmd /c calc"中calc代表

pycharm呼叫shell命令

在pycharm中呼叫shell命令 1、 # -*- coding:UTF-8 -*- import subprocess subprocess.call(["ls /home"], shell=True) #subprocess.call(["cat /root/d.txt | grep hel

java呼叫系統命令Demo

java呼叫系統命令範例 import java.io.BufferedReader; import java.io.InputStream; import java.io.InputStreamReader; public class SystemCommandCall {

Shell命令技巧

監控命令(每2秒執行一次) watch "ls -larth" 使用一個埠殺死程式 sudo fuser -k 8000/tcp 限制以下命令的記憶體使用 ulimit -Sv 1000# 1000 KBs = 1 MB ulimit -Sv unlimited # Remove limit

Java執行window命令linux命令

這裡強調一點,命令中用到的jdk還有其他一些需要配置的軟體的環境變數一定要配置成全域性的,外部可以讀取的,否則java根本無法執行命令。比如linux需要配置在 /etc/profile,用source /etc/profile可以即時生效。 下面是程式碼: packag

Python呼叫shell命令的幾種方法(在新程序中執行shell命令

有時候難免需要直接呼叫Shell命令來完成一些比較簡單的操作,比如mount一個檔案系統之類的。那麼我們使用Python如何呼叫Linux的Shell命令?下面來介紹幾種常用的方法: 1. os 模組 1.1. os模組的exec方法族 Python的exec系

java 呼叫 本地命令

Runtime rt = Runtime.getRuntime(); Process proc = rt.exec(cmd); InputStream stderr = proc.getInputS

go語言exec包呼叫shell命令

工程中需要用到ffmpeg,想直接用exec包呼叫shell命令。本來以為很簡單,結果折騰了一下午,最後查到了解決方案。 假如之前執行報錯的語句為: cmd := exec.Command("echo", "'helloworld!'") out, err := cmd.