本文首发于CSDN网站,下面的版本又通过进一步的修订。前端
全文共12k字,阅读需20分钟。node
我一直信奉简洁至上的原则,桌面窗口的数量越少,个人心情就越愉悦,开发的效率也就越高。反之,杂乱的桌面,暴涨的Chrome tab数量,或是无数的终端窗口,它们会逐步侵占个人注意力,分散个人思惟,最终令我难以专一。所以桌面上我不多放文件,使用Chrome时常点 OneTab 回收标签页,切进终端时使用tmux管理窗口。python
我很懒,懒到指望开机后不须要任何操做,本地的十几种web开发服务器就自动运行,固然我不但愿连续弹出十几个窗口或是tab,我须要的是静默无感知的启用服务,而后还能快速地进入到现场进行操做,web服务器运行时不占据终端窗口,关闭iTem2后操做现场不会被销毁。个人这些指望,tmux都能帮我实现,实际上,除了这些,tmux还能作得更多更好。git
到目前为止,tmux帮助我两年有余,它带给我许多惊喜。独乐不如众乐,愿你也能一同享受tmux带来的快乐。程序员
tmux是一款优秀的终端复用软件,它比Screen更增强大,至于如何强大,网上有大量的文章讨论了这点,本文再也不重复。tmux之因此受人们喜好,主要得益于如下三处功能:github
以上,只是主要功能,更多功能还在后头,接下来我将详细地介绍tmux的使用技巧。web
首先安装之。chrome
在Mac中安装:shell
# 先安装Homebrew,有则跳过
ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/install)"
# 安装tmux
brew install tmux复制代码
在Linux中安装:npm
sudo apt-get install tmux复制代码
开始以前,咱们先了解下基本概念:
tmux采用C/S模型构建,输入tmux命令就至关于开启了一个服务器,此时默认将新建一个会话,而后会话中默认新建一个窗口,窗口中默认新建一个面板。会话、窗口、面板之间的联系以下:
一个tmux session
(会话)能够包含多个window
(窗口),窗口默认充满会话界面,所以这些窗口中能够运行相关性不大的任务。
一个window
又能够包含多个pane
(面板),窗口下的面板,都处于同一界面下,这些面板适合运行相关性高的任务,以便同时观察到它们的运行状况。
新建一个tmux session很是简单,语法为tmux new -s session-name
,也能够简写为tmux
,为了方便管理,建议指定会话名称,以下。
tmux # 新建一个无名称的会话
tmux new -s demo # 新建一个名称为demo的会话复制代码
会话中操做了一段时间,我但愿断开会话同时下次还能接着用,怎么作?此时可使用detach命令。
tmux detach # 断开当前会话,会话在后台运行复制代码
也许你以为这个太麻烦了,是的,tmux的会话中,咱们已经可使用tmux快捷键了。使用快捷键组合Ctrl+b
+ d
,三次按键就能够断开当前会话。
断开会话后,想要接着上次留下的现场继续工做,就要使用到tmux的attach命令了,语法为tmux attach-session -t session-name
,可简写为tmux a -t session-name
或 tmux a
。一般咱们使用以下两种方式之一便可:
tmux a # 默认进入第一个会话
tmux a -t demo # 进入到名称为demo的会话复制代码
会话的使命完成后,必定是要关闭的。咱们可使用tmux的kill命令,kill命令有kill-pane
、kill-server
、kill-session
和 kill-window
共四种,其中kill-session
的语法为tmux kill-session -t session-name
。以下:
tmux kill-session -t demo # 关闭demo会话
tmux kill-server # 关闭服务器,全部的会话都将关闭复制代码
管理会话的第一步就是要查看全部的会话,咱们可使用以下命令:
tmux list-session # 查看全部会话
tmux ls # 查看全部会话,提倡使用简写形式复制代码
若是恰好处于会话中怎么办?别担忧,咱们可使用对应的tmux快捷键Ctrl+b
+ s
,此时tmux将打开一个会话列表,按上下键(⬆︎⬇︎)或者鼠标滚轮,可选中目标会话,按左右键(⬅︎➜)可收起或展开会话的窗口,选中目标会话或窗口后,按回车键便可完成切换。
关于快捷指令,首先要认识到的是:tmux的全部指令,都包含同一个前缀,默认为Ctrl+b
,输入完前缀事后,控制台激活,命令按键才能生效。前面tmux会话相关的操做中,咱们共用到了两个快捷键Ctrl+b
+ d
、Ctrl+b
+ s
,但这仅仅是冰山一角,欲窥tmux庞大的快捷键体系,请看下表。
表一:系统指令。
前缀 | 指令 | 描述 |
---|---|---|
Ctrl+b |
? |
显示快捷键帮助文档 |
Ctrl+b |
d |
断开当前会话 |
Ctrl+b |
D |
选择要断开的会话 |
Ctrl+b |
Ctrl+z |
挂起当前会话 |
Ctrl+b |
r |
强制重载当前会话 |
Ctrl+b |
s |
显示会话列表用于选择并切换 |
Ctrl+b |
: |
进入命令行模式,此时可直接输入ls 等命令 |
Ctrl+b |
[ |
进入复制模式,按q 退出 |
Ctrl+b |
] |
粘贴复制模式中复制的文本 |
Ctrl+b |
~ |
列出提示信息缓存 |
表二:窗口(window)指令。
前缀 | 指令 | 描述 |
---|---|---|
Ctrl+b |
c |
新建窗口 |
Ctrl+b |
& |
关闭当前窗口(关闭前需输入y or n 确认) |
Ctrl+b |
0~9 |
切换到指定窗口 |
Ctrl+b |
p |
切换到上一窗口 |
Ctrl+b |
n |
切换到下一窗口 |
Ctrl+b |
w |
打开窗口列表,用于且切换窗口 |
Ctrl+b |
, |
重命名当前窗口 |
Ctrl+b |
. |
修改当前窗口编号(适用于窗口从新排序) |
Ctrl+b |
f |
快速定位到窗口(输入关键字匹配窗口名称) |
表三:面板(pane)指令。
前缀 | 指令 | 描述 |
---|---|---|
Ctrl+b |
" |
当前面板上下一分为二,下侧新建面板 |
Ctrl+b |
% |
当前面板左右一分为二,右侧新建面板 |
Ctrl+b |
x |
关闭当前面板(关闭前需输入y or n 确认) |
Ctrl+b |
z |
最大化当前面板,再重复一次按键后恢复正常(v1.8版本新增) |
Ctrl+b |
! |
将当前面板移动到新的窗口打开(原窗口中存在两个及以上面板有效) |
Ctrl+b |
; |
切换到最后一次使用的面板 |
Ctrl+b |
q |
显示面板编号,在编号消失前输入对应的数字可切换到相应的面板 |
Ctrl+b |
{ |
向前置换当前面板 |
Ctrl+b |
} |
向后置换当前面板 |
Ctrl+b |
Ctrl+o |
顺时针旋转当前窗口中的全部面板 |
Ctrl+b |
方向键 |
移动光标切换面板 |
Ctrl+b |
o |
选择下一面板 |
Ctrl+b |
空格键 |
在自带的面板布局中循环切换 |
Ctrl+b |
Alt+方向键 |
以5个单元格为单位调整当前面板边缘 |
Ctrl+b |
Ctrl+方向键 |
以1个单元格为单位调整当前面板边缘(Mac下被系统快捷键覆盖) |
Ctrl+b |
t |
显示时钟 |
tmux的丝滑分屏功能正是得益于以上系统、窗口、面板的快捷指令,只要你愿意,你就能够解除任意的快捷指令,而后绑上你喜欢的指令,固然这就涉及到它的可配置性了,请继续往下读。
除了快捷指令外,tmux还提供了相似vim的配置性功能。可配置性是软件的一项进阶级功能,只有具有了可配置性,软件才有了鲜活的个性,用户才能体会到操做的快感。
相信只要你用过几回tmux,就会发现Ctrl+b
指令前缀,着实不太方便。这两个键相距太远,按键成本过高了。所以咱们首先须要将它更换为距离更近的Ctrl+a
组合键,或者不经常使用的 ` 键(固然其余键也是能够的)。
tmux的用户级配置文件为~/.tmux.conf
(没有的话就建立一个),修改快捷指令,只须要增长以下三行便可。
set -g prefix C-a #
unbind C-b # C-b即Ctrl+b键,unbind意味着解除绑定
bind C-a send-prefix # 绑定Ctrl+a为新的指令前缀
# 从tmux v1.6版起,支持设置第二个指令前缀
set-option -g prefix2 ` # 设置一个不经常使用的`键做为指令前缀,按键更快些复制代码
修改的~/.tmux.conf
配置文件有以下两种方式能够令其生效:
Ctrl+b
指令前缀,而后按下系统指令:
,进入到命令模式后输入source-file ~/.tmux.conf
,回车后生效。既然快捷指令如此方便,更为优雅的作法是新增一个加载配置文件的快捷指令 ,这样就能够随时随地load新的配置了,以下所示。
# 绑定快捷键为r
bind r source-file ~/.tmux.conf \; display-message "Config reloaded.."复制代码
请特别注意,在已经建立的窗口中,即便加载了新的配置,旧的配置依然有效(只要你新加的功能没有覆盖旧的配置,所以若是你第一次绑定快捷指令为x
键,而后又改成绑定y
键,那么x
和y
都将有效),新建会话不受此影响,将直接采用新的配置。
既然咱们已经迈出配置化的第一步,那么接下来咱们能够作得更多。
tmux中,使用最多的功能之一就是新增一个面板。水平方向新增面板的指令是 prefix
+ "
,垂直方向是 prefix
+ %
,"
和 %
须要两个键同时按下才能完成,加上指令前缀至少须要3~4次按键才能组成一个完整的指令,同时这个两个键也不够醒目和方便,所以咱们能够绑定两个更经常使用的指令 -
、|
,以下所示:
unbind '"'
bind - splitw -v -c '#{pane_current_path}' # 垂直方向新增面板,默认进入当前目录
unbind %
bind | splitw -h -c '#{pane_current_path}' # 水平方向新增面板,默认进入当前目录复制代码
默认状况下,tmux的多窗口之间的切换以及面板大小调整,须要输入指令才能完成,这一过程,涉及到的指令较多,并且操做麻烦,特别是面板大小调整,指令难以一步到位,这个时候开启鼠标支持就完美了。
对于tmux v2.1(2015.10.28)以前的版本,需加入以下配置:
setw -g mode-mouse on # 支持鼠标选取文本等
setw -g mouse-resize-pane on # 支持鼠标拖动调整面板的大小(经过拖动面板间的分割线)
setw -g mouse-select-pane on # 支持鼠标选中并切换面板
setw -g mouse-select-window on # 支持鼠标选中并切换窗口(经过点击状态栏窗口名称)复制代码
有的地方可能会出现set-window-option
的写法,setw
就是它的别名。
对于tmux v2.1及以上的版本,仅需加入以下配置:
set-option -g mouse on # 等同于以上4个指令的效果复制代码
须要注意的是,开启鼠标支持后,iTem2默认的鼠标选中即复制功能须要同时按下 Alt
键,才会生效。
鼠标支持确实能带来很大的便捷性,特别是对于习惯了鼠标操做的tmux新手,但对于键盘爱好者而言,这不是什么好消息,对他们而言,双手不离键盘是基本素质。
虽然指令前缀加方向键
能够切换面板,但方向键
太远,不够快,不够Geek。不要紧,咱们能够将面板切换升级为熟悉的h
、j
、k
、l
键位。
# 绑定hjkl键为面板切换的上下左右键
bind -r k select-pane -U # 绑定k为↑
bind -r j select-pane -D # 绑定j为↓
bind -r h select-pane -L # 绑定h为←
bind -r l select-pane -R # 绑定l为→复制代码
-r
表示可重复按键,大概500ms以内,重复的h
、j
、k
、l
按键都将有效,完美支持了快速切换的Geek需求。
除了上下左右外, 还有几个快捷指令能够设置。
bind -r e lastp # 选择最后一个面板
bind -r ^e last # 选择最后一个窗口
bind -r ^u swapp -U # 与前一个面板交换位置
bind -r ^d swapp -D # 与后一个面板交换位置复制代码
习惯了全键盘操做后,命令的便捷性不言而喻。既然面板切换的指令均可以升级,面板大小调整的指令天然也不能落后。以下配置就能够升级你的操做:
# 绑定Ctrl+hjkl键为面板上下左右调整边缘的快捷指令
bind -r ^k resizep -U 10 # 绑定Ctrl+k为往↑调整面板边缘10个单元格
bind -r ^j resizep -D 10 # 绑定Ctrl+j为往↓调整面板边缘10个单元格
bind -r ^h resizep -L 10 # 绑定Ctrl+h为往←调整面板边缘10个单元格
bind -r ^l resizep -R 10 # 绑定Ctrl+l为往→调整面板边缘10个单元格复制代码
以上,resizep
即resize-pane
的别名。
当窗口中面板的数量逐渐增多时,每一个面板的空间就会逐渐减小。为了保证有足够的空间显示内容,tmux从v1.8版本起,提供了面板的最大化功能,输入tmux-prefix+z
,就能够最大化当前面板至窗口大小,只要再重复输入一次,便恢复正常。那么tmux v1.8如下的版本,怎么办呢?别急,有大神提供了以下的解决方案。
首先编写一个zoom脚本,该脚本经过新建一个窗口,交换当前面板与新的窗口默认面板位置,来模拟最大的功能;经过重复一次按键,还原面板位置,并关闭新建的窗口,来模拟还原功能,以下所示:
#!/bin/bash -f
currentwindow=`tmux list-window | tr '\t' ' ' | sed -n -e '/(active)/s/^[^:]*: *\([^ ]*\) .*/\1/gp'`;
currentpane=`tmux list-panes | sed -n -e '/(active)/s/^\([^:]*\):.*/\1/gp'`;
panecount=`tmux list-panes | wc | sed -e 's/^ *//g' -e 's/ .*$//g'`;
inzoom=`echo $currentwindow | sed -n -e '/^zoom/p'`;
if [ $panecount -ne 1 ]; then
inzoom="";
fi
if [ $inzoom ]; then
lastpane=`echo $currentwindow | rev | cut -f 1 -d '@' | rev`;
lastwindow=`echo $currentwindow | cut -f 2- -d '@' | rev | cut -f 2- -d '@' | rev`;
tmux select-window -t $lastwindow;
tmux select-pane -t $lastpane;
tmux swap-pane -s $currentwindow;
tmux kill-window -t $currentwindow;
else
newwindowname=zoom@$currentwindow@$currentpane;
tmux new-window -d -n $newwindowname;
tmux swap-pane -s $newwindowname;
tmux select-window -t $newwindowname;
fi复制代码
不妨将该脚本存放在~/.tmux
目录中(没有则新建目录),接下来只须要绑定一个快捷指令就行,以下。
unbind z
bind z run ". ~/.tmux/zoom"复制代码
经过上面的zoom脚本,面板能够轻松地最大化为一个新的窗口。那么反过来,窗口是否是能够最小化为一个面板呢?
试想这样一个场景:当你打开多个窗口后,而后想将其中几个窗口合并到当前窗口中,以便对比观察输出。
实际上,你的要求就是将其它窗口变成面板,而后合并到当前窗口中。对于这种操做,咱们能够在当前窗口,按下prefix
+ :
,打开命令行,而后输入以下命令:
join-pane -s window01 # 合并名称为window01的窗口的默认(第一个)面板到当前窗口中
join-pane -s window01.1 # .1显式指定了第一个面板,.2就是第二个面板(我本地将面板编号起始值设置为1,默认是0)复制代码
每次执行join-pane
命令都会合并一个面板,而且指定的窗口会减小一个面板,直到面板数量为0,窗口关闭。
除了在当前会话中操做外,join-pane
命令甚至能够从其它指定会话中合并面板,格式为join-pane -s [session_name]:[window].[pane]
,如join-pane -s 2:1.1
即合并第二个会话的第一个窗口的第一个面板到当前窗口,当目标会话的窗口和面板数量为0时,会话便会关闭。
注:上一节中的swap-pane
命令与join-pane
语法基本一致。
bind m command-prompt "splitw -h 'exec man %%'" # 绑定m键为在新的panel打开man
# 绑定P键为开启日志功能,以下,面板的输出日志将存储到桌面
bind P pipe-pane -o "cat >>~/Desktop/#W.log" \; display "Toggled logging to ~/Desktop/#W.log"复制代码
tmux会话中,Mac的部分命令如 osascript
、open
、pbcopy
或 pbpaste
等可能会失效(失效命令未列全)。
部分bug列表以下:
对此,咱们能够经过安装reattach-to-user-namespace
包装程序来解决这个问题。
brew install reattach-to-user-namespace复制代码
在~/.tmux.conf
中添加配置:
set -g default-command "reattach-to-user-namespace -l $SHELL"复制代码
这样你的交互式shell最终可以从新链接到用户级的命名空间。因为链接状态可以被子进程继承,故以上配置保证了全部从 shell 启动的命令可以被正确地链接。
有些时候,咱们可能会在不一样的操做系统中共享配置文件,若是你的tmux版本大于1.9,咱们还可使用if-shell
来判断是否Mac系统,而后再指定default-command
。
if-shell 'test "$(uname -s)" = Darwin' 'set-option -g default-command "exec reattach-to-user-namespace -l $SHELL"'复制代码
对于tmux v1.8及更早的版本,可使用以下包装后的配置:
set-option -g default-command 'command -v reattach-to-user-namespace >/dev/null && exec reattach-to-user-namespace -l "$SHELL" || exec "$SHELL"'复制代码
以上,$SHELL
对应于你的默认Shell,一般是/usr/bin/bash
或 /usr/local/bin/zsh
。
tmux中操做文本,天然离不开复制模式,一般使用复制模式的步骤以下:
`+[
进入复制模式空格键
开始复制,移动光标选择复制区域回车键
复制选中文本并退出复制模式`+]
粘贴文本查看复制模式默认的快捷键风格:
tmux show-window-options -g mode-keys # mode-keys emacs复制代码
默认状况下,快捷键为emacs
风格。
为了让复制模式更加方便,咱们能够将快捷键设置为熟悉的vi
风格,以下:
setw -g mode-keys vi # 开启vi风格后,支持vi的C-d、C-u、hjkl等快捷键复制代码
除了快捷键外,复制模式的启用、选择、复制、粘贴等按键也能够向vi
风格靠拢。
bind Escape copy-mode # 绑定esc键为进入复制模式
bind -t vi-copy v begin-selection # 绑定v键为开始选择文本
bind -t vi-copy y copy-selection # 绑定y键为复制选中文本
bind p pasteb # 绑定p键为粘贴文本(p键默认用于进入上一个窗口,不建议覆盖)复制代码
以上,绑定 v
、y
两键的设置只在tmux v2.4版本如下才有效,对于v2.4及以上的版本,绑定快捷键须要使用 -T
选项,发送指令须要使用 -X
选项,请参考以下设置:
bind -T copy-mode-vi v send-keys -X begin-selection
bind -T copy-mode-vi y send-keys -X copy-selection-and-cancel复制代码
tmux复制操做的内容默认会存进buffer
里,buffer
是一个粘贴缓存区,新的缓存老是位于栈顶,它的操做命令以下:
tmux list-buffers # 展现全部的 buffers
tmux show-buffer [-b buffer-name] # 显示指定的 buffer 内容
tmux choose-buffer # 进入 buffer 选择页面(支持jk上下移动选择,回车选中并粘贴 buffer 内容到面板上)
tmux set-buffer # 设置buffer内容
tmux load-buffer [-b buffer-name] file-path # 从文件中加载文本到buffer缓存
tmux save-buffer [-a] [-b buffer-name] path # 保存tmux的buffer缓存到本地
tmux paste-buffer # 粘贴buffer内容到会话中
tmux delete-buffer [-b buffer-name] # 删除指定名称的buffer复制代码
以上buffer操做在不指定buffer-name时,默认处理是栈顶的buffer缓存。
在tmux会话的命令行输入时,能够省略上述tmux前缀,其中list-buffers的操做以下所示:
choose-buffer的操做以下所示:
默认状况下,buffers内容是独立于系统粘贴板的,它存在于tmux进程中,且能够在会话间共享。
存在于tmux进程中的buffer缓存,虽然能够在会话间共享,但不能直接与系统粘贴板共享,难免有些遗憾。幸运的是,如今咱们有成熟的方案来实现这个功能。
在Linux上使用粘贴板
一般,Linux中可使用xclip
工具来接入系统粘贴板。
首先,须要安装xclip
。
sudo apt-get install xclip复制代码
而后,.tmux.conf
的配置以下。
# buffer缓存复制到Linux系统粘贴板
bind C-c run " tmux save-buffer - | xclip -i -sel clipboard"
# Linux系统粘贴板内容复制到会话
bind C-v run " tmux set-buffer \"$(xclip -o -sel clipboard)\"; tmux paste-buffer"复制代码
按下prefix
+ Ctrl
+ c
键,buffer缓存的内容将经过xlip
程序复制到粘贴板,按下prefix
+ Ctrl
+ v
键,tmux将经过xclip
访问粘贴板,而后由set-buffer命令设置给buffer缓存,最后由paste-buffer粘贴到tmux会话中。
在Mac上使用粘贴板
咱们都知道,Mac自带 pbcopy
和 pbpaste
命令,分别用于复制和粘贴,但在tmux命令中它们却不能正常运行。这里我将详细介绍下缘由:
Mac的粘贴板服务是在引导命名空间注册的。命名空间存在层次之分,更高级别的命名空间拥有访问低级别命名空间(如root引导命名空间)的权限,反之却不行。流程建立的属于Mac登陆会话的一部分,它会被自动包含在用户级的引导命名空间中,所以只有用户级的命名空间才能访问粘贴板服务。tmux使用守护进程(3)库函数建立其服务器进程,在Mac OS X 10.5中,苹果改变了守护进程(3)的策略,将生成的过程从最初的引导命名空间移到了根引导命名空间。而根引导命名空间访问权限较低,这意味着tmux服务器,和它的子进程,一同失去了原引导命名空间的访问权限(即无权限访问粘贴板服务)。
如此,咱们可使用一个小小的包装程序来从新链接到合适的命名空间,而后执行访问用户级命名空间的粘贴板服务,这个包装程序就是reattach-to-user-namespace
。
那么,Mac下.tmux.conf
的配置以下:
# buffer缓存复制到Mac系统粘贴板
bind C-c run "tmux save-buffer - | reattach-to-user-namespace pbcopy"
# Mac系统粘贴板内容复制到会话
bind C-v run "reattach-to-user-namespace pbpaste | tmux load-buffer - \; paste-buffer -d"复制代码
reattach-to-user-namespace
做为包装程序来访问Mac粘贴板,按下prefix
+ Ctrl
+ c
键,buffer缓存的内容将复制到粘贴板,按下prefix
+ Ctrl
+ v
键,粘贴板的内容将经过 load-buffer 加载,而后由 paste-buffer 粘贴到tmux会话中。
为了在复制模式中使用Mac系统的粘贴板,可作以下配置:
# 绑定y键为复制选中文本到Mac系统粘贴板
bind-key -T copy-mode-vi 'y' send-keys -X copy-pipe-and-cancel 'reattach-to-user-namespace pbcopy'
# 鼠标拖动选中文本,并复制到Mac系统粘贴板
bind-key -T copy-mode-vi MouseDragEnd1Pane send -X copy-pipe-and-cancel "pbcopy"复制代码
完成以上配置后记得重启tmux服务器。至此,复制模式中,按y
键将保存选中的文本到Mac系统粘贴板,随后按Command
+ v
键即可粘贴。
信息时代,数据尤其重要。tmux保护现场的能力依赖于tmux进程,若是进程退出,则意味着会话数据的丢失,所以关机重启后,tmux中的会话将被清空,这不是咱们想要见到的。幸运的是,目前有这样两款插件:Tmux Resurrect
和 Tmux Continuum
,能够永久保存tmux会话(它们均适用于tmux v1.9及以上版本)。
Tmux Resurrect无须任何配置,就可以备份tmux会话中的各类细节,包括窗口、面板的顺序、布局、工做目录,运行程序等等数据。所以它能在系统重启后彻底地恢复会话。因为其幂等的恢复机制,它不会试图去恢复一个已经存在的窗口或者面板,因此,即便你不当心多恢复了几回会话,它也不会出现问题,这样主动恢复时咱们就没必要担忧手抖多按了一次。另外,若是你是tmuxinator用户,我也建议你迁移到 tmux-resurrect插件上来,具体请参考Migrating from tmuxinator
。
Tmux Resurrec安装过程以下所示:
cd ~/.tmux
mkdir plugins
git clone https://github.com/tmux-plugins/tmux-resurrect.git复制代码
安装后需在~/.tmux.conf
中增长一行配置:
run-shell ~/.tmux/plugins/tmux-resurrect/resurrect.tmux复制代码
至此安装成功,按下prefix + r
重载tmux配置。
Tmux Resurrec提供以下两个操做:
prefix
+ Ctrl + s
,tmux状态栏在保存开始,保存后分别提示"Saving...","Tmux environment saved !"。prefix
+ Ctrl + r
,tmux状态栏在恢复开始,恢复后分别提示"Restoring...","Tmux restore complete !"。保存时,tmux会话的详细信息会以文本文件的格式保存到~/.tmux/resurrect
目录,恢复时则今后处读取,因为数据文件是明文的,所以你彻底能够自由管理或者编辑这些会话状态文件(若是备份频繁,记得按期清除历史备份)。
可选的配置
Tmux Resurrec自己是免配置开箱即用的,但同时也提供了以下选项以便修改其默认设置。
set -g @resurrect-save 'S' # 修改保存指令为S
set -g @resurrect-restore 'R' 修改恢复指令为R
# 修改会话数据的保持路径,此处不能使用除了$HOME, $HOSTNAME, ~以外的环境变量
set -g @resurrect-dir '/some/path'复制代码
默认状况下只有一个保守的列表项(即vi vim nvim emacs man less more tail top htop irssi mutt
)能够恢复,对此 Restoring programs doc 解释了怎么去恢复额外的项目。
进阶的备份
除了基础备份外,Tmux Resurrec还提供进阶的备份功能,以下所示:
进阶的备份功能默认不开启,须要特别配置。
1)恢复vim 和 neovim 会话,须要完成以下两步:
经过vim的vim-obsession插件保存vim/neovim会话。
cd ~/.vim/bundle
git clone git://github.com/tpope/vim-obsession.git
vim -u NONE -c "helptags vim-obsession/doc" -c q复制代码
在~/.tmux.conf
中增长两行配置:
set -g @resurrect-strategy-vim 'session' # for vim
set -g @resurrect-strategy-nvim 'session' # for neovim复制代码
2)恢复面板内容,需在~/.tmux.conf
中增长一行配置:
set -g @resurrect-capture-pane-contents 'on' # 开启恢复面板内容功能复制代码
目前使用该功能时,请确保tmux的default-command
没有包含&&
或者||
操做符,不然将致使bug。(查看default-command
的值,请使用命令tmux show -g default-command
。)
3)恢复shell的历史记录,需在~/.tmux.conf
中增长一行配置:
set -g @resurrect-save-shell-history 'on'复制代码
因为技术的限制,保存时,只有无前台任务运行的面板,它的shell历史记录才能被保存。
可能你嫌手动保存和恢复太过麻烦,别担忧,这不是问题。Tmux Continuum 在 Tmux Resurrec的基础上更进一步,如今保存和恢复所有自动化了,如你所愿,能够无感使用tmux,不用再担忧备份问题。
Tmux Continuum安装过程以下所示(它依赖Tmux Resurrect,请保证已安装Tmux Resurrect插件):
cd ~/.tmux/plugins
git clone https://github.com/tmux-plugins/tmux-continuum.git复制代码
安装后需在~/.tmux.conf
中增长一行配置:
run-shell ~/.tmux/plugins/tmux-continuum/continuum.tmux复制代码
Tmux Continuum默认每隔15mins备份一次,我设置的是一天一次:
set -g @continuum-save-interval '1440'复制代码
关闭自动备份,只需设置时间间隔为 0
便可:
set -g @continuum-save-interval '0'复制代码
想要在tmux启动时就恢复最后一次保存的会话环境,需增长以下配置:
set -g @continuum-restore 'on' # 启用自动恢复复制代码
若是不想要启动时自动恢复的功能了,直接移除上面这行就行。想要绝对肯定自动恢复不会发生,就在用户根目录下建立一个tmux_no_auto_restore
空文件(建立命令:touch ~/tmux_no_auto_restore
),该文件存在时,自动恢复将不触发。
对于tmux高级用户(可能就是你)而言,同时运行多个tmux服务器也是有可能的。你可能并不但愿后面启用的几个tmux服务器自动恢复或者自动保存会话。所以Tmux Continuum会优先在第一个启用的tmux服务器中生效,随后启用的tmux服务器再也不享受自动恢复或自动保存会话的待遇。
实际上,无论Tmux Continuum功能有没有启用,或者多久保存一次,咱们都有办法从状态栏知晓。Tmux Continuum提供了一个查看运行状态的插值#{continuum_status}
,它支持status-right
和 status-left
两种状态栏设置,以下所示:
set -g status-right 'Continuum status: #{continuum_status}'复制代码
tmux运行时,#{continuum_status}
将显示保存的时间间隔(单位为分钟),此时状态栏会显示:
Continuum status: 1440复制代码
若是其自动保存功能关闭了,那么状态栏会显示:
Continuum status: off复制代码
借助Tmux Continuum插件,Mac重启时,咱们甚至能够选择在Terminal
或者 iTerm2
中自动全屏启用tmux。
为此,需在~/.tmux.conf
中增长一行配置:
set -g @continuum-boot 'on'复制代码
Mac下,自动启用tmux还支持以下选项:
set -g @continuum-boot-options 'fullscreen'
,Terminal
自动全屏,tmux命令在Terminal
中执行。set -g @continuum-boot-options 'iterm'
, iTerm2
替换 Terminal
应用,tmux命令在iTerm2
中执行。set -g @continuum-boot-options 'iterm,fullscreen'
,iTerm2
自动全屏,tmux命令在iTerm2
中执行。Linux中则没有这些选项,它只能设置为自动启用tmux服务器。
以上,咱们直接安装了tmux插件。这没有问题,可当插件愈来愈多时,咱们就会须要统一的插件管理器。所以官方提供了tpm(支持tmux v1.9及以上版本)。
tpm安装过程以下所示:
cd ~/.tmux/plugins
git clone https://github.com/tmux-plugins/tpm复制代码
安装后需在~/.tmux.conf
中增长以下配置:
# 默认须要引入的插件
set -g @plugin 'tmux-plugins/tpm'
set -g @plugin 'tmux-plugins/tmux-sensible'
# 引入其余插件的示例
# set -g @plugin 'github_username/plugin_name' # 格式:github用户名/插件名
# set -g @plugin 'git@github.com/user/plugin' # 格式:git@github插件地址
# 初始化tmux插件管理器(保证这行在~/.tmux.conf的很是靠后的位置)
run '~/.tmux/plugins/tpm/tpm'复制代码
而后按下prefix + r
重载tmux配置,使得tpm生效。
基于tpm插件管理器,安装插件仅需以下两步:
~/.tmux.conf
中增长新的插件,如set -g @plugin '...'
。prefix
+ I
键下载插件,并刷新tmux环境。更新插件,请按下prefix
+ U
键,选择待更新的插件后,回车确认并更新。
卸载插件,需以下两步:
~/.tmux.conf
中移除插件所在行。prefix
+ alt
+ u
移除插件。tmux多会话链接实时同步的功能,使得结对编程成为了可能,这也是开发者最喜欢的功能之一。如今就差一步了,就是借助tmate把tmux会话分享出去。
tmate是tmux的管理工具,它能够轻松的建立tmux会话,而且自动生成ssh连接。
安装tmate
brew install tmate复制代码
使用tmate新建一个tmux会话
tmate复制代码
此时屏幕下方会显示ssh url,以下所示:
查看tmate生成的ssh连接
tmate show-messages复制代码
生成的ssh url以下所示,其中一个为只读,另外一个可编辑。
使用tmate远程共享tmux会话,受制于多方的网络质量,必然会存在些许延迟。若是共享会话的多方拥有同一个远程服务器的帐号,那么咱们可使用组会话
解决这个问题。
先在远程服务器上新建一个公共会话,命名为groupSession
。
tmux new -s groupSession复制代码
其余用户不去直接链接这个会话,而是经过建立一个新的会话来加入上面的公共会话groupSession
。
tmux new -t groupSession -s otherSession复制代码
此时两个用户均可以在同一个会话里操做,就会好像第二个用户链接到了groupSession
的会话同样。此时两个用户均可以建立新建的窗口,新窗口的内容依然会实时同步,可是其中一个用户切换到其它窗口,对另一个用户没有任何影响,所以在这个共享的组会话中,用户各自的操做能够经过新建窗口来执行。即便第二个用户关闭otherSession
会话,共享会话groupSession
依然存在。
组会话在共享的同时,又保留了相对的独立,很是适合结对编程场景,它是结对编程最简单的方式,若是帐号不能共享,咱们就要使用下面的方案了。
开始以前咱们须要确保用户对远程服务器上同一个目录拥有相同的读写权限,假设这个目录为/var/tmux/
。
使用new-session(简写new)建立会话时,使用的是默认的socket位置,默认socket没法操做,因此咱们须要建立一个指定socket文件的会话。
tmux -S /var/tmux/sharefile复制代码
另外一个用户进入时,须要指定socket文件加入会话。
tmux -S /var/tmux/sharefile attach复制代码
这样,两个不一样的用户就能够共享同一个会话了。
一般状况下,不一样的用户使用不一样的配置文件来建立会话,可是,使用指定socket文件建立的tmux会话,会话加载的是第一个建立会话的用户的~/.tmux.conf
配置文件,随后加入会话的其余用户,依然使用同一份配置文件。
要想tmux更加人性化、性能更佳,不妨参考下以下配置。
set -g base-index 1 # 设置窗口的起始下标为1
set -g pane-base-index 1 # 设置面板的起始下标为1复制代码
set -g status-utf8 on # 状态栏支持utf8
set -g status-interval 1 # 状态栏刷新时间
set -g status-justify left # 状态栏列表左对齐
setw -g monitor-activity on # 非当前窗口有内容更新时在状态栏通知
set -g status-bg black # 设置状态栏背景黑色
set -g status-fg yellow # 设置状态栏前景黄色
set -g status-style "bg=black, fg=yellow" # 状态栏前景背景色
set -g status-left "#[bg=#FF661D] ❐ #S " # 状态栏左侧内容
set -g status-right 'Continuum status: #{continuum_status}' # 状态栏右侧内容
set -g status-left-length 300 # 状态栏左边长度300
set -g status-right-length 500 # 状态栏左边长度500
set -wg window-status-format " #I #W " # 状态栏窗口名称格式
set -wg window-status-current-format " #I:#W#F " # 状态栏当前窗口名称格式(#I:序号,#w:窗口名称,#F:间隔符)
set -wg window-status-separator "" # 状态栏窗口名称之间的间隔
set -wg window-status-current-style "bg=red" # 状态栏当前窗口名称的样式
set -wg window-status-last-style "fg=red" # 状态栏最后一个窗口名称的样式
set -g message-style "bg=#202529, fg=#91A8BA" # 指定消息通知的前景、后景色复制代码
默认状况下,tmux中使用vim编辑器,文本内容的配色和直接使用vim时有些差距,此时须要开启256 colors的支持,配置以下。
set -g default-terminal "screen-256color"复制代码
或者:
set -g default-terminal "tmux-256color"复制代码
或者启动tmux时增长参数-2
:
alias tmux='tmux -2' # Force tmux to assume the terminal supports 256 colours复制代码
tmux默认会自动重命名窗口,频繁的命令行操做,将频繁触发重命名,比较浪费CPU性能,性能差的计算机上,问题可能更为明显。建议添加以下配置关闭rename机制。
setw -g automatic-rename off
setw -g allow-rename off复制代码
tmux默认会同步同一个会话的操做到全部会话链接的终端窗口中,这种同步机制,限制了窗口的大小为最小的会话链接。所以当你开一个大窗口去链接会话时,实际的窗口将自动调整为最小的那个会话链接的窗口,终端剩余的空间将填充排列整齐的小圆点,以下所示。
为了不这种问题,咱们能够在链接会话的时候,断开其余的会话链接。
tmux a -d复制代码
若是已经进入了tmux会话中,才发现这种问题,这个时候能够输入命令达到一样的效果。
`: a -d复制代码
tmux做为终端复用软件,支持纯命令行操做也是其一大亮点。你既能够启用可视化界面建立会话,也能够运行脚本生成会话,对于tmux依赖者而言,编写几个tmux脚本批量维护会话列表,快速重启、切换、甚至分享部分会话都是很是方便的。可能会有人说为何不用Tmux Resurrect呢?是的,Tmux Resurrect很好,一键恢复也很诱人,可是对于一个维护大量tmux会话的用户而言,一键恢复可能不见得好,分批次恢复多是他(她)更想要的,脚本化的tmux就很好地知足了这点。
脚本中建立tmux会话时,因为不须要开启可视化界面,须要输入-d
参数指定会话后台运行,以下。
tmux new -s init -d # 后台建立一个名称为init的会话复制代码
新建的会话,建议重命令会话的窗口名称,以便后续维护。
# 重命名init会话的第一个窗口名称为service
tmux rename-window -t "init:1" service复制代码
如今,能够在刚才的窗口中输入指令了。
# 切换到指定目录并运行python服务
tmux send -t "init:service" "cd ~/workspace/language/python/;python2.7 server.py" Enter复制代码
一个面板占用一个窗口可能太浪费了,咱们来分个屏吧。
# 默认上下分屏
tmux split-window -t "init:service"
# 切换到指定目录并运行node服务
tmux send -t "init:service" 'cd ~/data/louiszhai/node-webserver/;npm start' Enter复制代码
如今一个窗口拥有上下两个面板,是时候建立一个新的窗口来运行更多的程序了。
# 新建一个名称为tool的窗口
tmux neww -a -n tool -t init # neww等同于new window
# 运行weinre调试工具
tmux send -t "init:tool" "weinre --httpPort 8881 --boundHost -all-" Enter复制代码
另外新建窗口运行程序,有更方便的方式,好比使用 processes
选项。
tmux neww-n processes ls # 新建窗口并执行命令,命令执行结束后窗口将关闭
tmux neww-n processes top # 因为top命令持续在前台运行,所以窗口将保留,直到top命令退出复制代码
新的窗口,咱们尝试下水平分屏。
# 水平分屏
tmux split-window -h -t "init:tool"
# 切换到指定目录并启用aria2 web管理后台
tmux send -t "init:tool" "cd ~/data/tools/AriaNg/dist/;python -m SimpleHTTPServer 10108" Enter复制代码
相似的脚本,咱们能够编写一打,这样快速重启、切换、甚至分享会话都将更加便捷。
开机自动准备工做环境是一个很好的idea,但却很差实现。对于程序员而言,一个开机即用的计算机会节省大量的初始化操做,特别是前端工程师,本地经常会启用多个服务器,每次开机挨个启动将耗时耗力。为此,在遇到tmux以前,我经常拖延重启计算机的时机,一度连续运行Mac一月之久,直到它不堪重负。
有了tmux脚本化的基础,开机自动启用web服务器就不在话下了,接杯水的时间,计算机就重启恢复了满血。以下是操做步骤:
首先,上面的tmux脚本,能够合并到同一个文件中,指定文件权限为可执行,并命名为init.sh
(名称可自取)。
chmod u+x ./init.sh复制代码
而后,打开 系统偏好设置
- 用户与群组
- 登陆项
,点击添加按钮+
,选择刚刚保存的init.sh
脚本,最终效果以下:
至此,Mac开机将自动运行 init.sh
脚本,自动启用web服务器。
完成了上面这些配置,就真正实现了一键开机。
最后,附上我本地的配置文件 .tmux.conf,以及启动脚本 init.sh。
版权声明:转载需注明做者和出处。
本文做者: louis
本文连接: louiszhai.github.io/2017/09/30/…
相关文章