UNIX 高手的另外 10 个习惯

让咱们面对现实吧:坏习惯很难改变。可是您已经熟悉的习惯可能更难克服。有时,从新审视某些事情可能让您遇到“啊哈,我没想到它能作到这一点!”的时刻。在 Michael Stutz 的优秀文章“UNIX 高手的 10 个习惯”的基础上,本文将提供另外 10 个 UNIX 命令行命令、工具和技术,可使您成为更高效的 UNIX 命令行高手。html

您应当采纳的其余 10 个好习惯包括:正则表达式

  • 使用文件名自动完成功能 (file name completion)。
  • 使用历史扩展。
  • 重用之前的参数。
  • 使用 pushd 和 popd 管理目录导航。
  • 查找大型文件。
  • 不使用编辑器建立临时文件。
  • 使用 curl 命令行实用工具。
  • 最有效地利用正则表达式。
  • 肯定当前用户。
  • 使用 awk 处理数据。

使用文件名完成

若是不须要在命令提示符处键入长的、使人费解的文件名,这是否是很棒呢?的确,您不须要这样作。相反,您能够配置最流行的 UNIX Shell 以使用文件名完成。该功能在各个 Shell 中的工做方式略有不一样,所以我将向您展现如何在最流行的 Shell 中使用文件名完成。文件名完成使您能够更快地输入并避免错误。懒惰?也许吧。效率更高?固然!express

我正在运行哪一种 Shell?

若是您不知道目前使用的是哪种 Shell,会怎么样?虽然这个诀窍不是另外 10 个好习惯的正式组成部分,但它仍然颇有用。如清单 1 所示,您可使用 echo $0 或 ps -p $$ 命令显示您正在使用的 Shell。对于我来讲,运行的是 Bash Shell。bash

清单 1. 肯定您的 Shell
1
2
3
4
5
$ echo $0
-bash
$ ps –p $$
PID TTY           TIME CMD
6344 ttys000    0:00.02 –bash

C Shell

C Shell 支持最直接文件名完成功能。设置 filec 变量可启用该功能。(您可使用命令 set filec。)在您开始键入文件名后,能够按 Esc 键,Shell 将完成文件名,或完成尽量多的部分。例如,假设您拥有名为 file一、file2 和 file3 的文件。若是您键入 f,而后按 Esc 键,将填充 file,而您必须键入 12 或 3 来完成相应的文件名。服务器

Bash

Bash Shell 也提供了文件名完成,但使用 Tab 键代替 Esc 键。您在 Bash Shell 中不须要设置任何选项便可启用文件名完成,该选项是缺省设置的。Bash 还实现了其余功能。键入文件名的一部分后,按 Tab 键,若是有多个文件知足您的请求,而且您须要添加文本以选择其中一个文件,那么您能够多按 Tab 键两次,以显示与您目前键入的内容相匹配的文件的列表。使用以前名为 file一、file2 和 file3 的文件示例,首先键入 f。当您按一次 Tab 键时,Bash 完成 file;再按一次 Tab 键时,将展开列表 file1 file2 file3app

Korn Shell

对于 Korn Shell 用户,文件名完成取决于 EDITOR 变量的值。若是 EDITOR 设置为 vi,那么您键入部分名称,而后按 Esc 键,后跟反斜杠 (\) 字符。若是 EDITOR 设置为 emacs,那么您键入部分名称,而后按两次 Esc 键以完成文件名。curl

使用历史扩展

若是您为一系列命令使用相同的文件名,会发生什么状况?固然,有一种快捷方式能够快速得到您上次使用的文件名。如清单 2 所示,!$ 命令返回前一个命令使用的文件名。从文件 this-is-a-long-lunch-menu-file.txt 中搜索单词 pickles 的出现位置。搜索结束后,使用 vi 命令来编辑 this-is-a-long-lunch-menu-file.txt 文件,而不须要从新键入文件名。您使用感叹号 (!) 来访问历史,而后使用美圆符号 ($) 返回前一命令的最后字段。若是您反复用到长文件名,那么这是一个很是好的工具。编辑器

清单 2. 使用 !$ 得到前一个命令使用的文件名
1
2
3
$ grep pickles this-is-a-long-lunch-menu-file.txt
pastrami on rye with pickles and onions
$ vi !$

重用之前的参数

!$ 命令返回某个命令使用的上一个文件名参数。但若是某个命令使用多个文件名,而您只但愿重用其中一个文件名,该如何作?!:1 操做符返回某个命令使用的第一个文件名。清单 3 中的示例显示能够如何将此操做符与 !$ 运算符组合使用。在第一个命令中,将一个文件从新命名为更有意义的名称,但为了保持原始文件名可用,建立了一个符号连接。从新命名文件 kxp12.c 以提升可读性,而后使用 link 命令来建立到原始文件名的符号连接,以防在其余位置使用该文件名。!$ 操做符返回 file_system_access.c 文件名,而 !:1 操做符返回 kxp12.c 文件名,该文件名是上个命令的第一个文件名。ide

