终极 Shell

zsh

在开始今天的 MacTalk 之前,先问两个问题吧:

1、相对于其他系统,Mac 的主要优势是什么?
2、你们平时用哪种 Shell?
……

第一个童靴可以坐下了,Mac 的最大优势是 GUI 和命令行的完美结合,不要把所有注意力放在 Mac 性感的腰身和明媚的显示屏上好吧,这不是妹纸!第二个童靴你可以出去面壁了,讲了这么多期 MacTalk 你告诉我还在用 Windows 的 cmd, 你让 Mac 君情何以堪?哪怕你就说在用 Linux 的 Bash 我也就原谅你了,踢飞!

上次在「如何学习一门编程语言」里提到了 Shell,也有读者问到 Shell 的问题,所以这次给大家说说 Shell 的事。

我在「趣谈个人建站」里介绍过一点 Shell,自己的东西借用下不丢人,把扯淡的拿掉,干货留下,就是如下内容:

Shell是Linux/Unix的一个外壳,你理解成衣服也行。它负责外界与Linux内核的交互,接收用户或其他应用程序的命令,然后把这些命令转化成内核能理解的语言,传给内核,内核是真正干活的,干完之后再把结果返回用户或应用程序。

Linux/Unix提供了很多种Shell,为毛要这么多Shell?难道用来炒着吃么?那我问你,你同类型的衣服怎么有那么多件?花色,质地还不一样。写程序比买衣服复杂多了,而且程序员往往负责把复杂的事情搞简单,简单的事情搞复杂。牛程序员看到不爽的Shell,就会自己重新写一套,慢慢形成了一些标准,常用的Shell有这么几种,sh、bash、csh等,想知道你的系统有几种shell,可以通过以下命令查看:

cat /etc/shells

显示如下:

/bin/bash
/bin/csh
/bin/ksh
/bin/sh
/bin/tcsh
/bin/zsh

在 Linux 里执行这个命令和 Mac 略有不同,你会发现 Mac 多了一个 zsh,也就是说 OS X 系统预装了个 zsh,这是个神马 Shell 呢?

目前常用的 Linux 系统和 OS X 系统的默认 Shell 都是 bash,但是真正强大的 Shell 是深藏不露的 zsh, 这货绝对是马车中的跑车,跑车中的飞行车,史称『终极 Shell』,但是由于配置过于复杂,所以初期无人问津,很多人跑过来看看 zsh 的配置指南,什么都不说转身就走了。直到有一天,国外有个穷极无聊的程序员开发出了一个能够让你快速上手的zsh项目,叫做「oh my zsh」,Github 网址是:https://github.com/robbyrussell/oh-my-zsh。这玩意就像「X天叫你学会 C++」系列,可以让你神功速成,而且是真的。

好,下面我们看看如何安装、配置和使用 zsh。

安装zsh


如果你用 Mac,就可以直接看下一节
如果你用 Redhat Linux,执行:sudo yum install zsh
如果你用 Ubuntu Linux,执行:sudo apt-get install zsh
如果你用 Windows……去洗洗睡吧。

安装完成后设置当前用户使用 zsh:chsh -s /bin/zsh,根据提示输入当前用户的密码就可以了。

安装oh my zsh


首先安装 git,安装方式同上,把 zsh 换成 git 即可。

安装「oh my zsh」可以自动安装也可以手动安装。

自动安装:

wget https://github.com/robbyrussell/oh-my-zsh/raw/master/tools/install.sh -O - | sh

手动安装:

git clone git://github.com/robbyrussell/oh-my-zsh.git ~/.oh-my-zsh
cp ~/.oh-my-zsh/templates/zshrc.zsh-template ~/.zshrc

都不复杂,安装完成之后退出当前会话重新打开一个终端窗口,你就可以见到这个彩色的提示了:

zsh1

配置


zsh 的配置主要集中在用户当前目录的.zshrc里,用 vim 或你喜欢的其他编辑器打开.zshrc,在最下面会发现这么一行字:

