shell入门(又)

摘录一些我不熟悉的:

* 命令行的历史记录

通过按向上方向键,您可以向后遍历近来在该控制台下输入的命令。用向下方向键可以向前遍历。与 SHIFT 键连用的话,您还可以遍历以往在该控制台中的输出。您也可以编辑旧的命令,然后再运行。

<CTRL r> 后,shell 就进入“reverse-i(ncremental)-search”(向后增量搜索)模式。现在输入您要找的命令的首字母:

(reverse-i-search)`':. 敲入 ‘i’可能会变成:

(reverse-i-search)`i': isdnctrl hangup ippp0

如果您再按 <ENTER> 键,上面的命令将再次执行。而如果您按了向右、向左方向键或 <ESC> , 上面的命令将回到普通的命令行,这样您就可以进行适当编辑。

* section index * top

* 编辑命令行

通过光标和功能键(Home、End 等键),您可以浏览并编辑命令行,如果您需要,还可以用键盘的快捷方式来完成一般的编辑:

  • <CTRL k>:删除从光标到行尾的部分
  • <CTRL u>:删除从光标到行首的部分
  • <ALT d>:删除从光标到当前单词结尾的部分
  • <CTRL w>:删除从光标到当前单词开头的部分
  • <CTRL a>:将光标移到行首
  • <CTRL e>:将光标移到行尾
  • <ALT a>:将光标移到当前单词头部
  • <ALT e>:将光标移到当前单词尾部
  • <CTRL y>:插入最近删除的单词
  • <!$>:重复前一个命令最后的参数。
    例如:您用命令 mkdir peter/pan/documents/tinkerbell 新建了一个目录,现在您向用命令‘cd’进入该目录,您可以用 cd !$,shell 将把前一个命令‘mkdir’的参数添加到现在的‘cd’后面。

当您更深入 Linux 的疆域后,将看到这些快捷方式在其他应用程序下输入时,有时也有效,比如,在浏览器中的输入框中。

* section index * top

* 可用的 Shell 快捷方式

Mandrake Linux 带有不少快捷方式,其中一部分是 bash 原来就有的,而还有一些则是为您预先设置的(在后面您将看到如何设置)

由于 home 目录是每位用户的活动中心,许多 Unix 对此有特殊的快捷方式。
‘~’就是您的 home 目录的简写形式。我们假设您在其他目录,想把一个名为‘sometext’的文件复制到您 home 目录下的 ‘docs’子目录中。除了输入:
cp sometext /home/myusername/docs
您还可以用简写:
cp sometext ~/docs
理论上,这也可以应用在命令‘cd’上。无论当前路径在哪里,cd ~ 将回到您的 home 目录。其实还可以简化,只要键入 cd ,就可以返回 home 目录了。

Mandrake Linux 为您提供了一些预先设置的快捷方式(称为‘别名’,aliases)

  • cd.. :将执行‘cd ..’(回到上一极目录)
  • d :将执行‘ls’(列出目录内容,list directory)
  • l :将执行‘ls’(列出目录内容,list directory)
  • la :将执行‘ls -a’(列出目录的所有内容,包括以点号开头的隐藏文件)
  • ll :将执行‘ls -l -k’(以长格式列出目录内容,包括一些文件属性,并以 KB 而不是 byte 为单位显示文件大小)
  • ls :将执行‘ls -F --color=auto’(列出目录内容,加上文件类型标识,并使用颜色)
  • md :将执行‘mkdir’(新建目录)
  • p :将执行‘cd -’(回到前一次所在的目录)
  • rd :将执行‘rmdir’(删除空的目录)
  • s :将执行‘cd ..’(回到上一级目录)
  • used :将执行‘du -sm * | sort -n’(由小到大排列文件或子目录所占磁盘的大小,以 MB 为单位)

现在,您应该对 shell 及一些快捷方式有了进一步的了解,下面我们来看看除了应用一些简单的命令,shell 还能作什么。



* 命令的排列

现在您将看到一些常用的命令排列。您可能想在一行中给出所有命令,然后就可以把注意力转移到其他地方。没问题,shell 允许您在不同的命令之间,放上特殊的排列字符 (queuing characters) 。 这儿将介绍最常用的两种。
请注意,为了看起来更清楚,我在这些字符两旁加了空格。而在实际应用中,您不一定要这么做,‘ls -a ; du -hs’和‘ls -a;du -hs’的效果是一样的。

command1 ; command2

先执行 command1 ,不管 command1 是否出错,接下来执行 command2
例如:
ls -a ; du -hs
将先在屏幕上列出目录中的所有内容,然后列出所有目录及其子目录所占磁盘大小。