清单 3. 组合使用 !$ 和 !:1
1
2
$ mv kxp12.c file_system_access.c
$ ln –s !$ !:1

使用 pushd 和 popd 管理目录导航

UNIX 支持各类目录导航工具。我最喜欢的两款提升工做效率的工具是 pushd 和 popd。您固然了解 cd 命令用于更改您的当前目录。若是您要在多个目录中导航,但但愿可以快速返回某个位置,该如何作?pushd 和 popd 命令建立一个虚拟目录堆栈,pushd 命令用来更改您的当前目录并将其存储在堆栈中,而 popd 命令用来从堆栈的顶部移除目录并使您返回该位置。您可使用 dirs 命令来显示当前目录堆栈,而不会压入或弹出新目录。清单 4 显示如何使用 pushd 和 popd 命令在目录树中快速导航。函数

清单 4. 使用 pushd 和 popd 在目录树中导航
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
$ pushd .
~ ~
$ pushd /etc
/etc ~ ~
$ pushd /var
/var /etc ~ ~
$ pushd /usr/local/bin
/usr/local/bin /var /etc ~ ~
$ dirs
/usr/local/bin /var /etc ~ ~
$ popd
/var /etc ~ ~
$ popd
/etc ~ ~
$ popd
~ ~
$ popd

pushd 和 popd 命令还支持使用参数处理目录堆栈。使用 +n 或 -n 参数,其中 n 是一个数字,您能够向左或向右移动堆栈,如清单 5 所示。

清单 5. 旋转目录堆栈
1
2
3
4
5
6
$ dirs
/usr/local/bin /var /etc ~ ~
$ pushd +1
/var /etc ~ ~ /usr/local/bin
$ pushd -1
~ /usr/local/bin /var /etc ~

查找大型文件

是否须要找出您的全部空闲磁盘空间被什么占用了?您可使用如下几个工具来管理您的存储设备。如清单 6 所示,df 命令为您显示每一个可用卷上已使用的块的总数,以及空闲空间的百分比。

清单 6. 肯定卷的使用状况
1
2
3
4
5
6
7
$ df
Filesystem                            512-blocks      Used  Available Capacity  Mounted on
/dev/disk0s2                           311909984 267275264   44122720    86%    /
devfs                                        224       224          0   100%    /dev
fdesc                                          2         2          0   100%    /dev
map -hosts                                     0         0          0   100%    /net
map auto_home                                  0         0          0   100%    /home

是否但愿查找大型文件?使用 find 命令时附带 -size 参数。清单 7 显示了如何使用 find 命令来查找大于 10MB 的文件。请注意,-size 参数以 KB 为单位计量大小。

清单 7. 查找大于 10MB 的全部文件
1
$ find / -size +10000k –xdev –exec ls –lh {}\;

不使用编辑器建立临时文件

如下是一个简单示例:您须要快速建立一个简单临时文件,但不但愿启动您的编辑器。使用带有 > 文件重定向操做符的 cat 命令。如清单 8 所示,使用不带文件名的 cat 命令只回显向标准输入键入的任何内容;> 重定向将该输入捕获到指定的文件中。请注意,您在结束键入时必须提供文件结束字符,一般为 Ctrl-D。

清单 8. 快速建立临时文件
1
2
3
4
5
$ cat > my_temp_file.txt
This is my temp file text
^D
$ cat my_temp_file.txt
This is my temp file text

须要执行相同操做,可是附加到现有文件而不是建立新文件。如清单 9 所示,改用 >> 操做符。>> 文件重定向操做符向现有文件附加内容。

清单 9.快速向文件附加内容
1
2
3
4
5
6
$ cat >> my_temp_file.txt
More text
^D
$ cat my_temp_file.txt
This is my temp file text
More text

使用 curl 命令行实用工具

我是否能够从命令行访问 Web?你疯了吗?没有,这就是 curl 的用途!curl 命令使您可使用 HTTP、HTTPS、FTP、FTPS、Gopher、DICT、TELNET、LDAP 或 FILE 协议从服务器检索数据。如清单 10 所示,我可使用 curl 命令从美国国家气象局了解我所在位置(纽约州布法罗市)的当前天气情况。当与 grep 命令组合使用时,我能够检索布法罗市的天气情况。使用 -s 命令行选项来禁止 curl 处理输出。

清单 10. 使用 curl 检索当前天气情况
1
2
$ curl –s http://www.srh.noaa.gov/data/ALY/RWRALY | grep BUFFALO
BUFFALO        MOSUNNY   43  22  43 NE13      30.10R