# Customize to your needs…

可以在此处定义自己的环境变量和别名,当然,oh my zsh 在安装时已经自动读取当前的环境变量并进行了设置,你可以继续追加其他环境变量。

接下来进行别名的设置,我自己的部分配置如下:

alias cls='clear'
alias ll='ls -l'
alias la='ls -a'
alias vi='vim'
alias javac="javac -J-Dfile.encoding=utf8"
alias grep="grep --color=auto"
alias -s html=mate   # 在命令行直接输入后缀为 html 的文件名,会在 TextMate 中打开
alias -s rb=mate     # 在命令行直接输入 ruby 文件,会在 TextMate 中打开
alias -s py=vi       # 在命令行直接输入 python 文件,会用 vim 中打开,以下类似
alias -s js=vi
alias -s c=vi
alias -s java=vi
alias -s txt=vi
alias -s gz='tar -xzvf'
alias -s tgz='tar -xzvf'
alias -s zip='unzip'
alias -s bz2='tar -xjvf'

zsh 的牛粪之处在于不仅可以设置通用别名,还能针对文件类型设置对应的打开程序,比如:

alias -s html=mate,意思就是你在命令行输入 hello.html,zsh会为你自动打开 TextMat 并读取 hello.html; alias -s gz='tar -xzvf',表示自动解压后缀为 gz 的压缩包。

总之,只有想不到,木有做不到,吓尿了吧。

设置完环境变量和别名之后,基本上就可以用了,如果你是个主题控,还可以玩玩 zsh 的主题。在 .zshrc 里找到ZSH_THEME,就可以设置主题了,默认主题是:

ZSH_THEME=”robbyrussell”

oh my zsh 提供了数十种主题,相关文件在~/.oh-my-zsh/themes目录下,你可以随意选择,也可以编辑主题满足自己的变态需求,我采用了默认主题robbyrussell,不过做了一点小小的改动:

PROMPT='%{$fg_bold[red]%}➜ %{$fg_bold[green]%}%p%{$fg[cyan]%}%d %{$fg_bold[blue]%}$(git_prompt_info)%{$fg_bold[blue]%}% %{$reset_color%}>'
#PROMPT='%{$fg_bold[red]%}➜ %{$fg_bold[green]%}%p %{$fg[cyan]%}%c %{$fg_bold[blue]%}$(git_prompt_info)%{$fg_bold[blue]%} % %{$reset_color%}'

对照原来的版本,我把 c 改为 d,c 表示当前目录,d 表示绝对路径,另外在末尾增加了一个「 > 」,改完之后的效果是这样的:

zsh2

大家可以尝试进行改造,也算个趣事。

最后我们来说说插件。

插件


oh my zsh 项目提供了完善的插件体系,相关的文件在~/.oh-my-zsh/plugins目录下,默认提供了100多种,大家可以根据自己的实际学习和工作环境采用,想了解每个插件的功能,只要打开相关目录下的 zsh 文件看一下就知道了。插件也是在.zshrc里配置,找到plugins关键字,你就可以加载自己的插件了,系统默认加载 git ,你可以在后面追加内容,如下:

plugins=(git textmate ruby autojump osx mvn gradle)

下面简单介绍几个:

1、git:当你处于一个 git 受控的目录下时,Shell 会明确显示 「git」和 branch,如上图所示,另外对 git 很多命令进行了简化,例如 gco=’git checkout’、gd=’git diff’、gst=’git status’、g=’git’等等,熟练使用可以大大减少 git 的命令长度,命令内容可以参考~/.oh-my-zsh/plugins/git/git.plugin.zsh

2、textmate:mr可以创建 ruby 的框架项目,tm finename 可以用 textmate 打开指定文件。

3、osx:tab 增强,quick-look filename 可以直接预览文件,man-preview grep 可以生成 grep手册 的pdf 版本等。

