使用 Linux/Unix 进行文本处理

v

使用 Linux/Unix 进行文本处理

正则表达式

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

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

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

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

正则表达式

文本检索

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

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


  1. > man grep | col -b > grephelp.txt

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


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

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


  1. > grep --color "find" grephelp.txt

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


  1. > grep -H -n --color "find" grephelp.txt
  2. grephelp.txt:252: To find all occurrences of the word `patricia' in a file:
  3. grephelp.txt:256: To find all occurrences of the pattern `.Pp' at the beginning of a line:
  4. 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 行。现在,我们在匹配行的前后分别额外显示两行:


  1. > grep -A 2 -B 2 -H -n --color "find" grephelp.txt
  2. grephelp.txt-250-
  3. grephelp.txt-251-EXAMPLES
  4. grephelp.txt:252: To find all occurrences of the word `patricia' in a file:
  5. grephelp.txt-253-
  6. grephelp.txt-254- $ grep 'patricia' myfile
  7. --
  8. --
  9. grephelp.txt-254- $ grep 'patricia' myfile
  10. grephelp.txt-255-
  11. grephelp.txt:256: To find all occurrences of the pattern `.Pp' at the beginning of a line:
  12. grephelp.txt-257-
  13. grephelp.txt-258- $ grep '^\.Pp' myfile
  14. --
  15. --
  16. grephelp.txt-263- match any character.
  17. grephelp.txt-264-
  18. grephelp.txt:265: To find all lines in a file which do not contain the words `foo' or
  19. grephelp.txt-266- `bar':
  20. grephelp.txt-267-

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

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

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

ack

文本替换

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


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

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

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

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

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


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

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


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

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


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

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

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

文本去重


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

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


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

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


  1. > uniq sonnet116.txt
  2. Let me not to the marriage of true minds
  3. Admit impediments. Love is not love
  4. Which alters when it alteration finds,
  5. Or bends with the remover to remove:
  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. Within his bending sickle's compass come;
  12. Love alters not with his brief hours and weeks,
  13. But bears it out even to the edge of doom:
  14. If this be error and upon me proved,
  15. I never writ, nor no man ever loved.

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


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

文本排序

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


  1. > cat report.txt
  2. March,19
  3. June,50
  4. February,17
  5. May,18
  6. August,16
  7. April,31
  8. May,18
  9. July,26
  10. January,24
  11. August,16

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


  1. > sort report.txt
  2. April,31
  3. August,16
  4. August,16
  5. February,17
  6. January,24
  7. July,26
  8. June,50
  9. March,19
  10. May,18
  11. May,18

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


  1. > sort -u report.txt
  2. April,31
  3. August,16
  4. February,17
  5. January,24
  6. July,26
  7. June,50
  8. March,19
  9. May,18

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


  1. > sort -u -M report.txt
  2. January,24
  3. February,17
  4. March,19
  5. April,31
  6. May,18
  7. June,50
  8. July,26
  9. August,16

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


  1. > sort -u -t',' -k2 report.txt
  2. August,16
  3. February,17
  4. May,18
  5. March,19
  6. January,24
  7. July,26
  8. April,31
  9. June,50

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

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


  1. > sort -u -r -t',' -k2 report.txt
  2. June,50
  3. April,31
  4. July,26
  5. January,24
  6. March,19
  7. May,18
  8. February,17
  9. August,16

文本统计

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

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


  1. > wc -w grephelp.txt
  2. 1571 grephelp.txt

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


  1. > uniq sonnet116.tx6 | wc -l
  2. 14

你还应该试试 Awk 与 Perl

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

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

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

本文来自合作伙伴“Linux中国”,原文发表于2013-04-02.

时间: 2024-09-13 19:54:56

使用 Linux/Unix 进行文本处理的相关文章

Recoll:Unix和Linux桌面的文本搜索工具

Recoll是一个Unix和Linux桌面的文本搜索工具.Recoll可以搜索文件名和文件内的关键字. Recoll可以为你做到以下这些. 它可以搜索任何格式的文件 支持通配符 可以依据文件的作者.类型.大小和格式等条件进行搜索 可以搜索存储在任何位置的文件,例如文件.归档文件.邮件附件等 支持桌面和网站集成 具备火狐插件,可索引web页的历史 点击一下搜索结果就可以在本地编辑器内打开或者显示文本预览 它是免费的开源的,在GPL许可下发布 在Ubuntu/Linux Mint上安装Recoll

Linux/Unix的精巧约定两例及其简析:目录权限和文本行数