清单 11 所示,您也可使用 curl 命令来下载 HTTP 托管的文件。使用 -o 参数来指定保存输出的位置。

清单 11. 使用 curl 下载 HTTP 承载的文件
1
$ curl -o archive.tar http://www.somesite.com/archive.tar

这实际上只是您使用 curl 命令能够完成的操做的提示。您只需在命令提示符处键入 man curl 显示 curl 命令的完整使用信息,就能够开始了解更多内容。

最有效地利用正则表达式

大量 UNIX 命令使用正则表达式做为参数。从技术角度而言,正则表达式 是表示某种模式的字符串(也就是说,由字母、数字和符号组成的字符序列),用于定义零或更长的字符串。正则表达式使用元字符(例如,星号 [*] 和问号 [?])来匹配其余字符串的部分或所有内容。正则表达式不必定包含通配符,但通配符可使正则表达式在搜索模式和处理文件时发挥更大的做用。表 1 显示了一些基本正则表达式序列。

表 1. 正则表达式序列

清单 12 显示了与 grep 命令一块儿使用的一些基本正则表达式。

清单 12. 使用正则表达式和 grep
1
2
3
4
5
6
7
8
9
10
11
12
13
14
$ # Lists your mail
$ grep '^From: ' /usr/mail/$USER  
$ # Any line with at least one letter 
$ grep '[a-zA-Z]'  search-file.txt
$ # Anything not a letter or number
$ grep '[^a-zA-Z0-9] search-file.txt
$ # Find phone numbers in the form 999-9999
$ grep '[0-9]\{3\}-[0-9]\{4\}' search-file.txt
$ # Find lines with exactly one character
$ grep '^.$' search-file.txt
$ #  Find any line that starts with a period "."         
$ grep '^\.' search-file.txt
$ # Find lines that  start with a "." and 2 lowercase letters
$ grep '^\.[a-z][a-z]' search-file.txt

有大量书籍专门讲述正则表达式。有关命令行正则表达式的深刻描述,建议您阅读 developerWorks 文章“对话 UNIX,第 9 部分:正则表达式。”

肯定当前用户

有时,您可能但愿肯定某个特定用户是否运行过您的管理脚本。为找出答案,您可使用 whoami 命令来返回当前用户的名称。清单 13 显示了独自运行的 whoami 命令;清单 14 显示了使用 whoami 确保当前用户不是根用户的 Bash 脚本的摘录。

清单 13. 从命令行使用 whoami
1
2
$ whoami
John
清单 14. 在脚本中使用 whoami
1
2
3
4
5
if [ $(whoami) = "root" ]
then
    echo "You cannot run this script as root."
    exit 1
fi

使用 awk 处理数据

awk 命令彷佛始终处在 Perl 的阴影下,但它对于简单、基于命令行的数据处理来讲是一个快速、实用的工具。清单 15 显示了如何开始使用 awk 命令。若要获取文件中每行文本的长度,请使用 length() 函数。若要查看字符串 ing 是否出如今文件文本中,请使用 index() 函数,该函数返回 ing 首次出现的位置,这样您就可使用它来进行进一步的字符串处理。若要 tokenize(也就是说,将一行拆分为单词长度的片断)某个字符串,请使用 split() 函数。

清单 15. 基本 awk 处理
1
2
3
4
5
6
7
8
9
10
11
$ cat text
testing the awk command
$ awk '{ i = length($0); print i }' text
23
$ awk '{ i = index($0,”ing”); print i}' text
5
$ awk 'BEGIN { i = 1 } { n = split($0,a," "); while (i <= n) {print a[i]; i++;} }' text
testing
the
awk
command

打印文本文件中的指定字段是一项简单的 awk 任务。在清单 16 中,sales 文件包含每一个销售人员的姓名,后跟每个月销售数字。您可使用 awk 命令来快速得到每月的销售总额。缺省状况下,awk 将每一个以逗号分隔的值视为不一样的字段。您使用 $n 操做符来访问每一个字段。

清单 16. 使用 awk 对数据进行汇总
1
2
3
4
5
6
7
8
9
10
$cat sales
Gene,12,23,7
Dawn,10,25,15
Renee,15,13,18
David,8,21,17
$ awk -F, '{print $1,$2+$3+$4}' sales
Gene 42
Dawn 50
Renee 46
David 46

awk 命令能够很复杂并应用于普遍的情景中。若要更完整地学习 awk 命令,请从命令 man awk 开始,并参阅参考资料部分提供的资源。

结束语

成为命令行高手须要进行一些实践。按照相同的方式处理问题很简单,由于您已经习惯了。扩展您的命令行资源能够显著提升您的工做效率,并促使您朝着 UNIX 命令行高手的方向前进!

相关文章
相关标签/搜索