我正在尝试配置 tmux,以便在按下“enter”键时它始终刷新客户端(或客户端状态栏)。我目前使用
bind -n enter send-key Enter; refresh -S
但行为很奇怪。它确实每次都会发送Enter
,因此我可以正常使用该密钥,但refresh -S
并不总是执行。我使用状态栏设置测试配置
set -g status-left "#{pane_current_path}"
当输入命令时,它应该立即更新cd
。但这并没有发生。当进入cd
某处时,状态栏仅在标准周期中更新,即有时会延迟。如果我在状态栏不久后输入另一个命令(或空行),cd
状态栏确实会立即更新。
同样的情况也发生在send-key KPEnter
not just 上Enter
。
有人知道这里发生了什么吗?
这是一个竞争条件。
扩展
#{pane_current_path}
需要 tmux 服务器了解该进程的当前工作目录是窗格中的“主要目录”。tmux 服务器如何学习或已经知道这是哪个进程并不重要,重点是该进程不是 tmux 的一部分,它是一个单独的进程。通常,就像您的情况一样,窗格中的“主进程”是交互式 shell。
tmux 服务器可以通过检查 来获知该 shell 的当前工作目录
/proc/<PID>/cwd
,其中<PID>
表示 shell 的 PID;或通过某种等效方法。当您输入
cd whatever
Enter并send-key Enter; refresh -S
工作时,send-key Enter
使 shell 开始解释cd whatever
。重要的是send-key
只是发送密钥,它不等待此操作的任何效果;shell刚刚开始解释cd whatever
,解释和执行需要一些时间;但tmux没有任何延迟地refresh -S
继续进行,这使得 tmux 服务器开始刷新客户端的状态行,这涉及评估#{pane_current_path}
. shell(由 触发send-key
)和 tmux(由 触发)并行refresh
执行其任务,并且不能保证 shell 在评估之前能够实际更改其当前工作目录。#{pane_current_path}
如果首先评估格式字符串,那么您将在状态行中看到旧的工作目录。
cd
make#{pane_current_path}
返回新的当前工作目录后不久的另一个命令或空行,因为Enter这里已经足够晚了,shell 已设法更改其工作目录。cd
导致交互式 shell 在不竞争的情况下刷新状态行的一个好方法是在实际更改工作目录后cd
触发自身:refresh -S
如果 shell 位于 tmux 内部,那么上面的代码片段将创建一个重载
cd
内置函数的 shell 函数。笔记:
tmux
.cd
将导致代码片段失败或行为异常。cd ()
在 Bash 中,您可以通过替换为 non-portable来解决这个问题function cd
。cd
。tmux
(tmux refresh -S &
) 将使 shell 不等待它,同时不会破坏解决方案。我指出这一点是因为在您最初尝试使用bind
shell 时也不会等待,所以也许这就是您更喜欢的。但是,如果异步tmux
失败(尽管可能性很小)并打印错误消息,则该消息可能会出现在下一个 shell 提示符之后,这会令人困惑。因此,在我看来,同步调用更为优雅tmux
。确实,tmux
异步调用可以节省几毫秒;不过,在交互式 shell 中您很可能不会注意到其中的差异。