linux高手的十个命令行习惯 献给新手们

分类:CentOS教程 阅读:60402 次

本文提出了 10个值得采用的 UNIX命令行习惯——帮助您克服许多常见使用怪癖,并在该过程中提高命令行工作效率的好习惯。下面列出了这 10个好习惯,之后对进行了更详细的描述。

采用 10个好习惯

要采用的十个好习惯为:

在单个命令中创建目录树。
更改路径;不要移动存档。
将命令与控制操作符组合使用。
谨慎引用变量。
使用转义序列来管理较长的输入。
在列表中对命令分组。
在 find之外使用 xargs。
了解何时 grep应该执行计数——何时应该绕过。
匹配输出中的某些字段,而不只是对行进行匹配。
停止对 cat使用管道。

在单个命令中创建目录树

使用 mkdir的 -p选项并在单个命令中创建所有父目录及其子目录要容易得多。但是即使对于知道此选项的管理员,他们在命令行上创建子目录时也仍然束缚于逐步创建每级子目录。花时间有意识地养成这个好习惯是值得的:


清单 2.好习惯 1的示例:使用一个命令来定义目录树
~ $ mkdir -p tmp/a/b/c

清单 3.好习惯 1的另一个示例:使用一个命令来定义复杂的目录树
~ $ mkdir -pproject/{lib/ext,bin,src,doc/{html,info,pdf},demo/stat/a}

清单 4.好习惯2的示例:使用选项 -C来解压缩 .tar存档文件
~ $ tar xvf-Ctmp/a/b/c newarc.tar.gz

相对于将存档文件移动到您希望在其中解压缩它的位置,切换到该目录,然后才解压缩它,养成使用 -C的习惯则更加可取——当存档文件位于其他某个位置时尤其如此。



将命令与控制操作符组合使用

您可能已经知道,在大多数 Shell中,您可以在单个命令行上通过在命令之间放置一个分号 (;)来组合命令。该分号是 Shell控制操作符,

使用&&控制操作符来组合两个命令,以便仅当 第一个命令返回零退出状态时才运行第二个命令。换句话说,如果第一个命令运行成功,则第二个命令将运行。如果第一个命令失败,则第二个命令根本就不运行。例如:


清单 5.好习惯 3的示例:将命令与控制操作符组合使用
~ $ cd tmp/a/b/c && tar xvf ~/archive.tar

在此例中,存档的内容将提取到 ~/tmp/a/b/c目录中,除非该目录不存在。如果该目录不存在,则 tar命令不会运行,因此不会提取任何内容。

仅当另一个命令返回非零退出状态时才运行某个命令

类似地,||控制操作符分隔两个命令,并且仅当第一个命令返回非零退出状态时才运行第二个命令。换句话说,如果第一个命令成功,则第二个命令不会运行。如果第一个命令失败,则第二个命令才会运行。在测试某个给定目录是否存在时,通常使用此操作符,如果该目录不存在,则创建它:


清单 6.好习惯 3的另一个示例:将命令与控制操作符组合使用
~ $ cd tmp/a/b/c || mkdir -p tmp/a/b/c

始终要谨慎使用 Shell扩展和变量名称。一般最好将变量调用包括在双引号中,除非您有不这样做的足够理由。类似地,如果您直接在字母数字文本后面使用变量名称,则还要确保将该变量名称包括在方括号 ([])中,以使其与周围的文本区分开来。否则,Shell将把尾随文本解释为变量名称的一部分——并且很可能返回一个空值。清单 8提供了变量的各种引用和非引用及其影响的示例。


