按照官方说明,tmux是一个终端复用软件。我接触tmux也就是这几天的事情,但已经发现其强大。作为一个文艺程序员,有必要向大家分享一下,这么好的东东怎敢藏着掖着。
先用起来再说
假设你已经装好tmux,那么在终端中直接输入tmux
并回车,就进入了tmux,退出的话大家都懂的,输入exit
或C-d
即可。进入tmux后你在终端中爱干嘛就干嘛,那就有人会问了,在进入tmux之前已经有终端了,那还折腾这个干啥?别急,tmux的作用在于终端复用,也就是说可以把一个终端变成多个终端。在tmux中按C-b c
,你就会看到开启了一个新窗口,并在其中打开了一个新的终端。那又有人问了,这有啥用,Linux的终端软件本身就支持打开多个标签页,还折腾这个干啥?额,这个问题问得好,如果你只是在本地开多个终端敲一下ls
什么的,貌似确实没什么必要。但想想,你如果要通过SSH登录到远程服务器,有了tmux,你只需要SSH一次就行了,否则,如果你想要在编辑文件的同时到shell中执行某个命令,就得开两个SSH连接,如果你还要同时用数据库,或者要tail
某个日志文件的话,那就还得再开几个了,我这前就是这么干的。有了tmux,你只需要一个SSH连接,然后打开tmux,再多敲几次C-b c
就好了。对了,你开了几个窗口,怎么在窗口之间切换呢?C-b p
是切换到前一个窗口,C-b n
是切换到后一个窗口,C-b w
是列出所有的窗口随你选,关闭窗口的快捷键是C-b &
。你一定发现快捷键中都带C-b
,这是tmux的默认快捷键前缀,tmux通过这个前缀才知道你感兴趣的是她而不是别人。好了,你已经可以使用tmux了,总结一下学到的几个快捷键:
C-b c 创建一个新窗口 C-b & 关闭当前窗口 C-b p 切换到上一个窗口 C-b n 切换到下一个窗口
会话,窗口,面板
tmux有三个基本概念:会话,窗口和面板。当你输入tmux
后,tmux实际做的事是首先创建一个会话,然后在这个会话中创建一个窗口,这个窗口只包含一个面板,你在其中看到的终端实际上都属于tmux的某个面板。关于窗口的操作在上一节中已经说明,面板又是什么呢?有人说了,你一个窗口就占满一屏,要看另外一个窗口得切换过去,我想要的是同时看到多个窗口的内容,好了,面板就是干这个事儿的。当你在某个窗口中按下C-b %
时,你会发现窗口被横向分成两个面板,而如果按下C-b "
,窗口则被纵向分成两个面板,按C-b o
会跳到下个面板,C-b加方向键
则会在*选择各面板,C-b x
会关闭当前面板。
一个会话可以包含多个窗口,我们可以用会话来组织窗口,会话相当于一个工作环境,比如我们可以创建一个work的会话,里面放我们平时开发需要的窗口,tmux默认会创建一个会话,我们也可以通过参数创建指定名称的会话,例如
tmux new-session -s work
在控制台中输入这条命令就可以创建一个名为work的会话并进入它,我们也可以同时指定这个会话中第一个窗口的名字,比如mysql:
tmux new-session -s work -n mysql
关于会话有两个非常重要的操作,即attach和detach,attach就是让某个会话到前台来运行,而detach则是将某个会话放到后台。通常,当我们打开tmux时,tmux在创建一个会话的同时也会attach到这个会话,所以我们会立即看到tmux的窗口。在某个会话中,我们按C-b d
会detach这个会话,也就会回到原先的终端控制台,但实际上并没有退出这个会话,比如你可以通过在终端中输入下列命令重新attach到work会话:
tmux attach -t work
一个时刻可能存在多个会话,在tmux外面通过下列命令可以列出当前有哪些会话:
tmux ls
在tmux里面可以按下C-b s
来列出会话,然后可以选择一个切换过去。
在我们创建会话的同时也可以加上-d
选项以表示不自动attach到这个新会话:
tmux new-session -s work -d
这样的话你会通过tmux ls
看到新创建的work会话,并可以通过tmux attach -t work
attach到该会话中。
在控制台上可以通过下列命令关闭会话work:
tmux kill-session -t work
上面的一些以tmux打头的命令都是tmux提供的外部接口,是在命令行*我们直接运行的,事实上,除了对会话的操作之外,在外部也可以直接创建窗口,例如:
tmux new-window -n project -t work
这条命令可以针对work会话创建一个名为project的窗口。
在创建会话和窗口时甚至可以指定要执行的命令:
tmux new-session -s work -n dev -d "ssh user@example.com"
以上命令会在创建work会话后,在其第一个窗口的控制台中执行SSH命令以远程连接到服务器中。
创建窗口时也与此类似,可将命令写在最后。
更为强大的是,tmux可以让你向某个会话[:窗口:面板]发送任意按键:
tmux send-keys -t work:editor 'emacs' C-m
上述命令能够向work会话的editor窗口发送emacs加回车,即在这个窗口中打开emacs编辑器。
有了tmux的各种强大的外部命令,我们就可以自定义脚本,自动化很多事情:
tmux new-session -s work -n mysql -d 'mysql'
tmux new-window -n editor -t work
tmux send-keys -t work:editor 'emacs' C-m
将上述命令写在一个shell脚本里,就可以一键完成下述功能:创建一个名为work的会话,第一个窗口命名为mysql,并在其中执行mysql命令进入mysql,第二个窗口名为editor,并在其中打开emacs。
配置文件
除了强大的外部命令之外,tmux也可以通过自己的配置文件来自定义很多东西,在tmux运行时,会读取/etc/tmux.conf
中的配置作为系统级别的配置,然后会在当前用户的家目录里面读取.tmux.conf
这个配置文件,如果都找不到,就用默认配置。一般情况下,我们只需要在我们家目录里面创建.tmux.conf
这个文件来进行自定义配置即可。
在配置文件中,我们可以用到tmux提供的很多命令,可以设置很多变量的值或者给tmux配色等,实际上,很多命令既有外部命令行接口,又可以在配置文件中使用,还可以在tmux内部的命令模式中使用(C-b :命令
)。比如上节中用脚本完成的事情同样也可以用配置文件完成,在配置文件中不用加tmux:
new-session -s work -n mysql -d 'mysql'
new-window -n editor -t work
send-keys -t work:editor 'emacs' C-m
按键同步发送
我一开始用tmux就是为了向多个终端同步发送一些命令,比如用于多台相同环境的机器上的配置文件同步修改和程序发布。其实这在tmux中只需要一条命令就可搞定,比如有两台机器,首先在一个窗口中开两个面板,分别连到这两台机器,然后按C-b :set synchronize-panes
即可开/关面板的同步模式,开启后,在一个面板中输入的按键也会同步发送到另一个面板中,是不是非常容易?
永不断线
tmux实际上是server/client模式,当创建第一个会话时,server就会运行,以后的会话都以attach/detach的方式被server服务,当所有会话都退出时,server才会退出。在我们工作中,经常是SSH连接到服务器上去写代码,但由于网络的不稳定性,会偶尔发生SSH连接中断的情况,这样我们之前的会话就会丢失,如果是一个nohup的后台任务还好,只要我们重新连上去之后,还是可以找到的(但至少需要回到事发现场,比如重新cd到原来的文件夹等),而如果我们正在写代码,还没来得及保存,很可能就杯具了(除非某些编辑器刚好把你最新的代码自动保存过)。tmux不会存在这个问题,不信,你可以在终端中打开tmux,然后关闭这个终端,再打开一个新的终端,输入tmux ls
看看,你会发现之前的会话仍然还在,你需要做的只是attach它就行!那么我们平常在SSH连接到服务器上后,再通过tmux去开启我们的工作会话,不管你在里面干什么,都不用担心断线,因为tmux server会帮你保存你的会话,你随时可以attach它,而且你的工作现场一直在那,连目录都不用切换,除非你真的kill掉它。
结对编程
结对编程是不是很酷?事实上我也没试过。很简单,因为它的会话可以被多个client attach,比如A和B两人分别在家通过SSH连接到服务器上,然后都attach到同一个会话中,那么A和B的操作彼此都可以看到,简直碉堡了!
参考资料
关于tmux的命令快捷键,请在tmux中输入
C-b ?
请在终端中输入
man tmux
《tmux: Productive Mouse-Free Development》
好了,其实我才接触几天,了解得也不是很深,其中的一些理解也可能有误,再加上时间有限,所以错误在所难免,以后可能会根据我的领悟去修改完善这篇入门指南。不过我主要是想抛砖引玉,希望大家都能去用用,去共同发掘这块宝藏。子曾经曰过:神器在手,天下我有!但愿tmux能在你的工作中助你一臂之力。