實例1:
package main
import (
"bufio"
"fmt"
"os"
"os/exec"
//"strings"
)
func main() {
reader := bufio.NewReader(os.Stdin)
cmdString, err := reader.ReadString('\n')
if err != nil {
fmt.Fprintln(os.Stderr, err)
}
fmt.Println(cmdString)
//cmdString = strings.TrimSuffix(cmdString, "\n")
cmdString = cmdString[:len(cmdString)-2]
cmd := exec.Command(cmdString)
cmd.Stderr = os.Stderr
cmd.Stdout = os.Stdout
cmd.Run()
}
windows內置命令實例
package main
import (
"fmt"
"os/exec"
)
func main() {
c := exec.Command("cmd", "/C", "del", "D:\\a.txt")
exec.Command()
if err := c.Run(); err != nil {
fmt.Println("Error: ", err)
}
}
golang os/exec 執行外部命令
exec包執行外部命令,它將os.StartProcess進行包裝使得它更容易映射到stdin和stdout,並且利用pipe連接i/o.
func LookPath(file string) (string, error) //LookPath
LookPath在環境變量中查找科執行二進制文件,如果file中包含一個斜杠,則直接根據絕對路徑或者相對本目錄的相對路徑去查找
package main
import (
"fmt"
"os/exec"
)
func main() {
f, err := exec.LookPath("curl")
if err != nil {
fmt.Println(err)
}
fmt.Println(f) // /bin/ls
}
輸出:
D:\curl\curl.exe
type Cmd //表示一個正在准備或者正在運行的外部命令
type Cmd struct {
Path string //運行命令的路徑,絕對路徑或者相對路徑
Args []string // 命令參數
Env []string //進程環境,如果環境為空,則使用當前進程的環境
Dir string //指定command的工作目錄,如果dir為空,則comman在調用進程所在當前目錄中運行
Stdin io.Reader //標准輸入,如果stdin是nil的話,進程從null device中讀取(os.DevNull),stdin也可以時一個文件,否則的話則在運行過程中再開一個goroutine去
//讀取標准輸入
Stdout io.Writer //標准輸出
Stderr io.Writer //錯誤輸出,如果這兩個(Stdout和Stderr)為空的話,則command運行時將響應的文件描述符連接到os.DevNull
ExtraFiles []*os.File
SysProcAttr *syscall.SysProcAttr
Process *os.Process //Process是底層進程,只啟動一次
ProcessState *os.ProcessState //ProcessState包含一個退出進程的信息,當進程調用Wait或者Run時便會產生該信息.
}
func Command(name string, arg ...string) *Cmd //command返回cmd結構來執行帶有相關參數的命令,它僅僅設定cmd結構中的Path和Args參數,如果name參數中不包含路徑分隔符,command使用LookPath來解決路徑問題,否則的話就直接使用name;Args直接跟在command命令之后,所以在Args中不需要添加命令.
package main
import (
"bytes"
"fmt"
"log"
"os/exec"
"strings"
)
func main() {
cmd := exec.Command("cmd.exe", "C", "dir")
cmd.Stdin = strings.NewReader("some input")
var out bytes.Buffer
cmd.Stdout = &out
err := cmd.Run()
if err != nil {
log.Fatal(err)
}
fmt.Printf("in all caps:%q\n", out.String())
}
func (c *Cmd) CombinedOutput() ([]byte, error) //運行命令,並返回標准輸出和標准錯誤
package main
import (
"fmt"
"os/exec"
)
func main() {
cmd := exec.Command("cmd", "/C", "dir")
out, err := cmd.CombinedOutput()
if err != nil {
fmt.Println(err)
}
fmt.Println(string(out))
}
func (c *Cmd) Output() ([]byte, error) //運行命令並返回其標准輸出
package main
import (
"fmt"
"os/exec"
)
func main() {
cmd := exec.Command("cmd", "/C", "dir")
out, err := cmd.Output()
if err != nil {
fmt.Println(err)
}
fmt.Println(string(out))
}
注意:Output()和CombinedOutput()不能夠同時使用,因為command的標准輸出只能有一個,同時使用的話便會定義了兩個,便會報錯
func (c *Cmd) Run() error //開始指定命令並且等待他執行結束,如果命令能夠成功執行完畢,則返回nil,否則的話邊會產生錯誤 func (c *Cmd) Start() error //使某個命令開始執行,但是並不等到他執行結束,這點和Run命令有區別.然后使用Wait方法等待命令執行完畢並且釋放響應的資源
package main
import (
"fmt"
"os/exec"
)
func main() {
cmd := exec.Command("cmd", "/C", "dir")
err := cmd.Run()
if err != nil {
fmt.Println(err)
}
fmt.Println(cmd.Start())
}
輸出:
exec: already started
注:一個command只能使用Start()或者Run()中的一個啟動命令,不能兩個同時使用.
func (c *Cmd) StderrPipe() (io.ReadCloser, error) //StderrPipe返回一個pipe,這個管道連接到command的標准錯誤,當command命令退出時,Wait將關閉這些pipe func (c *Cmd) StdinPipe() (io.WriteCloser, error) //StdinPipe返回一個連接到command標准輸入的管道pipe
package main
import (
"fmt"
"os"
"os/exec"
)
func main() {
cmd := exec.Command("cat")
stdin, err := cmd.StdinPipe()
if err != nil {
fmt.Println(err)
}
_, err = stdin.Write([]byte("tmp.txt"))
if err != nil {
fmt.Println(err)
}
stdin.Close()
cmd.Stdout = os.Stdout
cmd.Start()
}
func (c *Cmd) StdoutPipe() (io.ReadCloser, error) //StdoutPipe返回一個連接到command標准輸出的管道pipe
package main
import (
"fmt"
"io/ioutil"
"os/exec"
"time"
)
func main() {
cmd := exec.Command("dir")
stdout, err := cmd.StdoutPipe()
if err != nil {
fmt.Println(err)
}
cmd.Run()
go func() {
content, err := ioutil.ReadAll(stdout)
if err != nil {
fmt.Println("err: ", err)
}
fmt.Println(string(content))
}()
time.Sleep(1 * time.Second)
fmt.Println("main is ending")
}
func (c *Cmd) Wait() error //Wait等待command退出,他必須和Start一起使用,如果命令能夠順利執行完並順利退出則返回nil,否則的話便會返回error,其中Wait會是放掉所有與cmd命令相關的資源
type Error //Error返回科執行二進制文件名字不能夠執行的原因的錯誤
type Error struct {
Name string
Err error
func (e *Error) Error() string type ExitError //一個command不能夠正常退出的error
type ExitError struct {
*os.ProcessState
}
func (e *ExitError) Error() string