清单 8.好习惯 4的示例:引用(和非引用)变量
~ $ ls tmp/
a b
~ $ VAR="tmp/*"
~ $ echo $VAR
tmp/a tmp/b
~ $ echo "$VAR"
tmp/*
~ $ echo $VARa
~ $ echo "$VARa"
~ $ echo "${VAR}a"
tmp/*a
~ $ echo ${VAR}a
tmp/a
~ $



使用转义序列来管理较长的输入

您或许看到过使用反斜杠 (\)来将较长的行延续到下一行的代码示例,并且您知道大多数 Shell都将您通过反斜杠联接的后续行上键入的内容视为单个长行。然而,您可能没有在命令行中像通常那样利用此功能。如果您的终端无法正确处理多行回绕,或者您的命令行比通常小(例如在提示符下有长路经的时候),反斜杠就特别有用。反斜杠对于了解键入的长输入行的含义也非常有用,如以下示例所示:


清单 9.好习惯 5的示例:将反斜杠用于长输入
~ $ cd tmp/a/b/c || \
> mkdir -p tmp/a/b/c && \
> tar xvf -C tmp/a/b/c ~/archive.tar

或者,也可以使用以下配置:


清单 10.好习惯 5的替代示例:将反斜杠用于长输入
~ $ cd tmp/a/b/c \
>|| \
> mkdir -p tmp/a/b/c \
>&& \
> tar xvf -C tmp/a/b/c ~/archive.tar



然而,当您将输入行划分到多行上时,Shell始终将其视为单个连续的行,因为它总是删除所有反斜杠和额外的空格。

注意:在大多数 Shell中,当您按向上箭头键时,整个多行输入将重绘到单个长输入行上。

在列表中对命令分组

大多数 Shell都具有在列表中对命令分组的方法,以便您能将它们的合计输出向下传递到某个管道,或者将其任何部分或全部流重定向到相同的地方。您一般可以通过在某个 Subshell中运行一个命令列表或通过在当前 Shell中运行一个命令列表来实现此目的。

在 Subshell中运行命令列表

使用括号将命令列表包括在单个组中。这样做将在一个新的 Subshell中运行命令,并允许您重定向或收集整组命令的输出,如以下示例所示:


清单 11.好习惯 6的示例:在 Subshell中运行命令列表
~ $ ( cd tmp/a/b/c/ || mkdir -p tmp/a/b/c && \
> VAR=$PWD; cd ~; tar xvf -C $VAR archive.tar ) \
> | mailx admin -S "Archive contents"


在此示例中,该存档的内容将提取到 tmp/a/b/c/目录中,同时将分组命令的输出(包括所提取文件的列表)通过邮件发送到地址 admin。

在当前 Shell中运行命令列表

将命令列表用大括号 ({})括起来,以在当前 Shell中运行。确保在括号与实际命令之间包括空格,否则 Shell可能无法正确解释括号。

清单 12.好习惯 6的另一个示例:在当前 Shell中运行命令列表
~ $ { cp ${VAR}a . && chown -R guest.guest a && \
> tar cvf newarchive.tar a; } | mailx admin -S "Newarchive"


在 find之外使用 xargs

使用 xargs工具作为筛选器,以充分利用从 find命令挑选的输出。find运行通常提供与某些条件匹配的文件列表。此列表被传递到 xargs上,后者然后使用该文件列表作为参数来运行其他某些有用的命令,如以下示例所示:


清单 13. xargs工具的经典用法示例
~ $ find some-file-criteria some-file-path | \
> xargs some-great-command-that-needs-filename-arguments

在最简单的调用形式中,xargs就像一个筛选器,它接受一个列表(每个成员分别在单独的行上)作为输入。该工具将那些成员放置在单个空格分隔的行上:


清单 14. xargs工具产生的输出示例
~ $ xargs
a
b
c
Control-D
a b c
~ $

您可以发送通过 xargs来输出文件名的任何工具的输出,以便为其他某些接受文件名作为参数的工具获得参数列表,如以下示例所示:

清单 15. xargs工具的使用示例
~/tmp $ ls -1 | xargs
December_Report.pdf README a archive.tar mkdirhier.sh
~/tmp $ ls -1 | xargs file
December_Report.pdf: PDF document, version 1.3
README: ASCII text
a: directory
archive.tar: POSIX tar archive
mkdirhier.sh: Bourne shell script text executable
~/tmp $

xargs命令不只用于传递文件名。您还可以在需要将文本筛选到单个行中的任何时候使用它:

清单 16.好习惯 7的示例:使用 xargs工具来将文本筛选到单个行中
~/tmp $ ls -l | xargs
-rw-r--r-- 7 joe joe 12043 Jan 27 20:36 December_Report.pdf-rw-r--r-- 1 \
root root 238 Dec 03 08:19 README drwxr-xr-x 38 joe joe 354082 Nov02 \
16:07 a -rw-r--r-- 3 joe joe 5096 Dec 14 14:26 archive.tar-rwxr-xr-x 1 \
joe joe 3239 Sep 30 12:40 mkdirhier.sh
~/tmp $

避免通过管道将 grep发送到 wc -l来对输出行数计数。grep的 -c选项提供了对与特定模式匹配的行的计数,并且一般要比通过管道发送到 wc更快,如以下示例所示:


清单 17.好习惯 8的示例:使用和不使用 grep的行计数
~ $ time grep and tmp/a/longfile.txt | wc -l
2811
real 0m0.097s
user 0m0.006s
sys0m0.032s
~ $ time grep -c and tmp/a/longfile.txt
2811
real 0m0.013s
user 0m0.006s
sys0m0.005s
~ $


除了速度因素外,-c选项还是执行计数的好方法。对于多个文件,带 -c选项的 grep返回每个文件的单独计数,每行一个计数,而针对 wc的管道则提供所有文件的组合总计数。

然而,不管是否考虑速度,此示例都表明了另一个要避免地常见错误。这些计数方法仅提供包含匹配模式的行数——如果那就是您要查找的结果,这没什么问题。但是在行中具有某个特定模式的多个实例的情况下,这些方法无法为您提供实际匹配实例数量的真实计数。归根结底,若要对实例计数,您还是要使用 wc来计数。首先,使用-o选项(如果您的版本支持它的话)来运行 grep命令。此选项仅输出匹配的模式,每行一个模式,而不输出行本身。但是您不能将它与 -c选项结合使用,因此要使用 wc -l来对行计数,如以下示例所示:


清单 18.好习惯 8的示例:使用 grep对模式实例计数
~ $ grep -o and tmp/a/longfile.txt | wc -l
3402
~ $
在此示例中,grep对行进行筛选,并输出其修改日期和名称中带 Dec的所有文件。因此,诸如 December_Report.pdf等文件是匹配的,即使它自从一月份以来还未修改过。这可能不是您希望的结果。为了匹配特定字段中的模式,最好使用 awk,其中的一个关系运算符对确切的字段进行匹配,如以下示例所示:

清单 20.好习惯 9的示例:使用 awk来查找特定字段中的模式
~/tmp $ ls -l | awk '$6 == "Dec"'
-rw-r--r--3 joe joe 5096 Dec 14 14:26archive.tar
-rw-r--r--1 root root238 Dec 03 08:19README
~/tmp $
grep的一个常见的基本用法错误是通过管道将 cat的输出发送到 grep以搜索单个文件的内容。这绝对是不必要的,纯粹是浪费时间,因为诸如 grep这样的工具接受文件名作为参数。您根本不需要在这种情况下使用 cat,如以下示例所示:


清单 21.好习惯和坏习惯 10的示例:使用带和不带 cat的grep
~ $ time cat tmp/a/longfile.txt | grep and
2811
real 0m0.015s
user 0m0.003s
sys0m0.013s
~ $ time grep and tmp/a/longfile.txt
2811
real 0m0.010s
user 0m0.006s
sys0m0.004s
~ $