exec包执行外部命令,它将os.StartProcess进行包装使得它更容易映射到stdin和stdout,并且利用pipe连接i/o.
func LookPath(file string) (string, error) //LookPath在环境变量中查找科执行二进制文件,如果file中包含一个斜杠,则直接根据绝对路径或者相对本目录的相对路径去查找
func main() {
f, err := exec.LookPath("ls")
if err != nil {
fmt.Println(err)
}
fmt.Println(f) // /bin/ls
}
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时便会产生该信息.
}
//获取执行结果
arg := "ffprobe -i "
arg += flag
arg += " -show_streams|grep start_time"
cmd := exec.Command("/bin/sh", "-c", arg)
out, err := cmd.Output()
if err != nil {
fmt.Println(err)
}
start_time := string(out)
//关闭ffmpeg
cmd := ""
cmd += "a="
cmd += flag
cmd += ";b=`ps -ef |grep $a|grep -v grep`;if [ \"x$b\" != \"x\" ];then ps -ef|grep $a|grep -v grep|cut -c 9-15 | xargs kill -9 ;fi"
lsCmd := exec.Command("/bin/sh", "-c", cmd)
fmt.Println(cmd)
err := lsCmd.Run()
if err != nil {
panic(err)
fmt.Println("cmd:%s faild", cmd)
}
}
实践
同步
lsCmd := exec.Command("bash", "-c", "/home/shell/modifyhls.sh /data/channellist/channel03/index.m3u8 10 a a ")
lsOut, err := lsCmd.Output() //或者start()
if err != nil {
panic(err)
}
fmt.Println("> ls -a -l -h")
fmt.Println(string(lsOut))
异步
lsCmd := exec.Command("bash", "-c", "/home/shell/modifyhls.sh /data/channellist/channel03/index.m3u8 10 a a & ")
err := lsCmd.Start()
if err != nil {
panic(err)
}
写函数
func exec_shell(s string) {
lsCmd := exec.Command("bash", "-c", "/home/shell/modifyhls.sh /data/channellist/channel03/index.m3u8 10 a a & ")
err := lsCmd.Start()
if err != nil {
panic(err)
}
}