command1 && command2

只有当 command1 正确运行完毕后,才执行 command2
例如:
$ ls -a bogusdir && du -hs
将返回 ls: bogusdir: No such file or directory ,而‘du’则根本没有运行(这是因为您没有‘bogusdir’目录)。如果您将符号换成了‘;’,‘du’将被执行。

为了进一步说明‘;’和‘&&’的区别,及一般命令排列的用处,下面举一个经典的例子:Linux 内核的编译和安装。
要编译、安装 Linux ,您需要执行一串命令:‘make dep’、‘make clean’、‘make bzImage’、‘make modules’、‘make modules_install’和‘make install’。如果要等一个命令完成后,再输入下一个,再等,再输入,……,那就太麻烦了。另一方面,每个命令只有当前面的命令都正确执行完毕后,才 能开始执行。如果您用‘;’来排列命令,则即使有命令执行失败,后面的也照常运行,最后,您可能在‘/boot’目录下得到一个有问题的内核映像(image)。而用‘&&’:

make dep && make clean && make bzImage && make modules && make modules_install && make install

不需要中途打断,就可以编译内核及其模块,并完成后面的安装。

* section index * top

* 命令的任务调度

当您在终端里运行一个命令或开启一个程序时,终端要等到命令或程序运行完毕后,才能再被使用。在 Unix 中,我们称这样的命令或程序在前台(foreground)运行。如果您想在终端下运行另一个命令,则需要再打开一个新的终端。

但这里还有一个更优雅的办法,称为任务调度(jobbing)或后台(backgrounding)。当您运用任务的调度或将命令置于后台,终端就立即解放了,这样一来,终端立即就可以接受新的输入。为实现这样的目的,您只需在命令后面添加一个 & :

gqview &

告诉 shell 将图片查看器‘GQview’放到后台去执行(即当成 job 来运行)
命令 jobs 将告诉您,在这个终端窗口中,运行着哪些命令与程序:

jobs

             [1]+ Running gqview &

当您要关闭终端窗口时,这一点就很重要,因为关闭终端将导致所有在其中运行的任务都将被中止,在此例中,如果您关闭了终端,由这个终端开启的 GQview 程序也将被关闭。

但如何将前台运行的一个程序放到后台去?没问题:

gqview
<CTRL z>

[2]+ Stopped gqview

bg

[2]+ gqview &

组合键 <CTRL z> 将挂起终端中正在运行的程序,然后您就可以用 bg 命令将其放到后台去执行。

请注意,在后台运行图形应用程序有时候是有用处的,这样可以在终端下显示这个程序的出错信息,虽然这对您可能没有直接的帮助,当如果碰到了麻烦,向别人询问时,这些出错提示就有用武之地了。

一些图形程序,很可能还处在测试期(Beta),尽管在后台执行,也会在终端中输出一些信息。如果您对此不满,可以用下面命令:

command >&/dev/null &

这不仅将程序送到后台执行,还将其输出发到‘/dev/null’文件。‘/dev/null’是系统的“碎纸机” (shredder),所有送到那里的信息都将消失殆尽。

* section index * top

* 命令的替换

命令替换(Command substitution)是一项很实用的功能。我们假设,您想看看 XFree86 文档中的 ‘README.mouse’文件,但您不知道这个文件的位置。但您是位机灵的用户,已经听说了‘locate’命令,也安装了‘slocate’包,您就可以用:

locate README.mouse

发现那个文件在‘/usr/X11R6/lib/X11/doc’。现在您就可以在终端里用‘less’或在文件管理器中进入那个目录然后读取文件。而命令替换可以给您带来一些便捷:

less $(locate README.mouse)

一步到位。命令‘locate README.mouse’的输出(= /usr/X11R6/lib/X11/doc/README.mouse)作为‘less’的参数,然后就可以显示文件内容了。

这种机制的语法是:

command1 $(command2)

除了‘$( )’,您还可以用后引号(backquote)

command1 `command2`

这样虽然可以减少输入,但可读性差,而且很容易就和没有替换功能的一般单引号混淆。我更欣赏前一种方法,但这最终起决于您。

这里有另外一个例子。我们假设,您打算结束一个名为‘rob’的程序。您先得用命令‘pidof’找出相应的进程号(   Process ID),然后以这个 PID 为参数,运行‘kill’命令,这样就可以结束‘rob’程序。除了用:

pidof rob
567
kill 567

您还可以试试:

kill `pidof rob`

怎么样,效率有所提高吧?

在下一篇中,我将接着介绍 shell 的另外两种实用的机制:文件名匹配、输出重定向。



