2.5 Shell的基本元素 2.5.1 命令与参数
命令名称是命令行的第一个项目,后面跟着选项。选项可有可无,还可以合并,如ls -lt比ls -l -t更方便。长选项在标准工具GNU版本,以及X Window System中使用的越来越普遍,如patch --verbose --backup。分号;可用来分隔同一行里的多条命令,Shell会依次执行这些命令。用&符号,则Shell将在后台执行其前面的命令,Shell不用等待该命令完成就可以继续执行下一个命令。
Shell识别三种基本命令:内建命令、Shell函数以及外部命令。
1.内建命令就是由Shell本身所执行的命令。有些命令是由于必要性(需要改变父Shell进程的属性)才内建的。如cd用来改变目录,read将用户输入传给Shell变量。另外一些是为了效率,如test命令,还有I/O命令echo与printf。
2.Shell函数用Shell语言写成的,可以像命令那样引用的代码。
3.外部命令就是Shell子进程所执行的命令,基本过程如下:a.新建一个父Shell进程的子进程。(fork)b.在新进程里搜索PATH变量列出的目录,找到特定命令。当命令名称含有斜杠/时,略过路径查找步骤。c.在新进程里,用找到的程序取代执行中的Shell程序并执行。(exec)d.程序完成后,父Shell进行继续从终端读取下一条命令,或执行脚本里的下一条命令。 2.5.2 变量
Shell变量名称的开头是一个字符或下划线,后面可以接着任意长度的字母、数字或下划线。变量名称和变量保存的字符串的长度都没有限制。在变量名称前加$字符取出Shell变量的值。当给变量赋予的值包含空格时,需要加上引号。
$ myvar=this_is_a_long_string$ echo $myvar
$ first=issac middle=bashevis last=singer 单行可进行多次赋值$ fullname="issac bashevis singer"$ fullname="$first $middle $last"$ oldname=$fullname 2.5.3 简单的echo输出
$ echo Now is the time for all good menNow is the time for all good men
echo有版本上的差异,但只要是使用最简单的形式,其echo的可移植性不会有问题。
BSD版本的echo,-n参数会省略结尾的换行符号。$ echo -n "Enter your name: "Enter your name: _
System V的echo会处理转义字符,\c表示忽略换行符号。$ echo "Enter your name: \c" 2.5.4 华丽的printf输出
printf命令模仿C程序库里的printf()程序,几乎复制了该函数的所有功能。$ printf "Hello, world\n"$ printf "The first program always prints '%s, %s'!\n" Hello world最大不同在于:printf不像echo那样自动提供一个换行符,必须显式指定\n。 2.5.5 基本的I/O重定向
标准输入/输出可能是软件设计原则里最重要的概念了。程序不必知道也不用关心它的输入与输出背后是什么设备:磁盘上的文件、终端、磁带机、网络连接或是另一个执行中的程序!当程序启动时,可以预期的是,标准输入输出都已打开,且已准备好供其使用。
许多Unix程序都遵循这一设计原则。默认情况下,它们会读取标准输入、写入标准输出,并将错误信息传递到标准错误输出。这类程序常叫做过滤器(filter)。默认的标准输入、标准输出以及标准错误输出都是终端。
$ cat 未指定任何参数,读取标准输入,写入标准输出now is the time 由用户输入 now is the time 由cat返回for all good menfor all good men^D
在你登录时,Unix便将默认的标准输入、输出及错误输出安排成你的终端。 重定向与管道
以<改变标准输入:tr -d '\r' < dos-file.txt (命令tr -d '\r'将删除回车)以>改变标准输出,覆盖目的文件:tr -d '\r' < dos-file.txt > UNIX-file.txt以>>附加到文件,附加到文件结尾:for f in dos-file*.txtdo tr -d '\r' >> big-UNIX-file.txtdone
以|建立管道:tr -d '\r' < dos-file.txt | sort > UNIX-file.txt将program1的标准输出修改为program2的标准输入。虽然<与>可将输入与输出连接到文件,不过管道的执行速度比使用临时文件的程序快上十倍。
构造管道时,应该试着让每个阶段的数据量变少。换句话说,如果你有两个要完成的步骤与先后次序无关,你可以把让数据量变少的那一步放在管道的前端,这样可以提升脚本的性能。例如,使用sort排序之前,先以grep找出相关的行,这样可以让sort少做些事。 特殊文件:/dev/null与/dev/tty
/dev/null就是大家所熟知的位桶(bit bucket),写到此文件的数据都会被系统丢掉。读取它则会立即返回文件结束符号(end-of-file)。
当程序打开/dev/tty时,Unix会自动将它重定向到一个终端,这在程序必须读取人工输入时特别有用。以下stty命令用来控制终端的各种设置。$ printf "Enter new password: " 提示输入$ stty -echo 关闭自动打印输入字符的功能$ read pass < /dev/tty 读取密码$ printf "\nEnter again: " 提示再输入一次$ read pass2 < /dev/tty 再读取一次以确认$ stty echo 恢复自动打印输入字符的功能$ echo$ echo pass is $pass, pass2 is $pass2 2.5.6 基本命令查找
如果要编写自己的脚本,最好准备自己的bin目录来存放它们,并且让Shell能自动找到它们。只要建立自己的bin目录,并将它加入$PATH中的列表即可。$ PATH=$PATH:$HOME/bin
要让修改永久生效,在.profile文件中把你的bin目录加入$PATH,每次登录时Shell都会读取.profile文件。
空项目或点号表示当前目录,如PATH=:/bin:/user/bin或PATH=/bin::/user/bin
2.6 访问Shell脚本的参数
Shell脚本的命令行参数,同时也是函数的参数。各参数都由整数来命名,超过9就应该用大括号。echo first arg is $1echo tenth arg is {$10}
查找某用户现在是否登录。$ cat > finduser#! /bin/sh# finduser -- find if user in argument one is login nowwho | grep $1^D$ chmod +x finduser$ ./finduser betsy