go执行cmd模块——exec

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))
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Generalzy

倍感荣幸文章对您有帮助

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值