在后台shell中分配变量

din*_*igo 1 bash shell

我想在后台分配一个变量,而我正在显示这样的进度指示器

sleep_echo(){
  sleep $1
  echo $2
}
show_ellapsed_time () {
  #...
}

(mivar=$(sleep_echo 3 hello)) &
show_ellapsed_time $!
echo $mivar
echo "after asignment"
Run Code Online (Sandbox Code Playgroud)

当我在终端中运行它时,它输出进度指示器,然后回声myvar

processing 00:00:03 [\]
hello
after asignment
Run Code Online (Sandbox Code Playgroud)

但是,当我在脚本中运行它时,分配不会发生

processing 00:00:03 [\]

after asignment
Run Code Online (Sandbox Code Playgroud)

为什么这会失败?是不是调用了te函数?

Fre*_*red 6

子进程无法通过变量赋值与其父进程通信.

在Bash中,可以通过多种方式创建子进程:

  • 启动外部命令
  • 通过用语句封闭语句来启动子shell ()
  • 使用重定向(例如管道)时启动子shell
  • 在后台启动任何内容 &

外部命令接收标记为导出的所有shell变量的副本(例如export VAR,local -x VAR)

子壳接收所有变量的副本.这很有用,但很容易意识到你处于一个新的环境中.父进程永远不会看到在子进程中进行的任何赋值.

你可以做什么:

  • 使用进程间通信来捕获输出
  • 将所需信息放在文件中,因为所有进程都可以访问文件(假设有足够的权限)
  • 使用命名FIFO,这是一种特殊类型的文件,允许单独的进程进行通信(请参阅参考资料mkfifo).
  • 以一种方式构造脚本,通过使用正确的构造来构建管道,在主上下文中保留最需要它的代码块.

对于最后一个,使用类似的进程替换<(COMMAND)并且>(COMMAND)通常比使用的典型管道好得多|,例如允许while循环体(或者,实际上,任何其他语句,替换被馈送到/来自)保留在上下文中的脚本.这允许循环内的变量赋值具有预期的效果.

在你的情况下,你有这些线:

(mivar=$(sleep_echo 3 hello)) &
show_ellapsed_time $!
Run Code Online (Sandbox Code Playgroud)

也许你可以用以下内容替换它们:

show_ellapsed_time &
BG_PID=$!
mivar=$(sleep_echo 3 hello)
kill $BG_PID
Run Code Online (Sandbox Code Playgroud)

我不知道你show_ellapsed_time知道如果没有你传递给该函数的参数,它是否可以工作,但是赋值可以工作.您还可以将主shell的PID传递给函数,如下所示:

show_ellapsed_time $$ &
Run Code Online (Sandbox Code Playgroud)