4、autojump:zsh 和 autojump 的组合形成了 zsh 下最强悍的插件,今天我们主要说说这货。

首先安装autojump,如果你用 Mac,可以使用 brew 安装:

brew install autojump

如果是 Linux,去下载 autojump 的最新版本,比如:

git clone git://github.com/joelthelion/autojump.git

解压缩后进入目录,执行

./install.py

最后把以下代码加入.zshrc:

[[ -s ~/.autojump/etc/profile.d/autojump.sh ]] && . ~/.autojump/etc/profile.d/autojump.sh

至此,安装、配置、插件三位一体,终极 Shell 全面登场。退出终端会话重新登录,开始感受 zsh 的训疾如风!

使用 zsh


1、兼容 bash,原来使用 bash 的兄弟切换过来毫无压力,该咋用咋用。

2、强大的历史纪录功能,输入 grep 然后用上下箭头可以翻阅你执行的所有 grep 命令。

3、智能拼写纠正,输入gtep mactalk * -R,系统会提示:zsh: correct ‘gtep’ to ‘grep’ [nyae]? 比妹纸贴心吧,她们向来都是让你猜的……

4、各种补全:路径补全、命令补全,命令参数补全,插件内容补全等等。触发补全只需要按一下或两下 tab 键,补全项可以使用 ctrl+n/p/f/b上下左右切换。比如你想杀掉 java 的进程,只需要输入 kill java + tab键,如果只有一个 java 进程,zsh 会自动替换为进程的 pid,如果有多个则会出现选择项供你选择。ssh + 空格 + 两个tab键,zsh会列出所有访问过的主机和用户名进行补全

5、智能跳转,安装了autojump之后,zsh 会自动记录你访问过的目录,通过 j + 目录名 可以直接进行目录跳转,而且目录名支持模糊匹配和自动补全,例如你访问过hadoop-1.0.0目录,输入j hado 即可正确跳转。j –stat 可以看你的历史路径库。

6、目录浏览和跳转:输入 d,即可列出你在这个会话里访问的目录列表,输入列表前的序号,即可直接跳转。

7、在当前目录下输入 .. 或 … ,或直接输入当前目录名都可以跳转,你甚至不再需要输入 cd 命令了。

