Shell入门:掌握Linux,OS X,Unix的Shell环境
================================================================================
在Linux或类Unix系统中,每个用户和进程都运行在一个特定环境中。这个环境包含了变量、设置、别名、函数以及更多的东西。下面是对Shell环境下一些常用命令的简单介绍,包括每个命令如何使用的例子,以及在命令行下设定你自己的环境来提高效率。

![](http://s0.cyberciti.org/uploads/cms/2015/01/bash-shell-welcome-image.jpg)

### 找出你当前的shell ###

在终端应用中输入下面命令中的任意一个:

    ps $$
    ps -p $$

或者

    echo "$0"

输出范例:

[![图1: Finding out your shell name](http://s0.cyberciti.org/uploads/cms/2015/01/finding-your-shell-like-a-pro.jpg)][1]

*图1:找出当前的shell*

### 找出所有已安装的shell ###

找到已安装shell的完整路径:

    type -a zsh
    type -a ksh
    type -a sh
    type -a bash

输出范例:

[![Fig.02: Finding out your shell path](http://s0.cyberciti.org/uploads/cms/2015/01/finding-and-verifying-shell-path.jpg)][2]

*图2:找出shell的路径*

文件/etc/shells里包含了系统所支持的shell列表。每一行代表一个shell,是相对根目录的完整路径。用这个[cat命令][3]来查看这些数据:

    cat /etc/shells

输出范例:

    # List of acceptable shells for chpass(1).
    # Ftpd will not allow users to connect who are not using
    # one of these shells.
     
    /bin/bash
    /bin/csh
    /bin/ksh
    /bin/sh
    /bin/tcsh
    /bin/zsh
    /usr/local/bin/fish

### 临时改变当前shell ###

只需要输入shell的名字。在下面的例子里,我从bash切换到了zsh:

    zsh

这只是临时改变了系统shell。也叫做子shell。要从子/临时shell退出,输入下面的命令或者按下CTRL-D:

    exit

### 找出子shell的层级或临时shell的嵌套层级 ###

每个bash实例启动后,变量$SHLVL的值都会加一。输入下面的命令:

    echo "$SHLVL"

示例输出:

[![Fig. 03: Bash shell nesting level (subshell numbers)](http://s0.cyberciti.org/uploads/cms/2015/01/a-nested-shell-level-command.jpg)][4]

*图3:Bash shell嵌套层级(子shell数目)*

### 通过chsh命令永久变更系统shell ###

想要把当前系统shell从bash永久换成zsh?试试这个:

    chsh -s /bin/zsh

想把其他用户的shell从bash永久换成ksh?试试这个:

    sudo chsh -s /bin/ksh userNameHere

### 查看当前的环境变量 ###

你需要用到:

    env
    env | more
    env | less
    env | grep 'NAME'

示例输出:

    TERM_PROGRAM=Apple_Terminal
    SHELL=/bin/bash
    TERM=xterm-256color
    TMPDIR=/var/folders/6x/45252d6j1lqbtyy_xt62h40c0000gn/T/
    Apple_PubSub_Socket_Render=/tmp/launch-djaOJg/Render
    TERM_PROGRAM_VERSION=326
    TERM_SESSION_ID=16F470E3-501C-498E-B315-D70E538DA825
    USER=vivek
    SSH_AUTH_SOCK=/tmp/launch-uQGJ2h/Listeners
    __CF_USER_TEXT_ENCODING=0x1F5:0:0
    PATH=/usr/bin:/bin:/usr/sbin:/sbin:/usr/local/bin:/opt/X11/bin:/usr/local/go/bin:/usr/local/sbin/modemZapp:/Users/vivek/google-cloud-sdk/bin
    __CHECKFIX1436934=1
    PWD=/Users/vivek
    SHLVL=2
    HOME=/Users/vivek
    LOGNAME=vivek
    LC_CTYPE=UTF-8
    DISPLAY=/tmp/launch-6hNAhh/org.macosforge.xquartz:0
    _=/usr/bin/env
    OLDPWD=/Users/vivek

下面是bash shell里一些常见变量的列表:

![Fig.04: Common bash environment variables](http://s0.cyberciti.org/uploads/cms/2015/01/common-shell-vars.jpg)

*图4:常见bash环境变量*

> **注意**:下面这些环境变量没事不要乱改。很可能会造成不稳定的shell会话:
> 
> SHELL
> 
> UID
> 
> RANDOM
> 
> PWD
> 
> PPID
> 
> SSH_AUTH_SOCK
> 
> USER
> 
> HOME
> 
> LINENO

### 显示环境变量的值 ###

使用下面任意一条命令显示环境变量HOME的值:

    ## 使用printenv ##
    printenv HOME
     
    ## 或者用echo ##
    echo "$HOME"
     
    # 考虑到可移植性,也可以用printf ##
    printf "%s\n" "$HOME"

示例输出:

    /home/vivek

### 增加或设定一个新环境变量 ###

下面是bash,zsh,sh和ksh的语法:

    ## 语法 ##
    VAR=value
    FOO=bar
     
    ## 设定vim为默认文本编辑器 ##
    EDITOR=vim
    export $EDITOR
     
    ## 考虑安全性,设定默认shell连接超时时间 ##
    TMOUT=300
    export TMOUT
     
    ## 你可以直接使用export命令设定命令的搜素路径 ##
    export PATH=$PATH:$HOME/bin:/usr/local/bin:/path/to/mycoolapps

然后,使用printenv或者echo或printf命令查看环境变量PATH,EDITOR,和TMOUT的值:

    printenv PATH
    echo "$EDITOR"
    printf "%s\n" $TMOUT

### 怎么修改一个现有的环境变量? ###

下面是语法:

    export VAR=value
    ## 或者 ##
    VAR=value
    export $VAR
     
    ## 把默认文本编辑器从vim改为emacs ##
    echo "$EDITOR" ## <--- 屏幕输出vim
    EDITOR=emacs   ## <--- 修改
    export $EDITOR ## <--- 让修改在其他会话生效
    echo "$EDITOR" ## <--- 屏幕输出emacs 

**tcsh shell下增加和修改变量**的语法是下面这样的:

    ## 语法
    setenv var value
    printenv var
     
    ## 设置变量foo的值为bar ##
    setenv foo bar
    echo "$foo"
    printenv foo
     
    ## 设置变量PATH ##
    setenv PATH $PATH\:$HOME/bin
    echo "$PATH"
     
    ## 设置变量PAGER ##
    setenv PAGER most
    printf "%s\n" $PAGER

### 找出bash shell的配置文件 ###

用下面的命令列出bash shell的文件:

    ls -l ~/.bash* ~/.profile /etc/bash* /etc/profile

示例输出:

[![Fig.05: List all bash environment configuration files](http://s0.cyberciti.org/uploads/cms/2015/01/list-bash-enviroment-variables.jpg)][5]

*图5:列出bash的所有配置文件*

要查看所有的bash配置文件,输入:

    less ~/.bash* ~/.profile /etc/bash* /etc/profile

可以使用文字编辑器比如vim或emacs来一个一个编辑bash配置文件:

    vim ~/.bashrc

编辑/etc/目录下的文件,输入:

    ## 首先是备份,以防万一
    sudo cp -v /etc/bashrc /etc/bashrc.bak.22_jan_15
     
    ########################################################################
    ## 然后,随心所欲随便改吧,好好玩玩shell环境或者提高一下效率:)                 ##
    ########################################################################
    sudo vim /etc/bashrc

### 被Bash shell初始化过程中应用的文件搞糊涂了吗? ###

下面的"bash初始化文件"流程图应该有些帮助:

![](http://s0.cyberciti.org/uploads/cms/2015/01/BashStartupfiles.jpg)

根据账户设定的默认shell,你的用户配置或系统配置可能是下面其中一种:

### 找出zsh shell配置文件 ###

zsh的[wiki][6]中建议用下面的命令:

    strings =zsh | grep zshrc

示例输出:

    /etc/zshrc
    .zshrc

输入下面的命令列出你的zsh shell文件:

    ls -l /etc/zsh/* /etc/profile ~/.z*

查看所有zsh配置文件:

    less /etc/zsh/* /etc/profile ~/.z*

### 找出ksh shell配置文件 ###

1. 查看~/.profile或者/etc/profile文件。

### 找出tcsh shell配置文件 ###

1. C shell查看~/.login,~/.cshrc文件。
2. TC shell查看~/.tcshrc和~/.cshrc文件。

### 我可以写个类似这样每次登录时都自动执行的脚本吗? ###

是的,把你的命令或别名或其他设定添加到~/.bashrc(bash shell)或者~/.profile(sh/ksh/bash)或者~/.login(csh/tcsh)文件中。

### 我可以写个类似这样每次登出都自动执行的脚本吗? ###

是的,把你的命令或别名或其他设定添加到~/.bash_logout(bash)或者~/.logout(csh/tcsh)文件。

### history:获取关于shell会话的更多信息 ###

输入history命令来查看本次会话的历史:

    history

示例输出:

        9  ls
       10  vi advanced-cache.php
       11  cd ..
       12  ls
       13  w
       14  cd ..
       15  ls
       16  pwd
       17  ls
       ....
       ..
       ...
       91  hddtemp /dev/sda
       92  yum install hddtemp
       93  hddtemp /dev/sda
       94  hddtemp /dev/sg0
       95  hddtemp /dev/sg1
       96  smartctl -d ata -A /dev/sda | grep -i temperature
       97  smartctl -d ata -A /dev/sg1 | grep -i temperature
       98  smartctl  -A /dev/sg1 | grep -i temperature
       99  sensors

输入history 20来查看命令历史的后20条:

    history 20

示例输出:

[![Fig.06: View session history in the bash shell using history command](http://s0.cyberciti.org/uploads/cms/2015/01/history-outputs.jpg)][7]

*图6:在bash shell中使用history命令查看会话历史*

你可以重复使用之前的命令。简单地按下[上]或[下]方向键就可以查看之前的命令。在shell提示符下按下[CTRL-R]可以向后搜索历史缓存或文件来查找命令。重复最后一次命令,只需要在shell提示符下输入!!就好了:

    ls -l /foo/bar
    !!

在以上的历史记录中找到命令#93 (hddtemp /dev/sda),输入:

    !93

### 使用sudo或su改变用户 ###

下面是语法:

    su userName
     
    ## 登录为tom用户 ##
    su tom
     
    ## 为用户tom打开一个新的shell会话 ##
    su tom
     
    ## 登录为root用户 ##
    su -
     
    ## sudo命令语法(必须在系统中配置有这个命令) ##
    sudo -s
    sudo tom

看看帖子"[Linux下使用其他用户身份运行命令][8]"更多地了解sudo,su和runuser命令。

### shell别名 ###

别名仅仅是命令的一个快捷方式。

### 列出所有的别名 ###

输入下面的命令:

    alias

示例输出:

    alias ..='cd ..'
    alias ...='cd ../../../'
    alias ....='cd ../../../../'
    alias .....='cd ../../../../'
    alias .4='cd ../../../../'
    alias .5='cd ../../../../..'
    alias bc='bc -l'
    alias cd..='cd ..'
    alias chgrp='chgrp --preserve-root'
    alias chmod='chmod --preserve-root'
    alias chown='chown --preserve-root'
    alias cp='cp -i'
    alias dnstop='dnstop -l 5 eth1'
    alias egrep='egrep --color=auto'
    alias ethtool='ethtool eth1'

### 设定一个别名 ###

bash/zsh语法:

    alias c='clear'
    alias down='sudo /sbin/shutdown -h now'

对于命令clear可以输入c别名,这样我们就可以输入c代替clear命令来清空屏幕:

    c

或者输入down来关闭基于Linux的服务器:

    down

你可以设定任意多的别名。看下"[Linux/Unix/Mac OS X系统中的30个方便的bash shell别名][9]"了解在类Unix系统中别名的实际应用。

### shell函数 ###

Bash/ksh/zsh函数允许你更进一步地配置shell环境。在这个例子中,我写了一个简单的名叫memcpu()的bash函数,用来显示前10个最占用CPU和内存的进程:

    memcpu() { echo "*** Top 10 cpu eating process ***"; ps auxf | sort -nr -k 3 | head -10;
    echo  "*** Top 10 memory eating process ***"; ps auxf | sort -nr -k 4 | head -10;  }

输入memcpu就可以在屏幕上看到下面的信息:

    memcpu
     
    *** Top 10 cpu eating process ***
    nginx    39559 13.0  0.2 264020 35168 ?        S    04:26   0:00      \_ /usr/bin/php-cgi
    nginx    39545  6.6  0.1 216484 13088 ?        S    04:25   0:04      \_ /usr/bin/php-cgi
    nginx    39471  6.2  0.6 273352 81704 ?        S    04:22   0:17      \_ /usr/bin/php-cgi
    nginx    39544  5.7  0.1 216484 13084 ?        S    04:25   0:03      \_ /usr/bin/php-cgi
    nginx    39540  5.5  0.1 221260 19296 ?        S    04:25   0:04      \_ /usr/bin/php-cgi
    nginx    39542  5.4  0.1 216484 13152 ?        S    04:25   0:04      \_ /usr/bin/php-cgi
    nixcraft 39543  5.3  0.1 216484 14096 ?        S    04:25   0:04      \_ /usr/bin/php-cgi
    nixcraft 39538  5.2  0.1 221248 18608 ?        S    04:25   0:04      \_ /usr/bin/php-cgi
    nixcraft 39539  5.0  0.1 216484 16272 ?        S    04:25   0:04      \_ /usr/bin/php-cgi
    nixcraft 39541  4.8  0.1 216484 14860 ?        S    04:25   0:04      \_ /usr/bin/php-cgi
     
    *** Top 10 memory eating process ***
    498      63859  0.5  4.0 2429652 488084 ?      Ssl   2014 177:41 memcached -d -p 11211 -u memcached -m 2048 -c 18288 -P /var/run/memcached/memcached.pid -l 10.10.29.68 -L
    mysql    64221  4.2  3.4 4653600 419868 ?      Sl    2014 1360:40  \_ /usr/libexec/mysqld --basedir=/usr --datadir=/var/lib/mysql --user=mysql --log-error=/var/log/mysqld.log --open-files-limit=65535 --pid-file=/var/run/mysqld/mysqld.pid --socket=/var/lib/mysql/mysql.sock
    nixcraft 39418  0.4  1.1 295312 138624 ?       S    04:17   0:02  |   \_ /usr/bin/php-cgi
    nixcraft 39419  0.5  0.9 290284 113036 ?       S    04:18   0:02  |   \_ /usr/bin/php-cgi
    nixcraft 39464  0.7  0.8 294356 99200 ?        S    04:20   0:02  |   \_ /usr/bin/php-cgi
    nixcraft 39469  0.3  0.7 288400 91256 ?        S    04:20   0:01  |   \_ /usr/bin/php-cgi
    nixcraft 39471  6.2  0.6 273352 81704 ?        S    04:22   0:17      \_ /usr/bin/php-cgi
    vivek    39261  2.2  0.6 253172 82812 ?        S    04:05   0:28      \_ /usr/bin/php-cgi
    squid     9995  0.0  0.5 175152 72396 ?        S     2014  27:00  \_ (squid) -f /etc/squid/squid.conf
    cybercit  3922  0.0  0.4 303380 56304 ?        S    Jan10   0:13  |   \_ /usr/bin/php-cgi
 
看下"[如何编写和应用shell函数][10]"了解更多信息。

### 综合一下:定制你自己的Linux或Unix bash shell工作环境 ###

现在,你将使用bash shell配置自己的环境。我只介绍bash。但是理论上zsh,ksh和其他常用shell都差不多。让我们看看如何调整shell来适合我作为系统管理员的需求。编辑你的~/.bashrc文件来附加设定。下面是一些常用的配置选项。

#### #1: 设定bash路径和环境变量 ####

    # 设定路径 ##
    export PATH=$PATH:/usr/local/bin:/home/vivek/bin:/opt/firefox/bin:/opt/oraapp/bin
     
    # 为cd命令设定路径
    export CDPATH=.:$HOME:/var/www
 
使用less或more命令作为翻页器:

    export PAGER=less

设定vim作为默认文本编辑器:

    export EDITOR=vim
    export VISUAL=vim
    export SVN_EDITOR="$VISUAL"

设定Oracle数据库特别要求的参数:

    export ORACLE_HOME=/usr/lib/oracle/xe/app/oracle/product/10.2.0/server
    export ORACLE_SID=XE
    export NLS_LANG=$($ORACLE_HOME/bin/nls_lang.sh)

设定JAVA_HOME和其他java路径,比如java版本:

    export JAVA_HOME=/usr/lib/jvm/java-6-sun/jre
     
    # 把ORACLE和JAVA加入到PATH里
    export PATH=$PATH:$ORACLE_HOME/bin:$JAVA_HOME/bin

[使用密钥实现免密码登录][11]让ssh远程登录更安全:

    # 再也不用输密码了
    /usr/bin/keychain $HOME/.ssh/id_rsa
    source $HOME/.keychain/$HOSTNAME-sh

最后,[打开bash命令补齐][12]

    source /etc/bash_completion

#### #2: 设定bash命令提示符 ####

设定[定制的bash提示符(PS1)][13]:

    PS1='{\u@\h:\w }\$ '

#### #3: 设定默认文件权限 ####

    ## 设定默认权限为644 ##
    umask 022

#### #4: 调整shell命令历史设定 ####

    # 不往命令历史里写入相同的行
    HISTCONTROL=ignoreboth
     
    # 忽略这些命令
    HISTIGNORE="reboot:shutdown *:ls:pwd:exit:mount:man *:history"
     
    # 通过HISTSIZE和HISTFILESIZE设定命令历史的长度
    export HISTSIZE=10000
    export HISTFILESIZE=10000
     
    # 为命令历史文件增加时间戳
    export HISTTIMEFORMAT="%F %T "
     
    # 附加到命令历史文件,而不是覆盖
    shopt -s histappend

#### #5: 设定shell会话的时区 ####

    ## 为我自己的shell会话设定IST(印度标准时间) ##
    TZ=Asia/Kolkata

#### #6: 设定shell行编辑接口 ####

    ## 使用vi风格的行编辑接口,替代bash默认的emacs模式 ##
    set -o vi

#### #7: 设定自己喜好的别名 ####

    ## 增加一些保护 ##
    alias rm='rm -i'
    alias cp='cp -i'
    alias mv='mv -i'
     
    ## Memcached ##
    alias mcdstats='/usr/bin/memcached-tool 10.10.29.68:11211 stats'
    alias mcdshow='/usr/bin/memcached-tool 10.10.29.68:11211 display'
    alias mcdflush='echo "flush_all" | nc 10.10.29.68 11211'
     
    ## 默认命令参数 ##
    alias vi='vim'
    alias grep='grep --color=auto'
    alias egrep='egrep --color=auto'
    alias fgrep='fgrep --color=auto'
    alias bc='bc -l'
    alias wget='wget -c'
    alias chown='chown --preserve-root'
    alias chmod='chmod --preserve-root'
    alias chgrp='chgrp --preserve-root'
    alias rm='rm -I --preserve-root'
    alias ln='ln -i'

下面是一些额外的OS X Unix bash shell别名:

    # 从bash打开桌面应用
    alias preview="open -a '$PREVIEW'"
    alias safari="open -a safari"
    alias firefox="open -a firefox"
    alias chrome="open -a google\ chrome"
    alias f='open -a Finder '
     
    # 清理那些.DS_Store文件
    alias dsclean='find . -type f -name .DS_Store -delete'

#### #8: 寡人好色 ####

    # 彩色的grep输出 
    alias grep='grep --color=auto'
    export GREP_COLOR='1;33'
     
    # 彩色的ls
    export LSCOLORS='Gxfxcxdxdxegedabagacad'
    # Gnu/linux的ls
    ls='ls --color=auto'
     
    # BSD/os x的ls命令
    # alias ls='ls -G'

#### #9: 设定自己喜好的bash函数 ####

    # 在屏幕上显示10个最近的历史命令
    function ht {
      history | awk '{a[$2]++}END{for(i in a){print a[i] " " i}}' | sort -rn | head
    }
     
    # host和ping命令的替代
    # 接受http:// 或 https:// 或 ftps:// 名称用作域或主机名
    _getdomainnameonly(){
    	local h="$1"
    	local f="${h,,}"
    	# remove protocol part of hostname
            f="${f#http://}"
            f="${f#https://}"
    	f="${f#ftp://}"
    	f="${f#scp://}"
    	f="${f#scp://}"
    	f="${f#sftp://}"
    	# remove username and/or username:password part of hostname
    	f="${f#*:*@}"
    	f="${f#*@}"
    	# remove all /foo/xyz.html*  
    	f=${f%%/*}
    	# show domain name only
    	echo "$f"
    }
     
     
    ping(){
    	local array=( $@ )  		# get all args in an array
    	local len=${#array[@]}          # find the length of an array
    	local host=${array[$len-1]}     # get the last arg
    	local args=${array[@]:0:$len-1} # get all args before the last arg in $@ in an array 
    	local _ping="/bin/ping"
    	local c=$(_getdomainnameonly "$host")
    	[ "$t" != "$c" ] && echo "Sending ICMP ECHO_REQUEST to \"$c\"..."
    	# pass args and host
    	$_ping $args $c
    }
     
    host(){
    	local array=( $@ )
    	local len=${#array[@]}
    	local host=${array[$len-1]}
    	local args=${array[@]:0:$len-1}
    	local _host="/usr/bin/host"
    	local c=$(_getdomainnameonly "$host")
    	[ "$t" != "$c" ] && echo "Performing DNS lookups for \"$c\"..."
      	$_host $args $c
    }

#### #10: 通过shell shopt命令设定bash shell行为 ####

最后,你可以[使用set和shopt命令调整bash shell环境][14]:

    # 目录拼写纠正
    shopt -q -s cdspell
     
    # 保证每次终端窗口改变大小后会更新显示
    shopt -q -s checkwinsize
     
    # 打开高级模式匹配功能
    shopt -q -s extglob
     
    # 退出时附加命令历史而不是覆盖
    shopt -s histappend
     
    # 在命令历史使用多行
    shopt -q -s cmdhist
     
    # 在后台任务结束时立刻通知
    set -o notify
     
    # 禁用[CTRL-D]来结束shell
    set -o ignoreeof

### 总结 ###

这个帖子不难理解。它简短地将如何定制用户环境从头介绍了一下。要深入了解bash/ksh/zsh/csh/tcsh/的能力,我建议你用下面的命令阅读man文档:

    man bash
    man zsh
    man tcsh
    man ksh

> 这篇文章由Aadrika T. J.贡献;由admin编辑并增加了额外内容。你也可以[为nixCraft做出贡献][15]。

--------------------------------------------------------------------------------

via: http://www.cyberciti.biz/howto/shell-primer-configuring-your-linux-unix-osx-environment/

作者:[nixCraft][a]
译者:[zpl1025](https://github.com/zpl1025)
校对:[wxy](https://github.com/wxy)

本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](http://linux.cn/) 荣誉推出

[a]:http://www.cyberciti.biz/tips/about-us
[1]:http://www.cyberciti.biz/howto/shell-primer-configuring-your-linux-unix-osx-environment/attachment/finding-your-shell-like-a-pro/
[2]:http://www.cyberciti.biz/howto/shell-primer-configuring-your-linux-unix-osx-environment/attachment/finding-and-verifying-shell-path/
[3]:http://www.cyberciti.biz/faq/linux-unix-appleosx-bsd-cat-command-examples/
[4]:http://www.cyberciti.biz/howto/shell-primer-configuring-your-linux-unix-osx-environment/attachment/a-nested-shell-level-command/
[5]:http://www.cyberciti.biz/howto/shell-primer-configuring-your-linux-unix-osx-environment/attachment/list-bash-enviroment-variables/
[6]:http://zshwiki.org/home/config/files
[7]:http://www.cyberciti.biz/howto/shell-primer-configuring-your-linux-unix-osx-environment/attachment/history-outputs/
[8]:http://www.cyberciti.biz/open-source/command-line-hacks/linux-run-command-as-different-user/
[9]:http://www.cyberciti.biz/tips/bash-aliases-mac-centos-linux-unix.html
[10]:http://bash.cyberciti.biz/guide/Chapter_9:_Functions
[11]:http://www.cyberciti.biz/faq/ssh-passwordless-login-with-keychain-for-scripts/
[12]:http://www.cyberciti.biz/faq/fedora-redhat-scientific-linuxenable-bash-completion/
[13]:http://www.cyberciti.biz/tips/howto-linux-unix-bash-shell-setup-prompt.html
[14]:http://bash.cyberciti.biz/guide/Setting_shell_options
[15]:http://www.cyberciti.biz/write-for-nixcraft/