使用 Unix 进行文本处理

正则表达式

翻译领域不乏让人摸不着头脑的词汇,比如“句柄”、“套接字”、“鲁棒性”。当然,“正则表达式”也属于这一类词汇。我刚接触正则表达式的时候,对这个名词感到非常迷惑。深入了解之后,才突然明白,原来所谓的 regular expression, 其实就是“有规律、有模式的字符串”而已。

很少有一门技术,只需要投入少量的学习成本即可获得巨大的价值回报。正则表达式就属于这一类技术。可惜很多人被它密码般的语法形式当头棒喝,甚至连门都不得而入。

为什么你应该学习正则表达式?其一,在实践中应用这门技术其实不难,只需理解为数不多的几个元字符以及并不复杂的语法,就能够获得强大的文本操控能力;其二,正则表达式往往能提供处理文本的最简单最高效的解决方法(有时也许是唯一的解法)。遇上复杂的情况,如果你不会正则表达式,就只好束手无策、黯然神伤了。

正则表达式入门容易,精通却难。本文并不打算挑战此项任务,如果你希望系统地学习正则表达式,请一定阅读 Jeffrey Friedl 的著作 精通正则表达式

regex.png

文本检索

grep 命令可以完成简单的文本搜索任务。

先来准备一份文本材料,把 grep 的帮助页保存为文本文件:

> man grep | col -b > grephelp.txt

下面,我想检索 grephelp.txt 文件中所有包含 "find" 这个单词的文本行:

> grep "find" grephelp.txt
     To find all occurrences of the word `patricia' in a file:
     To find all occurrences of the pattern `.Pp' at the beginning of a line:
     To find all lines in a file which do not contain the words `foo' or

我希望匹配到的文本使用不同的颜色显示,可以添加 --color 选项,默认的颜色是红色。

> grep --color "find" grephelp.txt

我希望在匹配结果中显示文件名和行号,使用 -H 选项可以显示文件名,使用 -n 选项可以显示行号:

> grep -H -n --color "find" grephelp.txt
grephelp.txt:252:     To find all occurrences of the word `patricia' in a file:
grephelp.txt:256:     To find all occurrences of the pattern `.Pp' at the beginning of a line:
grephelp.txt:265:     To find all lines in a file which do not contain the words `foo' or

很多时候,我们需要知道匹配行前后的上下文。-A-B 这两个选项会是你的好朋友。-A n 表示显示匹配行以及其后的 n 行;-B n 表示显示匹配行以及之前的 n 行。现在,我们在匹配行的前后分别额外显示两行:

> grep -A 2 -B 2 -H -n --color "find" grephelp.txt
grephelp.txt-250-
grephelp.txt-251-EXAMPLES
grephelp.txt:252:     To find all occurrences of the word `patricia' in a file:
grephelp.txt-253-
grephelp.txt-254-      $ grep 'patricia' myfile
--
--
grephelp.txt-254-      $ grep 'patricia' myfile
grephelp.txt-255-
grephelp.txt:256:     To find all occurrences of the pattern `.Pp' at the beginning of a line:
grephelp.txt-257-
grephelp.txt-258-      $ grep '^\.Pp' myfile
--
--
grephelp.txt-263-     match any character.
grephelp.txt-264-
grephelp.txt:265:     To find all lines in a file which do not contain the words `foo' or
grephelp.txt-266-     `bar':
grephelp.txt-267-

如果需要查找所有不包含 "find" 的文本行,该怎么做呢?很简单,使用 -v 选项即可。

grep 还有两个变体,egrep 和 fgrep。相对于仅支持基本正则模式(BREs)的 grep 来说,egrep 支持扩展正则模式(EREs),因而检索能力更为强大;fgrep 是所有三个工具中速度最快的一个,因为它完全不支持正则模式。

其实,我更喜欢一个叫做 ack 的工具。

ack

文本替换

tr 命令可以完成简单的字符转换任务。例如,可以通过 tr 把 grephelp.txt 文件转换为全文大写:

> cat grephelp.txt | tr '[:lower:]' '[:upper:]'