8、通配符搜索:ls -l **/*.sh,可以递归显示当前目录下的 shell 文件,文件少时可以代替 find,文件太多就歇菜了。

9、更强的别名:请参考配置一节。

10、插件支持:请参考插件一节。

……

看完这篇文章,你就知道,zsh一出,无人再与争锋!终极二字不是盖的。

如果你是个正在使用 shell程序员,如果你依然准备使用 bash,那就去面壁和忏悔吧,别说你订阅过 MacTalk!

感谢那位开发了 oh my zsh 的无聊程序员,他可能没有因此收获物质上的利益,但是他的代码提升了无数程序员的效率,节省了大量的时间,我们说,程序员改变世界!

MacTalk 写的多了,朋友和家人都不太理解,这玩意赚不了钱争不了名,写来何用?我想,能够学习、编码、写作,并做出一点帮助别人的东西,总好过炒股炒房的价值吧,尽管人家比咱「富」多了……

136 thoughts on “终极 Shell

  1. 手动安装oh my zsh ,安装完成之后退出启动还是bash,
    发现少了一条命令。(顺便求字体)

    4. Set zsh as your default shell:
    chsh -s /bin/zsh

  2. 如果使用 brew install autojump 自动安装 autojump 的话,那么根据 brew 的提示,.zshrc 中需要增加的代码应该是:

    [[ -s `brew --prefix`/etc/autojump.sh ]] && . `brew –prefix`/etc/autojump.sh

    撸主的配置只是适合手工安装的情况。
    希望其他童鞋注意一下,否则 zsh 会提示找不到 j 命令。

  3. 强哥,
    alias -s txt=Sublime Text 2
    alias -s txt=’Sublime Text 2′
    alias -s txt=”Sublime Text 2″

    zsh: command not found: Sublime
    这个怎么处理,zsh 不识别空格

  4. 居然有这么强大的好东西,果断得一试啊!就像刚接触vim的时候一样,好东西就是不管遇到什么语言,总能发挥出最强大的威力,用了是不舍得丢掉的,强哥,谢了啊!

  5. Pingback: 半月杂记 2013/07/26 | Kevin's Blog

  6. 谢谢你的文章,我已经开始尝试将shell换成zsh了。
    此外,
    [[ -s ~/.autojump/etc/profile.d/autojump.sh ]] && . ~/.autojump/etc/profile.d/autojump.sh
    这个配置不是每个人都适用的,这样更通用一些:
    [[ -s brew --prefix/etc/autojump.sh ]] && . brew --prefix/etc/autojump.sh

  7. 请问强哥,我是ubuntu小白,最近用的过程中发现:
    原来bash下使用到一个没有安装的package会提示,例如:
    The program ‘svn’ is currently not installed. You can install it by typing:
    sudo apt-get install subversion

    现在zsh没有提示了,只是说:
    zsh: command not found: svn

    怎么能把提示的功能加上去?
    谢谢了

  8. 请教:在arch上装了,然后PS1显示的是一个方框,konsole里面正常,是一个->,这个怎么处理?上次在bash搞powerline,也是折腾了半天,加上rxvt-unicode-256color就好了,这次不行……

  9. ➜ bin sh brew install autojump
    Warning: autojump-21.6.9 already installed, it’s just not linked
    ➜ bin cd
    ➜ ~ j -stat
    zsh: command not found: j

    求助,“it’s just not linked” 是怎么回事呢,跟着后面不能用‘j’

  10. 博主,最近换zsh,但是会出现不少问题,其中一个如下:用bash可以find到需要的文件,而用zsh却不能。麻烦解释下原因。

    ➜ /home/haoaina521 >sudo find / -name libstdc++.so.6*
    zsh: no matches found: libstdc++.so.6*
    ➜ /home/haoaina521 >bash
    haoaina521@haoaina521-R480-R440:~$ sudo find / -name libstdc++.so.6*
    [sudo] password for haoaina521:
    /usr/lib/i386-linux-gnu/libstdc++.so.6
    /usr/lib/i386-linux-gnu/libstdc++.so.6.0.16
    /usr/lib/libstdc++.so.6
    /usr/lib/libstdc++.so.6.0.16
    /home/haoaina521/Programes/iNodeClient_Linux/iNodeClient/libs/std/libstdc++.so.6
    haoaina521@haoaina521-R480-R440:~$

    • 好像是’‘ 匹配问题,我也遇到了类似的问题
      sudo apt-get remove –purge printer-driver
      ;#卸载打印  zsh为什么不能匹配那个’*',bash却可以。是我哪里配置错了么?

        • 上面的'*'被抽调了,呜呜,图片也显示不出来

          ➜ /home/yxj >sudo apt-get remove --purge printer-driver*
          zsh: no matches found: printer-driver*
          ➜ /home/yxj >bash
          yxj@yxj-Mint ~ $ sudo apt-get remove --purge printer-driver*
          正在读取软件包列表... 完成
          正在分析软件包的依赖关系树
          正在读取状态信息... 完成
          注意,根据正则表达式 printer-driver* 选中了 printer-driver-cjet
          注意,根据正则表达式 printer-driver* 选中了 printer-driver-foo2zjs
          注意,根据正则表达式 printer-driver* 选中了 printer-driver-sag-gdi
          ...

  11. 在zsh中能支持通配符搜索,如果使用javac需要编译指定某个目录及其子目录下的所有java源码, 能支持javac com/**.java类似这样的命令嘛?

  12. 我安装了zsh和autojump。我的系统是:
    LSB Version: :core-3.1-ia32:core-3.1-noarch:graphics-3.1-ia32:graphics-3.1-noarch
    Distributor ID: CentOS
    Description: CentOS release 5.4 (Final)
    Release: 5.4
    Codename: Final

    但是很奇怪的是:
    (1) 没有jumpstat命令
    (2)在用j命令的时候,j无法记忆我去过的目录,我必须使用j -a /xx/xx/xx 才能加进入,然后删除的话竟然也无法删除(使用j –purge)。另外一个疑问是如何删除j -s 中的单条目录(比如我自己加进去写错了)

    非常感谢~

  13. Pingback: Homfen's Blog | oh-my-zsh安装配置

  14. Mr池,我按照这个步骤设置,在 Mac 上安装完 oh-my-zsh 之后,发现找不到.zshrc文件,在 robbysussell 的 git 上看用cp ~/.oh-my-zsh/templates/zshrc.zsh-template ~/.zshrc 建立新 config,发现目录里也没有这个文件,不知道是咋回事啊?
    另外/.oh-my-zsh目录在 finder 里是没法查看的吗?

  15. Pingback: Mac 程序员的十种武器 | | Evolution UnitEvolution Unit

  16. Pingback: Mac 程序员的十种武器 | amaker

  17. Pingback: Better shell in Mac OS X: zsh - nonocast

  18. Pingback: ubuntu下web开发环境搭建 | 新世纪Linux社区

  19. 用zsh遇到了一个搞死人的问题。

    如果我在terminal(不用zsh)中跑gradle clean test,完全没问题。

    但是如果用zsh,则有一个包含了中文字符的junit测试无论如果都过不了。

    我开始以为是代码的问题,搞了好久啊。。。。。。

  20. 我开始认为是jersey的问题,后来认为是gradle的问题,还开了一个repo来试图重现问题:https://github.com/cuipengfei/reproduce-weird-issues/tree/master/gradle-encoding

    楼主有遇到过类似的encoding问题吗?

  21. Pingback: 在 Mac OS X 下打造赏心悦目的开发环境 – Cloud&Matrix | 查问题

  22. wushantekiMacBook-Pro:~ wushan$ git clone git://github.com/robbyrussell/oh-my-zsh.git ~/.oh-my-zsh
    fatal: destination path ‘/Users/wushan/.oh-my-zsh’ already exists and is not an empty directory.
    wushantekiMacBook-Pro:~ wushan$ cp ~/.oh-my-zsh/templates/zshrc.zsh-template ~/.zshrc
    cp: invalid option — ‘g’
    Try ‘cp –help’ for more information.
    我复制的时候出现了这个问题该怎么办啊 搞不定啊

  23. Pingback: oh-my-zsh安装配置 | Homfen's Blog

  24. Pingback: 【CLI】zsh调研(2)配置和使用 – 剑客|关注科技互联网

  25. Pingback: 在OSX上配置好用的终端 | TalkCool

  26. Pingback: shell: 增强你的bash,提高命令效率之终极利器oh-my-zsh – osetc.com

  27. Pingback: Shell(一):功能、配置和插件 – 剑客|关注科技互联网

  28. Pingback: OS X 新手必备 - 云岩

  29. Pingback: mac下常用工具箱 | 学习笔记

  30. Pingback: Ubuntu安装配置使用zsh | b404

  31. Pingback: Mac 开发者常用的工具 - 拾味

  32. Pingback: Mac OS X 10.11 El Capitan 系统个人化设置参考 – Zhenyu's Homepage

  33. Pingback: 终极 Shell 不同体验的 Terminal – 钰雪

  34. Pingback: fedora下安装配置zsh shell | 瓜哥blog

  35. autojump 现在这个版本的在用brew安装完后加 [[ -s $(brew --prefix)/etc/profile.d/autojump.sh ]] && . $(brew –prefix)/etc/profile.d/autojump.sh 到.zshrc里

  36. Pingback: Shell脚本编程总结及速查手册-自由港

  37. Pingback: Shell脚本编程总结及速查手册 - code123

发表评论