标准的 UNIX 命令使得 shell 脚本更加灵活.通过简单的编程结构把shell指令和系统命令结合起来,这才是脚本能力的所在.
基本的列出所有文件的命令.但是往往就是因为这个命令太简单,所以我们总是低估它.比如,用 -R 选项,这是递归选项,ls 将会以目录树的形式列出所有文件, 另一个很有用的选项是 -S ,将会按照文件尺寸列出所有文件, -t, 将会按照修改时间来列出文件,-i 选项会显示文件的inode(见 Example 12-4).
Example 12-1. 使用ls命令来创建一个烧录CDR的内容列表
1 #!/bin/bash 2 # ex40.sh (burn-cd.sh) 3 # 自动刻录CDR的脚本. 4 5 6 SPEED=2 # 如果你的硬件支持的话,你可以选用更高的速度. 7 IMAGEFILE=cdimage.iso 8 CONTENTSFILE=contents 9 DEVICE=cdrom 10 # DEVICE="0,0" 为了使用老版本的CDR 11 DEFAULTDIR=/opt # 这是包含需要被刻录内容的目录. 12 # 必须保证目录存在. 13 # 小练习: 测试一下目录是否存在. 14 15 # Uses Joerg Schilling's "cdrecord" package: 15 # 使用 Joerg Schilling 的 "cdrecord"包: 16 # http://www.fokus.fhg.de/usr/schilling/cdrecord.html 17 18 # 如果一般用户调用这个脚本的话,可能需要root身份 19 #+ chmod u+s /usr/bin/cdrecord 20 # 当然, 这会产生安全漏洞, 虽然这是一个比较小的安全漏洞. 21 22 if [ -z "$1" ] 23 then 24 IMAGE_DIRECTORY=$DEFAULTDIR 25 # 如果命令行没指定的话, 那么这个就是默认目录. 26 else 27 IMAGE_DIRECTORY=$1 28 fi 29 30 # 创建一个内容列表文件. 31 ls -lRF $IMAGE_DIRECTORY > $IMAGE_DIRECTORY/$CONTENTSFILE 32 # "l" 选项将给出一个"长"文件列表. 33 # "R" 选项将使这个列表递归. 34 # "F" 选项将标记出文件类型 (比如: 目录是以 /结尾, 而可执行文件以 *结尾). 35 echo "Creating table of contents." 36 37 # 在烧录到CDR之前创建一个镜像文件. 38 mkisofs -r -o $IMAGEFILE $IMAGE_DIRECTORY 39 echo "Creating ISO9660 file system image ($IMAGEFILE)." 40 41 # 烧录CDR. 42 echo "Burning the disk." 43 echo "Please be patient, this will take a while." 44 cdrecord -v -isosize speed=$SPEED dev=$DEVICE $IMAGEFILE 45 46 exit $? |
cat, 是单词 concatenate的缩写, 把文件的内容输出到stdout. 当与重定向操作符 (> 或 >>)结合使用时, 一般都是用来将多个文件连接起来.
1 # Uses of 'cat' 2 cat filename # 打印出文件内容. 3 4 cat file.1 file.2 file.3 > file.123 # 把3个文件连接到一个文件中. |
见 Example 12-25 和 Example 12-21.
在一个 管道 中, 可能有一种把stdin 重定向 到一个文件中的更有效的办法, 这种方法比 cat文件的方法更有效率.
|
tac 命令, 就是 cat的反转, 将从文件的结尾列出文件.
把每一行中的内容反转, 并且输出到 stdout上. 这个命令与 tac命令的效果是不同的, 因为它并不反转行序, 而是把每行的内容反转.
bash$ cat file1.txt This is line 1. This is line 2. bash$ tac file1.txt This is line 2. This is line 1. bash$ rev file1.txt .1 enil si sihT .2 enil si sihT |
这是文件拷贝命令. cp file1 file2 把 file1 拷贝到 file2, 如果存在 file2 的话,那 file2 将被覆盖 (见 Example 12-6).
特别有用的选项就是 -a 归档 选项 (为了copy一个完整的目录树), -u 是更新选项, 和 -r 与 -R 递归选项.
|
这是文件移动命令. 它等价于 cp 与 rm 命令的组合. 它可以把多个文件移动到目录中,甚至将目录重命名. 想查看 mv 在脚本中使用的例子, 见 Example 9-18 和 Example A-2.
当使用非交互脚本时,可以使用 mv 的-f (强制) 选项来避免用户的输入. 当一个目录被移动到一个已存在的目录时,那么它将成为目标目录的子目录.
|
删除(清除)一个或多个文件. -f 选项将强制删除文件,即使这个文件是只读的.并且可以用来避免用户输入(在非交互脚本中使用).
rm 将无法删除以破折号开头的文件.
解决这个问题的一个方法就是在要删除的文件的前边加上"./".
|
当使用递归参数 -r时, rm 命令将会删除整个目录树. 如果不慎使用 rm -rf *那整个目录树就真的完了. |
删除目录. 但是只有这个目录中没有文件 -- 当然会包含不可见的 点文件 [1] -- 的时候这个命令才会成功.
生成目录, 创建一个空目录. 比如, mkdir -p project/programs/December 将会创建出这个指定的目录, 即使project目录和programs目录都不存在. -p 选项将会自动产生必要的父目录, 这样也就同时创建了多个目录.
修改一个现存文件的属性 (见 Example 11-12).
1 chmod +x filename 2 # 使得文件filename对所有用户都可执行. 3 4 chmod u+s filename 5 # 设置"filename"文件的"suid"位. 6 # 这样一般用户就可以执行"filename", 他将拥有和文件宿主相同的权限. 7 # (这并不适用于shell 脚本) |
1 chmod 644 filename 2 # Makes "filename" readable/writable to owner, readable to 3 # 设置文件宿主的 r/w 权限,并对一般用户 3 # 设置读权限. 4 # (8进制模式). |
1 chmod 1777 directory-name 2 # 对这个目录设置r/w 和可执行权限, 并开放给所有人. 3 # 同时设置 "粘贴位". 4 # 这意味着, 只有目录宿主, 5 # 文件宿主, 当然, 还有root 6 # 可以删除这个目录中的任何特定的文件. |
修改文件属性. 这个命令与上边的 chmod 命令相类似, 但是有不同的选项和不同的调用语法, 并且这个命令只能工作在ext2文件系统中.
chattr 命令的一个特别有趣的选项是i. chattr +i filename 将使得这个文件被标记为永远不变. 这个文件将不能被修改, 连接, 或删除, 即使是root也不行. 这个文件属性只能被root设置和删除. 类似的, a 选项将会把文件标记为只能追加数据.
root# chattr +i file1.txt root# rm file1.txt rm: remove write-protected regular file `file1.txt'? y rm: cannot remove `file1.txt': Operation not permitted |
如果文件设置了s(安全)属性, 那么当这个文件被删除时,这个文件所在磁盘的块将全部被0填充.
如果文件设置了u(不可删除)属性, 那么当这个文件被删除后, 这个文件的内容还可以被恢复(不可删除).
如果文件设置了c(压缩)属性, 那么当这个文件在进行写操作时,它将自动被压缩,并且在读的时候, 自动解压.
使用命令chattr do设置的属性, 将不会显示在文件列表中(ls -l). |
创建文件链接, 前提是这个文件是存在的. "链接" 就是一个文件的引用, 也就是这个文件的另一个名字. ln 命令允许对同一个文件引用多个链接,并且是避免混淆的一个很好的方法 (见 Example 4-6).
ln 对于文件来说只不过是创建了一个引用, 一个指针而已, 因为创建出来的连接文件只有几个字节.
绝大多数使用ln 命令时使用是 -s 选项, 可以称为符号链接, 或软链接.使用 -s 选项的一个优点是它可以穿越文件系统来链接目录.
关于使用这个命令的语法还是有点小技巧的. 比如: ln -s oldfile newfile 将对老文件产生一个新的文件链接.
如果之前就存在newfile的话, 那么将会产生一个错误消息. |
链接给出了一种可以用多个名字来调用脚本的能力(当然这也适用于任何可执行的类型), 并且脚本的行为将依赖于脚本是如何被调用的.
Example 12-2. Hello or Good-bye
1 #!/bin/bash 2 # hello.sh: 显示"hello" 还是 "goodbye" 3 #+ 依赖于脚本是如何被调用的. 4 5 # 在当前目录下($PWD)为这个脚本创建一个链接: 6 # ln -s hello.sh goodbye 7 # 现在, 通过如下两种方法来调用这个脚本: 8 # ./hello.sh 9 # ./goodbye 10 11 12 HELLO_CALL=65 13 GOODBYE_CALL=66 14 15 if [ $0 = "./goodbye" ] 16 then 17 echo "Good-bye!" 18 # 当然, 在这里你也可以添加一些其他的 goodbye类型的命令.Some other goodbye-type commands, as appropriate. 19 exit $GOODBYE_CALL 20 fi 21 22 echo "Hello!" 23 # 当然, 在这里你也可以添加一些其他的 hello类型的命令. 24 exit $HELLO_CALL |
These 这两个命令用来查看系统命令或安装工具的手册和信息.当两者都可用时, info 页一般比 man也会包含更多的细节描述.
[1] |
Dotfiles 就是文件名以"."开头的文件, 比如 ~/.Xdefaults. 这样的文件在一般的 ls 命令使用中将不会被显示出来 (当然 ls -a 将会显示它们), 并且它们也不会被一个意外的 rm -rf *删除. 在用户的home目录中,Dotfiles 一般被用来当作安装和配置文件. |