目录
exec简介
exec包可以用于执行外部命令。简单来说,就是可以用该包来执行其它的应用程序(比如shell脚本,或者其它任意可执行程序)。它对os.StartProcess函数做了包装,以便更容易的对输入输出做处理,使用管道连接I/O,以及作其它的一些调整 。
exec 函数和Cmd的方法
func Command(name string, arg ...string) *Cmd
//方法返回一个*Cmd, 用于执行name指定的程序(携带arg参数)
func (c *Cmd) Run() error
//执行Cmd中包含的命令,阻塞直到命令执行完成
func (c *Cmd) Start() error
//执行Cmd中包含的命令,该方法立即返回,并不等待命令执行完成
func (c *Cmd) Wait() error
//该方法会阻塞直到Cmd中的命令执行完成,但该命令必须是被Start方法开始执行的
func (c *Cmd) Output() ([]byte, error)
//执行Cmd中包含的命令,并返回标准输出的切片
func (c *Cmd) CombinedOutput() ([]byte, error)
//执行Cmd中包含的命令,并返回标准输出与标准错误合并后的切片
func (c *Cmd) StdinPipe() (io.WriteCloser, error)
//返回一个管道,该管道会在Cmd中的命令被启动后连接到其标准输入
func (c *Cmd) StdoutPipe() (io.ReadCloser, error)
//返回一个管道,该管道会在Cmd中的命令被启动后连接到其标准输出
func (c *Cmd) StderrPipe() (io.ReadCloser, error)
//返回一个管道,该管道会在Cmd中的命令被启动后连接到其标准错误
Command方法获取一个命令执行对象,再用Run或者Output方法同步阻塞执行,或者用Start异步执行。
Run,Start,Wait的关系
func (c *Cmd) Run() error // 开始执行c包含的执行命令,并阻塞直到命令执行完毕,由于是阻塞的,所以它是同步方法和Output方法类似,只是前者很方便就能拿到程序的输出内容。Run不方便拿到程序执行输出的内容
func (c *Cmd) Start() error // 开始执行c包含的命令,但并不会等待该命令完成,它会立即返回。Wait方法会返回命令的返回状态码并在命令返回后释放相关的资源。所以可以把它看做是异步的执行方法
func (c *Cmd) Wait() error // Wait会阻塞直到该命令执行完成,这个命令必须和start命令配合使用,start开始执行,但不但表执行完成,wait等待命令执行完成。
func (c *Cmd) Run() error {
if err := c.Start(); err != nil {
return err
}
return c.Wait()
}
案例
只执行命令,不获取结果
func main() {
cmd := exec.Command("ls", "-l", "/Users/liuqingzheng/")
err := cmd.Run()
if err != nil {
fmt.Println("执行命令出错",err)
}
}
执行命令并获取结果
func main() {
cmd := exec.Command("ls", "-l", "/Users/liuqingzheng/")
res,err:=cmd.CombinedOutput()
if err != nil {
fmt.Println(err)
}
fmt.Println(string(res))
}
func main() {
res,err:= exec.Command("ls", "-l", "/Users/liuqingzheng/").Output()
if err != nil {
fmt.Println("执行出错:",err)
}
fmt.Println(string(res))
}
执行命令,区分stdout 和 stderr
func main() {
cmd := exec.Command("ls", "-l", "/Users/iuqingzheng/*.log")
var stdout, stderr bytes.Buffer
cmd.Stdout = &stdout // 标准输出
cmd.Stderr = &stderr // 标准错误
err := cmd.Run()
outStr, errStr := string(stdout.Bytes()), string(stderr.Bytes())
fmt.Printf("标准输出:\n%s\n标准错误:\n%s\n", outStr, errStr)
if err != nil {
fmt.Println("执行出错:",err)
}
}
使用管道,多条命令组合
// ps aux | grep redis
func main() {
c1 := exec.Command("ps", "aux") // 命令1
c2 := exec.Command("grep", "redis") // 命令2
c2.Stdin, _ = c1.StdoutPipe() // c1的输出,作为c2.输入
c2.Stdout = os.Stdout // c2的输出到控制台上
_ = c2.Start() // c2立即启动,不等结果返回
_ = c1.Run() // c1阻塞直到命令执行完成
_ = c2.Wait() //c2阻塞直到Cmd中的命令执行完成
}
设置程序级别的环境变量
func main() {
os.Setenv("name", "lqz")
//cmd := exec.Command("echo", os.ExpandEnv("$name"))
cmd := exec.Command("echo", os.ExpandEnv("$name"))
out, err := cmd.CombinedOutput()
if err != nil {
fmt.Println("执行出错:",err)
}
fmt.Printf("%s", out)
}
//ExpandEnv根据当前环境变量的值来替换字符串中的${var}或者$var。如果引用变量没有定义,则用空字符串替换
s := "hello $GOROOT"
fmt.Println(os.ExpandEnv(s))