《Shell脚本学习指南》第二章学习笔记


2.1 脚本编程语言与编译型语言的差异

编译型语言从源代码转换成目标代码,便能直接通过计算机执行。好处是高效,但运作于底层。
例如,在C++里,很难进行“将一个目录里所有的文件复制到另一个目录中”之类的简单操作。

脚本编程语言通常是解释型(Interpreted)的。由解释器读入程序代码,并将其转换成内部的形式,再执行。
解释器本身是一般的编译型程序。


2.2 为什么要使用Shell脚本

脚本执行效率通常不如编译型语言,但它已经足够快了,足以忽略它性能上的问题。
花一个小时写成的简单脚本,同样的功能用C或C++来编写实现,可能需要两天。
脚本编程语言的例子有awk、Perl、Python、Ruby与Shell。

因为Shell是各Unix系统之间通用的功能,并且经过了POSIX的标准化。只要用心写一次,
即可应用到很多系统中。


2.3 一个简单的脚本

自动化统计当前有多少用户登录系统。其中用到了who命令,wc -l统计行数以及管道。

$ cat > nusers 建立文件,使用cat复制终端的输入
who | wc -l
^D Ctrl-D表示end-of-file
$ chmod +x nusers 让文件拥有执行的权限
$ ./nusers 执行测试
6

这展现了小型Shell脚本的典型开发周期:
1. 直接在命令行上测试,找到能够完成工作的适当语法。
2. 将它们放进一个独立的脚本里,并为该脚本设置执行的权限。
3. 之后就能直接使用该脚本了。


2.4 自给自足的脚本:位于第一行的#!

当一个文件中开头的两个字符是#!时,内核会扫描该行其余的部分,看是否存在
可用来执行程序的解释器的完整路径。此外还会扫描是否有一个选项要传给解释器。

#! /bin/csh -f

下面是几个初级的陷阱:
1. 各系统对#!这一行的长度限制从64到1024字符都有,所以尽量不要超过64个字符。
2. 别在选项之后放置任何空白,因为空白也会跟着选项一起传递给被引用的程序。


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 men
Now 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 men
for 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*.txt
do
tr -d '\r' >> big-UNIX-file.txt
done

以|建立管道: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 $1
echo tenth arg is {$10}

查找某用户现在是否登录。
$ cat > finduser
#! /bin/sh
# finduser -- find if user in argument one is login now
who | grep $1
^D
$ chmod +x finduser
$ ./finduser betsy


2.7 简单的执行跟踪

打开执行跟踪功能会使得Shell显示每个被执行到的命令,并在前面加上“+”和一个空格。
$ sh -x nusers
+ who
+ wc -l

也可以在脚本里,用set -x命令打开执行跟踪,然后再用set +x命令关闭。
$ cat > trace1.sh
#! /bin/sh
set -x
echo 1st echo
set +x
echo 2nd echo
^D
$ chmod +x trace1.sh
$ ./trace1.sh
+ echo 1st echo
1st echo
+ set +x
2nd echo


你可能感兴趣的:(《Shell脚本学习指南》第二章学习笔记)