学玩*nux时候,碰到的一些问题,弄明白了后也就过去了.今天看到旁边的同学对目录权限有些模糊,给解释了一下.想想不如把这些问题都记下来. 设计其实包含的是一套约定.能运行.解决问题的约定都是可用的约定.但解决的多种约定方式或说是设计中,作一些比较可以感觉到哪个会更统一更简单.下文提到的两例Linux/Unix约定说明后,我也简单分析一下约定,找出看似复杂或是不直觉约定中内部包含的简洁统一. 清楚的同学,就当是个活动脑子的问题,过一遍小乐一下  如有谬误敬请指正. 目录权限的约定 目录的执行权限

10个 Linux/Unix下 Bash 和 KSH shell 的作业控制实例

Linux 和 Unix 属于多任务的操作系统,也就是说一个系统在同一时间段内能运行多重任务(进程).在这个新的博客系列,我将会列出相关的 Linux 和 Unix 作业(job)控制的命令,你可以通过这些命令在 Bash 或 Korn 还有 POSIX shell 下实现执行多重任务. 什么是作业控制? 作业控制不只是能够停止/挂起(stop/suspend)正在执行的进程(命令),也可以继续/唤醒(continue/resume)执行你需要的每一个进程.这完全可以用你的操作系统和 bash/

分布式DB2 10.5 for Linux, UNIX和Windows特性的区别

DB2 10.5 产品家族包含 6 个付费版本,一个单独付费的特性和一个免费包.本文的目的是帮助您理解它们之间的区别. 文中还会列出 DB2 10.5 中提供的新功能,比如 BLU Acceleration.DB2 pureScale 增强.SQL 兼容性增强和简化的产品包装. DB2 with BLU Acceleration 结合了高级.创 新的功能,以加速执行数据库和数据仓库的工作负载分析.DB2 with BLU Acceleration 还集成了 IBM Cognos® Busines

grep 命令系列:如何在 Linux/UNIX 中使用 grep 命令

grep 命令系列:如何在 Linux/UNIX 中使用 grep 命令 我该怎样在 Linux.Apple OS X 及其他类 UNIX 系统中使用 grep 命令,你能给我展示一些简单的例子吗? grep 命令用来搜索文本,或从给定的文件中搜索行内包含了给定字符串或单词的文件.通常来说,grep 显示匹配到的行.使用 grep 来搜索包括一个或多个正则表达式匹配到的文本行,然后只显示匹配到的行.grep 被视作在 Linux/ Unix 系统中最有用的命令之一. 你知道吗 grep 这个名字

windows-wine效果如何 linux unix mac os?

问题描述 wine效果如何 linux unix mac os? wine能支持哪类exe程序,效率又如何呢?即使能直接支行exe又何必学跨平台的工具那么麻烦呢? 解决方案 wine在os api层面虚拟化,因此执行的是nativecode.但是不同os下程序的behaviour有差异.如果程序性能开销不发生在这些os api调用上,那么差异不大.反之,因为api移植和实现的不同,还是有差异的.因为wine是api虚拟化,执行的是本地代码,所以不能在不同的机器上跨平台.比如x86的程序不能在ar

Linux/Unix下ODBC的安装、配置与编程

odbc|unix|编程 Linux/Unix下ODBC的安装.配置与编程 齐亮 (cavendish@eyou.com) 2002 年 7 月 本文主要内容是介绍ODBC的简单原理,以及如何在Linux/Unix下进行ODBC的安装.配置与编程. 一. ODBC原理 ODBC 是Open Database Connect 即开放数据库互连的简称,它是由Microsoft 公司于1991 年提出的一个用于访问数据库的统一界面标准,是应用程序和数据库系统之间的中间件.它通过使用相应应用平台上和所需

教你如何在Linux/Unix平台安装MySQL

mysql|unix [导读]介绍如何在Linux/Unix平台下安装MySQL. Linux下的安装步骤: 推荐使用RPM工具来进行Linux下的MySQL数据库安装.目前的MySQL rpm安装文件包是在SuSE Linux 7.3系统平台上打造而成的,但是在绝大多数支持RPM和glibc的其他Linux平台中也可以进行安装.如果选择通用的RPM安装包,那么RPM将静态链接到Linux的线程中.下面步骤执行了一个标准的最小化安装: shell> rpm -i MySQL-server-VER

为DB2 UDB for Linux, UNIX, and Windows提供Unicode支持的实用方法

简介 当今的应用程序常常设计为供国际化使用.这些应用程序可能需要处理不同语言的字符串 .而 Unicode 正是一种与语言无关的字符表示标准. 由于 Java 编程语言已经在内部使用 Unicode 来表示字符,所以国际化应用程序的开发已经容易多了.但是,不能只考虑应用程序端.后端的 数据库也必须能够处理 Unicode 字符.本文将讨论几个主题,帮助开发人员实现供国际化使用的 DB2 UDB 应用程序. DB2 中支持何种 Unicode 标准? 只有一种 Unicode 标准,但是存在不同的