* bash 配置文件

在您的 home 目录下,运行

ls .bash*

您将看到这些文件:

  • .bash_history :记录了您以前输入的命令,
  • .bash_logout :当您退出 shell 时,要执行的命令,
  • .bash_profile :当您登入 shell 时,要执行的命令,
  • .bashrc :每次打开新的 shell 时,要执行的命令。

请注意后两个的区别:‘.bash_profile’只在会话开始时被读取一次,而‘.bashrc’则每次打开新的终端(如新的 xterm 窗口)时,都要被读取。按照传统,您得将定义的变量,如 PATH ,放到‘.bash_profile’中,而象 aliases(别名)和函数之类,则放在‘.bashrc’。但由于‘.bash_profile’经常被设置成先读取‘.bashrc’的内容,您如果图省事的话,就把所有配置都放进‘.bashrc’。

这 些文件是每一位用户的设置。系统级的设置存储在‘/etc/profile’、‘/etc/bashrc’及目录‘/etc/profile.d’下的文 件中。但您得习惯用各自的配置文件:编辑不需要‘root’权限,还可以使您的设置更有个性。当系统级与用户级的设置发生冲突时,将采用用户的设置。

读取‘.bashrc’的内容,您如果要省点事的话,就把您所有的配置都放进‘.bashrc’。

上 面的这些文件是每位用户的设置,系统级的设置存储在‘/etc/profile’、‘/etc/bashrc’及目录‘/etc/profile.d’下 的文件中。您最好习惯使用各自的配置文件:编辑不需要‘root’权限,还可以使您的设置更具个性。当系统级与用户级的设置发生冲突时,将优先采用用户的 设置。

* section index * top

* 提示符

每次当您打开一个控制台(console)或 xterm 时,最先看到的就是提示符(prompt),类似于:

account@hostname ~ $

