UNIX 高手的另外10个习惯

=atitle>引言

当 您经常使用某个系统时往往会陷入某种固定使用模式有时您没有养成以尽可能最好方式做事习惯有时不良习惯甚至会导致出现混乱纠正此类 缺点最佳思路方法的就是有意识地采用抵制这些坏习惯好习惯本文提出了 10 个值得采用 UNIX 命令行习惯——帮助您克服许多常见使用怪癖并在该过程中提高命令行工作效率好习惯下面列出了这 10 个好习惯的后对进行了更详细描述

=atitle>采用 10 个好习惯

要采用十个好习惯为:

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

    清单 1 演示了最常见 UNIX 坏习惯的:次定义个目录树

    清单 1. 坏习惯 1 举例:单独定义每个目录树

    ~ $ =boldcode>mkdir tmp
    ~ $ =boldcode>cd tmp
    ~/tmp $ =boldcode>mkdir a
    ~/tmp $ =boldcode>cd a
    ~/tmp/a $ =boldcode>mkdir b
    ~/tmp/a $ =boldcode>cd b
    ~/tmp/a/b/ $ =boldcode>mkdir c
    ~/tmp/a/b/ $ =boldcode>cd c
    ~/tmp/a/b/c $




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


    清单 2. 好习惯 1 举例:使用个命令来定义目录树

    ~ $ =boldcode>mkdir -p tmp/a/b/c




    您可以使用此选项来创建整个复杂目录树(在脚本中使用是非常理想)而不只是创建简单层次结构例如:


    清单 3. 好习惯 1 个举例:使用个命令来定义复杂目录树

    ~ $ =boldcode>mkdir -p project/{lib/ext,bin,src,doc/{html,info,pdf},demo/stat/a}




    过去单独定义目录借口是您 mkdir 实现不支持此选项但是在大多数系统上不再是这样了IBM、AIX®、mkdir、GNU mkdir 和其他遵守单 UNIX 规范标准 (Single UNIX Specication) 系统现在都具有此选项

    对于仍然缺乏该功能少数系统您可以使用 mkdirhier 脚本(请参见参考资料)此脚本是执行相同功能 mkdir 包装:

    ~ $ =boldcode>mkdirhier project/{lib/ext,bin,src,doc/{html,info,pdf},demo/stat/a}



    =atitle>2. 更改路径;不要移动存档

    个不良使用模式是将 .tar 存档文件移动到某个目录该目录恰好是您希望在其中提取 .tar 文件目录其实您根本不需要这样做您可以随心所欲地将任何 .tar 存档文件解压缩到任何目录——这就是 -C-C 选项来指定要在其中解压缩该文件目录: 选项用途在解压缩某个存档文件时使用

    清单 4. 好习惯 2 举例:使用选项 -C 来解压缩 .tar 存档文件

    ~ $ =boldcode>tar xvf -C tmp/a/b/c arc.tar.gz




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


    =atitle>3. 将命令和控制操作符组合使用 您可能已经知道在大多数 Shell 中您可以在单个命令行上通过在命令的间放置个分号 (;) 来组合命令该分号是 Shell 控制操作符 虽然它对于在单个命令行上将离散命令串联起来很有用但它并不适用于所有情况例如假设您使用分号来组合两个命令其中第 2个命令正确执行完全依赖 于第个命令成功完成如果第个命令未按您预期那样退出第 2个命令仍然会运行——结果会导致失败相反应该使用更适当控制操作符(本文将描述 其中部分操作符)只要您 Shell 支持它们就值得养成使用它们习惯

    =smalltitle>仅当另个命令返回零退出状态时才运行某个命令

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

    清单 5. 好习惯 3 举例:将命令和控制操作符组合使用

    ~ $ =boldcode>cd tmp/a/b/c && tar xvf ~/archive.tar




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

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

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

    清单 6. 好习惯 3 个举例:将命令和控制操作符组合使用

    ~ $ =boldcode>cd tmp/a/b/c || mkdir -p tmp/a/b/c




    您还可以组合使用本部分中描述控制操作符每个操作符都影响最后命令运行:

    清单 7. 好习惯 3 组合举例:将命令和控制操作符组合使用

    ~ $ =boldcode>cd tmp/a/b/c || mkdir -p tmp/a/b/c && tar xvf -C tmp/a/b/c ~/archive.tar





    =atitle>4. 谨慎引用变量

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

    清单 8. 好习惯 4 举例:引用(和非引用)变量

    ~ $ =boldcode>ls tmp/
    a b
    ~ $ =boldcode>VAR="tmp/*"
    ~ $ =boldcode>echo $VAR
    tmp/a tmp/b
    ~ $ =boldcode>echo "$VAR"
    tmp/*
    ~ $ =boldcode>echo $VARa

    ~ $ =boldcode>echo "$VARa"

    ~ $ =boldcode>echo "${VAR}a"
    tmp/*a
    ~ $ =boldcode>echo ${VAR}a
    tmp/a
    ~ $





    =atitle>5. 使用转义序列来管理较长输入

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

    清单 9. 好习惯 5 举例:将反斜杠用于长输入

    ~ $ =boldcode>cd tmp/a/b/c || \
    > =boldcode>mkdir -p tmp/a/b/c && \
    > =boldcode>tar xvf -C tmp/a/b/c ~/archive.tar




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

    清单 10. 好习惯 5 替代举例:将反斜杠用于长输入

    ~ $ =boldcode>cd tmp/a/b/c \
    > =boldcode> || \
    > =boldcode>mkdir -p tmp/a/b/c \
    > =boldcode> && \
    > =boldcode>tar xvf -C tmp/a/b/c ~/archive.tar




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

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


    =atitle>6. 在列表中对命令分组

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

    =smalltitle>在 Subshell 中运行命令列表

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

    清单 11. 好习惯 6 举例:在 Subshell 中运行命令列表

    ~ $ =boldcode>( cd tmp/a/b/c/ || mkdir -p tmp/a/b/c && \
    > =boldcode>VAR=$PWD; cd ~; tar xvf -C $VAR archive.tar ) \
    > =boldcode>| mailx admin -S "Archive contents"




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

    当您在命令列表中重新定义环境变量并且您不希望将那些定义应用于当前 Shell 时使用 Subshell 更可取

    =smalltitle>在当前 Shell 中运行命令列表

    将命令列表用大括号 ({}) 括起来以在当前 Shell 中运行确保在括号和实际命令的间包括空格否则 Shell 可能无法正确解释括号此外还要确保列表中最后个命令以分号结尾如以下举例所示:


    清单 12. 好习惯 6 个举例:在当前 Shell 中运行命令列表

    ~ $ =boldcode>{ cp ${VAR}a . && chown -R guest.guest a && \
    > =boldcode>tar cvf archive.tar a; } | mailx admin -S "New archive"




    =atitle>7. 在 find 的外使用 xargs

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


    清单 13. xargs 工具经典使用方法举例

    ~ $ =boldcode>find some-file-criteria some-file-path | \
    > =boldcode>xargs some-great-command-that-needs-filename-arguments




    然而不要将 xargs 仅看作是 find 辅助工具;它是个未得到充分利用工具的当您养成使用它习惯时将会希望进行所有试验包括以下使用方法

    =smalltitle>传递空格分隔列表

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

    清单 14. xargs 工具产生输出举例

    ~ $ =boldcode>xargs
    =boldcode>a
    =boldcode>b
    =boldcode>c
    =boldcode>
    Control-D
    a b c
    ~ $




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


    清单 15. xargs 工具使用举例

    ~/tmp $ =boldcode>ls -1 | xargs
    December_Report.pdf README a archive.tar mkdirhier.sh
    ~/tmp $ =boldcode>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 $ =boldcode>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 Nov 02 \
    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 $




    =smalltitle>谨慎使用 xargs

    从技术上讲使用 xargs 很少遇到麻烦缺省情况下文件结束串是下划线 (_);如果将该作为单个输入参数来发送则它的后所有内容将被忽略为了防止这种情况发生可以使用 -e 标志它在不带参数情况下完全禁用结束


    =atitle>8. 了解何时 grep 应该执行计数——何时应该绕过

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


    清单 17. 好习惯 8 举例:使用和不使用 grep 行计数

    ~ $ =boldcode>time grep and tmp/a/longfile.txt | wc -l
    2811

    real 0m0.097s
    user 0m0.006s
    sys 0m0.032s
    ~ $ =boldcode>time grep -c and tmp/a/longfile.txt
    2811

    real 0m0.013s
    user 0m0.006s
    sys 0m0.005s
    ~ $




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

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


    清单 18. 好习惯 8 举例:使用 grep 对模式例子计数

    ~ $ =boldcode>grep -o and tmp/a/longfile.txt | wc -l
    3402
    ~ $




    在此例中 wc 要比第 2次 grep 并插入个虚拟模式(例如 grep -c)来对行进行匹配和计数稍快


    =atitle>9. 匹配输出中某些字段而不只是对行进行匹配

    当您只希望匹配输出行中特定字段模式时诸如 awk 等工具要优于 grep

    下面经过简化举例演示了如何仅列出 12 月修改过文件


    清单 19. 坏习惯 9 举例:使用 grep 来查找特定字段中模式

    ~/tmp $ =boldcode>ls -l /tmp/a/b/c | grep Dec
    -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
    -rw-r--r-- 3 joe joe 5096 Dec 14 14:26 archive.tar
    ~/tmp $




    在此举例中grep 对行进行筛选并输出其修改日期和名称中带 Dec 所有文件因此诸如 December_Report.pdf 等文件是匹配即使它自从月份以来还未修改过这可能不是您希望结果为了匹配特定字段中模式最好使用 awk其中个关系运算符对确切字段进行匹配如以下举例所示:


    清单 20. 好习惯 9 举例:使用 awk 来查找特定字段中模式

    ~/tmp $ =boldcode>ls -l | awk '$6 "Dec"'
    -rw-r--r-- 3 joe joe 5096 Dec 14 14:26 archive.tar
    -rw-r--r-- 1 root root 238 Dec 03 08:19 README
    ~/tmp $




    有关如何使用 awk 更多详细信息请参见参考资料


    =atitle>10. 停止对 cat 使用管道

    grep 个常见基本使用方法是通过管道将 cat 输出发送到 grep 以搜索单个文件内容这绝对是不必要纯粹是浪费时间诸如 grep 这样工具接受文件名作为参数您根本不需要在这种情况下使用 cat如以下举例所示:


    清单 21. 好习惯和坏习惯 10 举例:使用带和不带 cat grep

    ~ $ =boldcode>time cat tmp/a/longfile.txt | grep and
    2811

    real 0m0.015s
    user 0m0.003s
    sys 0m0.013s
    ~ $ =boldcode>time grep and tmp/a/longfile.txt
    2811

    real 0m0.010s
    user 0m0.006s
    sys 0m0.004s
    ~ $




    存在于许多工具中由于大多数工具都接受使用连 (-) 标准输入作为个参数因此即使使用 cat 来分散 stdin多个文件参数也通常是无效仅当您使用带多个筛选选项的 cat才真正有必要在管道前首先执行连接


    =atitle>结束语:养成好习惯

    最好检查下您命令行习惯中任何不良使用模式不良使用模式会降低您速度并且通常会导致意外本文介绍了 10 个新习惯它们可以帮助您摆脱许多最常见使用养成这些好习惯是加强您 UNIX 命令行技能积极步骤

    from:http://www.ibm.com/developerworks/cn/aix/library/au-badunixhabits.html
    Tags: 

    延伸阅读

最新评论

发表评论