简而言之,tr 的工作就是把第一个集合中的字符转换为第二个集合中的相应的字符。常用的字符集合有下面这些:

  • [:alnum:]:字母数字
  • [:alpha:]:字母
  • [:cntrl:] :控制字符
  • [:digit:]:数字
  • [:graph:]: 图形字符
  • [:lower:]:小写字母
  • [:print:]:可打印字符
  • [:punct:]:标点符号
  • [:space:]:空白字符
  • [:upper:]:大写字母
  • [:xdigit:]:十六进制数字

tr 命令的应用场景非常受限,如果希望进行更加灵活的模式替换,我们还有 sed(也就是 stream editor,流编辑器)。

把文件中所有的 "find" 文本替换为 "search":

> sed "s/find/search/g" grephelp.txt

这条命令中,s 表示执行“替换操作”,/find/search/ 表示把 "find" 替换为 "search",g 表示对一行中所有的匹配进行替换。sed 默认把处理结果打印到标准输出,我们可以通过重定向把处理结果转储到一个新文件中,或者使用选项 -i 把结果直接写回原文件(有风险,需谨慎):

> sed -i "s/find/search/g" grephelp.txt

把文件中所有的数字 n 替换为 "--n--" 的形式:

> sed -E "s/([0-9]+)/--\1--/g" grephelp.txt

选项 -E 表示在处理过程中使用扩展的正则模式(EREs),替换命令中的 \1 表示引用正则表达式的第一个捕获分组。请注意,-E 这个选项只在 Mac OS X 系统和 FreeBSD 系统上有效,其他 Unix 系统需要使用另一个等效的选项 -r

sed 的功能远不止这一些,篇幅所限,不可能详细讲解 sed 的用法。如果希望学习更多,请移步这篇文章

文本去重

> cat -n sonnet116.txt
     1  Let me not to the marriage of true minds
     2  Admit impediments. Love is not love
     3  Which alters when it alteration finds,
     4  Or bends with the remover to remove:
     5  O, no! it is an ever-fix`ed mark,
     6  O, no! it is an ever-fix`ed mark,
     7  That looks on tempests and is never shaken;
     8  It is the star to every wand'ring bark,
     9  Whose worth's unknown, although his heighth be taken.
    10  Love's not Time's fool, though rosy lips and cheeks
    11  Love's not Time's fool, though rosy lips and cheeks
    12  Love's not Time's fool, though rosy lips and cheeks
    13  Within his bending sickle's compass come;
    14  Love alters not with his brief hours and weeks,
    15  But bears it out even to the edge of doom:
    16  If this be error and upon me proved,
    17  I never writ, nor no man ever loved.

这是莎士比亚的一首十四行诗,只可惜第5行和第10行有重复(而且第10行重复了3次)。怎么查看文本中重复的行呢?uniq 命令可以帮助你。

> uniq -d sonnet116.txt
O, no! it is an ever-fix`ed mark,
Love's not Time's fool, though rosy lips and cheeks

选项 -d 表示仅输出重复的行。如果需要去重,使用不带选项的 uniq 命令就可以了:

> uniq sonnet116.txt
Let me not to the marriage of true minds
Admit impediments. Love is not love
Which alters when it alteration finds,
Or bends with the remover to remove:
O, no! it is an ever-fix`ed mark,
That looks on tempests and is never shaken;
It is the star to every wand'ring bark,
Whose worth's unknown, although his heighth be taken.
Love's not Time's fool, though rosy lips and cheeks
Within his bending sickle's compass come;
Love alters not with his brief hours and weeks,
But bears it out even to the edge of doom:
If this be error and upon me proved,
I never writ, nor no man ever loved.

想要查看每一行究竟重复了多少次?没问题,使用选项 -c