在默认设置下,提示符将显示您的用户名、主机名(默认是‘localhost’)、当前所在目录(在 Unix 中,‘~’表示您的 home 目录)
按照传统,最后一个字符可以标识您是普通用户($),还是‘root’(#)

您可以通过 $PS1 变量来设置提示符。命令

echo $PS1

将显示当前的设定。其中可用字符的含义在 man bash 的‘PROMPTING’部分有说明。

如何才能完成理想的设置呢?对于健忘的初学者来讲,默认设定有些不友好,因为提示符只显示当前目录的最后一部分。如果您看到象这样的提示符

tom@localhost bin $

您的当前目录可能是‘/bin’、‘/usr/bin’、‘/usr/local/bin’及‘/usr/X11R6/bin’。当然,您可以用

pwd (输出当前目录,print working directory)

能不能叫 shell 自动告诉您当前目录呢?
当然可以。这里我将提到的设定,包括提示符,大都包含在文件‘/etc/bashrc’中。您可以通过编辑各自 home 目录下的‘.bash_profile’和‘.bashrc’来改变设置。

man bash 中的‘PROMPTING’部分,对这些参数(parameter)有详细说明。您可以加入一些小玩意,如不同格式的当前时间,命令的历史记录号,甚至不同的颜色。

在‘~/.bashrc’中,我喜欢的设定是:

PS1="/[/033[1m/][/w]/[/033[0m/] "

‘root’在‘~/.bashrc’中的设定 是:

PS1="/[/033[0;31m/][/w]/[/033[0m/] "

这样我得到的提示符就是:

[/usr/bin]

当用‘root’时,变成:

[/usr/bin]

我已经除掉了主机名和用户名,因为我用不着这些。但我首先想一眼就能看出我的身份是普通用户还是‘root’。注意到,普通用户的提示符可以是黑底白字,或白底黑字。
要在终端上获得恰当的颜色调配, 您可以下载 这个脚本 ,赋予执行权限chmod +x color,然后再运行。

一种更适当的设定:

PS1="/u: /w//$ "

这样,提示符就变成:

user_name: /usr/bin$

您可以通过命令 export 来测试不同的设置(比如,export PS1="/u: /w//$ "。如果找到了适合的提示符,就将设置放到您的‘.bashrc'’中。这样,每次打开控制台或终端窗口时,都会生效。

您甚至可以给提示符设定主题(theme),也就是搭配不同的颜色,使其看起来象很棒的 ol 的 C64 提示符。如果您对此感兴趣,可以看一下 Bashish

* section index * top

* 改变 $PATH

‘$PATH’与‘$PS1’一样,也是环境变量。输入

set

将列出所有当前定义的环境变量。
您看到的这些环境变量在 shell 的配置文件中定义,可能是用户自己的配置文件,也可能是由‘root’通过‘/etc’下面的系统级文件定义的。如果您使用 X ,更多的一些变量将由 X 、您的窗口管理器或桌面环境的启动文件配置。

如 果对这些设置不很清楚,您暂时最好不要随便改动。了解如何改变 $PATH 变量很有用,因为这个变量决定了 shell 将到哪些目录中寻找命令或程序。如果要执行的命令的目录在 $PATH 中,您就不必输入这个命令的完整路径,直接输入命令就可以了。一些第三方软件没有将可执行文件放到 Linux 的标准目录中。因此,将这些非标准的安装目录添加到 $PATH 是一种解决的办法。此外,您也将看到如何处理一般的环境变量。

首先,作为惯例,所有环境变量名都是大写。由于 Linux 区分大小写,这点您要留意。当然,您可以自己定义一些变量,如‘$path’、‘$pAtH’,但 shell 不会理睬这些变量。
第二点是变量名有时候以‘$’开头,但有时又不是。当设置一个变量时,您直接用名称,而不需要加‘$’:

PATH=/usr/bin:/bin:/usr/local/bin:/usr/X11R6/bin

获取变量值的话,就要在变量名前加‘$’:

echo $PATH
/usr/bin:/bin:/usr/local/bin:/usr/X11R6/bin

否则的话,变量名就会被当作普通文本了:

echo PATH
PATH

处理 $PATH 变量要注意的第三点是:您不能只替换变量,而是要将新的字符串添加到原来的值中。在大多数情况下,您能用‘PATH=/some/directory’,因为这将删除 $PATH 中其他的所有目录,这样您在该终端运行程序时,就不得不给出完整路径。所以,只能作添加:

PATH=$PATH:/some/directory

这样,PATH 被设成当前的值(以 $PATH 来表示)+新添的目录。

到目前为止,您只为当前终端设置了新的 $PATH 变量。如果您打开一个新的终端,运行 echo $PATH , 将返回旧的 $PATH 值,而看不到您刚才添加的新目录。因为您先前定义的是一个局部环境变量(仅限于当前的终端)
要定义一个全局变量,使在以后打开的终端中生效,您需要将局部变量输出(export),可以用‘export’命令:

export PATH=$PATH:/some/directory

现在如果您打开一个新的终端,输入 echo $PATH ,也能看到新设置的 $PATH 了。请注意,命令‘export’只能改变当前终端及以后运行的终端里的变量。对于已经运行的终端没有作用。

为了将目录永久添加到您的 $PATH ,只要将‘export’的那行添加到您的‘.bash_profile’文件中。
请不要在‘.bashrc’中设置 PATH ,否则会导致 PATH 中目录的意外增长。您每次打开一个新的 shell ,‘.bashrc’都会作用。所以如果在该文件中添加目录,您每次打开一个终端,目录会被添加。这将导致 PATH 变量由于目录复制,不断地增长。



* 命令的别名、Shell 函数

记住所有的命令及各自带的可选项,然后每次一一输入,这确实有点枯燥。但幸运的是,您可以为常用命令定义快捷方式。这些快捷方式可以用较简单的命令别名(alias),或复杂一些的 shell 函数的语法来定义。

命令的别名

例如,我用下面的命令来上传 MUO 中的文件:

rsync -e ssh -z -t -r -vv --progress /home/tom/web/muo/rsmuo/docs muo:/www/mandrakeuser/docs

显然,如果每次都要逐一输入,那我早晚会变成木头。因此我在‘~/.bashrc’中定义了别名:

alias upmuo='rsync -e ssh -z -t -r -vv --progress /home/tom/web/muo/rsmuo/docs muo:/www/mandrakeuser/docs'

现在,我只要输入 upmuo 就可以完成上传任务了。

定义别名的语法是:

alias shortcut='command'

命令中有空格的话 ,就需要用引号(如在命令与可选项间就有空格)。请注意,您可以用单引号或双引号。如果您碰到这两者有不一样的地方,那就对了 ;-) 。

单引号将剥夺其中的所有字符的特殊含义,而双引号中的‘$’(参数替换)和‘`’(命令替换)例外。这意味着,如果您想在别名中应用变量或命令的替换,就得用双引号。看一下上面的例子,我在‘.bashrc’中定义了一个称为 MUOHOME 的变量:

export MUOHOME=$HOME/web/muo/rsmuo/docs

要在上面的别名中用上这个变量,我就必须用双引号:

alias upmuo="rsync -e ssh -z -t -r -vv --progress $MUOHOME muo:/www/mandrakeuser/docs"

否则,别名将查找一个名为‘$MUOHOME’的目录或文件。

您可以用‘alias’在命令行快速地创建别名,或将命令放到各自的‘~/.bashrc’,或放到系统级的‘/etc/profile.d/alias.sh’中(而在 Mandrake Linux 8 以前的版本里,用的是‘/etc/bashrc’)。要删除一个别名,只要输入:unalias alias 。只运行 alias 将列出您系统中所有定义的别名。

如果看一下‘~/.bashrc’和‘/etc/profile.d/alias.sh’,您会发现系统已经定义了一些别名。您可以为同一个命令定义多个别名。当然,您得先确认别名与其他程序名不同,比如象 alias rm='ls -l' 这样的就不能工作。您可以在命令行输入这些快捷方式,测试一下。如果 shell 找不到相同名称的命令,那您就可以将其用作别名了。

以下别名可能有用(不要忘了引号!)

  • alias rpmq='rpm -qa | grep' :现在 rpmq string 就将列出所有名称中含有 string 的已安装 RPM 包,
  • alias ls='ls -ho --color | more' : ls 将以彩色分页方式列出文件,文件大小以 KB为单位,
  • alias use='du --max-depth=1 | sort -n | more' : use 将子目录按大小排好,并以分页方式列出,
  • alias dkd='cd /usr/src/linux/Documentation' :常用的目录也可以用别名来表示。其他有希望的候选者如‘/mnt’中的子目录。
    目录的别名也可以是可移动的介质: alias dlm='/mnt/cdrom/Mandrake/RPMS/'

提示:将有相似功能的别名以相同字母开头,比如将所有目录的别名以‘d’作开头,这样有助于记忆。

我相信,您将会用到这些功能。

Shell 函数

写 shell 函数涉及到了 shell 脚本,超出了我们讨论的范围(也不在我的掌握范围之内 ;-))。事实上,shell 函数属于 shell 脚本,但可以在同一 shell 下被预载(preload)和执行(而一般的 shell 脚本至少要打开一个 sub-shell)

通过 shell 函数,您可以做很多 aliases 无法完成的事情。下面就是一个例子:

function apros() { apropos $1 | egrep -v '(3|/(n/)'; }

定义了一个新命令,称为‘apros’。apros name 将先执行‘apropos name(即在 man page 中搜索命令),然后将得到的输出送到管道(|),接着用‘egrep’过滤,排除第‘3’和第‘n’章节的 man page ,这个命令可能没什么大用处,但可以整理‘apropos’命令的输出。
函数允许您在函数内部任何位置,使用运行时的参数。而别名,则只允许在命令行尾放一个参数(比如前面的别名‘rpmq’)
‘$1’就是位置参数(positional parameter),表示函数第一个参数的位置标识符。依此类推,还有‘$2’等。

function apros() { apropos $1 | egrep -v "/($2"; }

如果您这样运行‘apros’命令:

apros name man_section_number

这个命令将搜索标题中含 name 的 man pages ,但排除 man_section_number 部分:

apros menu 3

将搜索标题含‘menu’的 man page ,但排除第三章节(关于编程的)。注意到您得引用(quote) 两次,而且还用到了双引号:

  • 您必须引用‘egrep’的搜索模式,这样可以不至于被 shell 误解。
  • 您必须用双引号,这样第二个参数才能被正确解释。
  • 您必须引用圆括号,这样使‘egrep’按字面意思对待对待参数。

是不是有点味道了?;-)

shell 函数的处理类似于别名:将其放到您的‘.bashrc’文件,这样就能永久生效了。

* section index * top

* 从这里出发

我 们谈到的只是 shell 的一个开头。掌握了shell 脚本,您就可以做很多事情,比如将任务自动化,纠正别人脚本中的错误,按照您的习惯定制 Mandrake Linux 系统。如果您打算学习某种复杂的编程语言,那 shell 脚本也是一个很好的开端,因为基本概念都是类似的。

BASH Programming - Introduction HOW-TO 将更深入这些主题,并且将把您带到 shell 编程的世界。然后可以继续阅读我强烈推荐的 Advanced Bash-Scripting Guide,作者是 Mendel Cooper 。

如 果您偏好纸书,那我推荐 S. Veeraraghavan 的《Teach Yourself Shell Programming》,Sams 出版社。我倒觉得 O'Reilly 公司由 Newham/Rosenblatt 写的《Learning the bash Shell》,不过尔尔,但这可能只有我这么看 ;-) 。

除了这些,就是练习,练习,再练习。阅读其他人写的 shell 脚本,看看他们在做什么,怎么做,为什么那样做。请不要用‘root’测试您的脚本。Have fun 。

 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值