> uniq -c sonnet116.txt
   1 Let me not to the marriage of true minds
   1 Admit impediments. Love is not love
   1 Which alters when it alteration finds,
   1 Or bends with the remover to remove:
   2 O, no! it is an ever-fix`ed mark,
   1 That looks on tempests and is never shaken;
   1 It is the star to every wand'ring bark,
   1 Whose worth's unknown, although his heighth be taken.
   3 Love's not Time's fool, though rosy lips and cheeks
   1 Within his bending sickle's compass come;
   1 Love alters not with his brief hours and weeks,
   1 But bears it out even to the edge of doom:
   1 If this be error and upon me proved,
   1 I never writ, nor no man ever loved.

文本排序

假设有这样一个报表文件,第一列是月份,第二列是当月的销售个数:

> cat report.txt
March,19
June,50
February,17
May,18
August,16
April,31
May,18
July,26
January,24
August,16

这个文件的内容不仅顺序是乱的,而且还有重复。我希望按字母表顺序排序,可以下面这个命令:

> sort report.txt
April,31
August,16
August,16
February,17
January,24
July,26
June,50
March,19
May,18
May,18

选项 -u (表示 unique)可以在排序结果中去除重复行:

> sort -u report.txt
April,31
August,16
February,17
January,24
July,26
June,50
March,19
May,18

能不能按照月份排序呢?选项 -M (表示 month-sort)可以帮助我们:

> sort -u -M report.txt
January,24
February,17
March,19
April,31
May,18
June,50
July,26
August,16

按照第二列的数字进行排序也是很简单的:

> sort -u -t',' -k2 report.txt
August,16
February,17
May,18
March,19
January,24
July,26
April,31
June,50

上面的例子中,选项 -t',' 表示以逗号为分隔符对文本进行列分割;-k2 表示对第2列进行排序。

当然了,把结果逆序排列也并非不可能:

> sort -u -r -t',' -k2 report.txt
June,50
April,31
July,26
January,24
March,19
May,18
February,17
August,16

文本统计

wc 命令用来完成文本统计工作,通过使用不同的选项,它可以统计文件中的字节数(-c),字符数(-m),单词数(-w)与行数(-l)。

例如,查看 grephelp.txt 这个文件总共有多少个单词:

> wc -w grephelp.txt
    1571 grephelp.txt

查看 sonnet116.txt 这个文件总共有多少不重复的行(废话,十四行诗当然是有14行):

> uniq sonnet116.tx6 | wc -l
      14

你还应该试试 Awk 与 Perl

如果上面介绍的工具仍然不能满足你,也许你需要火力更强的武器。试试 Awk 与 Perl 吧。

Awk 也是一款上古神器,它的年龄可能和 sed 不相上下。Awk 可谓是专门为了文本处理而生,它的语法和特性非常适合用于操纵文本和生成报表。如需学习,请参考 这篇文章,你会喜欢上它的。

长久以来,Perl 背负了“只写语言”的恶名。实际上,只要处理得当,用 Perl 一样可以写出模块清晰的、容易阅读和理解的代码。根据我的经验,使用 Perl 的场合 80% 以上与文本处理有关。Perl 内置的正则表达式支持可能是所有语言中最好的,再加上简洁紧凑的语法以及便利的操作符,这些特性帮助 Perl 成了文本处理领域当仁不让的霸主。

图片来自 xkcd.com
图片来自 xkcd.com
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 157,198评论 4 359
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,663评论 1 290
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 106,985评论 0 237
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,673评论 0 202
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 51,994评论 3 285
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,399评论 1 211
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,717评论 2 310
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,407评论 0 194
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,112评论 1 239
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,371评论 2 241
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,891评论 1 256
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,255评论 2 250
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,881评论 3 233
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,010评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,764评论 0 192
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,412评论 2 269
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,299评论 2 260

推荐阅读更多精彩内容

  • 知识点 sort uniq cut wc sed命令 awk命令 crontab定时器 sort sort 命令对...
  • Unix 进行文本处理 正则表达式 翻译领域不乏让人摸不着头脑的词汇,比如“句柄”、“套接字”、“鲁棒性”。当然,...
    鲍陈飞阅读 683评论 0 1
  • 本文承接之前写的三十分钟学会AWK一文,在学习完AWK之后,趁热打铁又学习了一下SED,不得不说这两个工具真的堪称...
    mylxsw阅读 4,359评论 3 74
  • sed与awk实例 文本间隔 在每一行后面增加一空行 将原来的所有空行删除并在每一行后面增加一空行。这样在输出的文...
    stuha阅读 1,840评论 0 21
  • linux资料总章2.1 1.0写的不好抱歉 但是2.0已经改了很多 但是错误还是无法避免 以后资料会慢慢更新 大...
    数据革命阅读 12,011评论 2 34