写点什么

文本搜索工具 ack 与 grep

作者:坚果
  • 2023-04-24
    广东
  • 本文字数:3877 字

    阅读完需:约 13 分钟

文本搜索工具ack与grep

作者:坚果

团队:坚果派

公众号:“大前端之旅”

润开鸿技术专家,华为 HDE,InfoQ 签约作者,OpenHarmony 布道师,擅长 HarmonyOS 应用开发、熟悉服务卡片开发,在“战码先锋”活动中作为大队长,累计培养三个小队长,带领 100+队员完成 Pr 的提交合入。欢迎通过主页或者私信联系我,加入坚果派,一起学习 OpenHarmony 应用开发。


今天在解读 AT 指令源码的时候有这样一个函数,hi_at_sys_cmd_register,但是我搜索不到它的出处,于是就有了对文本工具的探索。首先使用的是 grep,


下面是我的搜索结果



但是可以看到的是它的目录并不是很清晰,刚好在梁老师的推荐下,使用了 ack 工具。

ack

大家都说它是比 grep 好用的文本搜索工具

安装

# ubuntu下要安装ack-grep,因为在debian系中,ack这个名字被其他的软件占用了。sudo apt-get install ack-grep# alpine Linux-apk软件包管理器 安装 ackapk install ack
复制代码

特点

ack官网列出了这工具的 5 大卖点:


  1. 速度非常快,因为它只搜索有意义的东西。

  2. 更友好的搜索,忽略那些不是你源码的东西。

  3. 为源代码搜索而设计,用更少的击键完成任务。

  4. 非常轻便,移植性好。

  5. 免费且开源

实例

在记忆的时候大体上可以分为这几个部分:


Searching 代码搜索 Search output 搜索结果处理 File presentation 文件展示 File finding 文件查找 File inclusion/exclusion 文件过滤


grep 常用操作


grep -r 'hello_world' # 简单用法grep '^hello_world' . # 简单正则ls -l | grep .py # 管道用法
复制代码

Searching

简单的文本搜索,默认是递归的。


ack-grep helloack-grep -i helloack-grep -v helloack-grep -w helloack-grep -Q 'hello*'
复制代码

Search File

对搜索结果进行处理,比如只显示一个文件的一个匹配项,或者 xxx


ack-grep --line=1       # 输出所有文件第二行ack-grep -l 'hello'     # 包含的文件名ack-grep -L 'print'     # 非包含文件名
复制代码

File presentation

输出的结果是以什么方式展示呢,这个部分有几个参数可以练习下


ack-grep hello --pager='less -R'    # 以less形式展示ack-grep hello --noheading      # 不在头上显示文件ack-grep hello --nocolor        # 不对匹配字符着色
复制代码

File finding

没错,它可以查找文件,以省去你要不断的结合 find 和 grep 的麻烦,虽然在 linux 的思想是一个工具做好一件事。


ack-grep -f hello.py     # 查找全匹配文件ack-grep -g hello.py$    # 查找正则匹配文件ack-grep -g hello  --sort-files     # 查找然后排序
复制代码

File Inclusion/Exclusion

文件过滤,个人觉得这是一个很不错的功能。如果你曾经在搜索项目源码是不小心命中日志中的某个关键字的话,你会觉得这个有用。


ack-grep --python hello       # 查找所有python文件ack-grep -G hello.py$ hello   # 查找匹配正则的文件
复制代码


最后来看看用 grep 搜索的结果。



对比下来,是不是 ack 更加好用呢。


这里顺便记录一下 grep 工具的常用命令

grep

强大的文本搜索工具

补充说明

grep (全面搜索正则表达式并把行打印出来)是一种强大的文本搜索工具,它能使用正则表达式搜索文本,并把匹配的行打印出来。用于过滤/搜索的特定字符。可使用正则表达式能配合多种命令使用,使用上十分灵活。

选项

-a --text  # 不要忽略二进制数据。-A <显示行数>   --after-context=<显示行数>   # 除了显示符合范本样式的那一行之外,并显示该行之后的内容。-b --byte-offset                           # 在显示符合范本样式的那一行之外,并显示该行之前的内容。-B<显示行数>   --before-context=<显示行数>   # 除了显示符合样式的那一行之外,并显示该行之前的内容。-c --count    # 计算符合范本样式的列数。-C<显示行数> --context=<显示行数>或-<显示行数> # 除了显示符合范本样式的那一列之外,并显示该列之前后的内容。-d<进行动作> --directories=<动作>  # 当指定要查找的是目录而非文件时,必须使用这项参数,否则grep命令将回报信息并停止动作。-e<范本样式> --regexp=<范本样式>   # 指定字符串作为查找文件内容的范本样式。-E --extended-regexp             # 将范本样式为延伸的普通表示法来使用,意味着使用能使用扩展正则表达式。-f<范本文件> --file=<规则文件>     # 指定范本文件,其内容有一个或多个范本样式,让grep查找符合范本条件的文件内容,格式为每一列的范本样式。-F --fixed-regexp   # 将范本样式视为固定字符串的列表。-G --basic-regexp   # 将范本样式视为普通的表示法来使用。-h --no-filename    # 在显示符合范本样式的那一列之前,不标示该列所属的文件名称。-H --with-filename  # 在显示符合范本样式的那一列之前,标示该列的文件名称。-i --ignore-case    # 忽略字符大小写的差别。-l --file-with-matches   # 列出文件内容符合指定的范本样式的文件名称。-L --files-without-match # 列出文件内容不符合指定的范本样式的文件名称。-n --line-number         # 在显示符合范本样式的那一列之前,标示出该列的编号。-P --perl-regexp         # PATTERN 是一个 Perl 正则表达式-q --quiet或--silent     # 不显示任何信息。-R/-r  --recursive       # 此参数的效果和指定“-d recurse”参数相同。-s --no-messages  # 不显示错误信息。-v --revert-match # 反转查找。-V --version      # 显示版本信息。   -w --word-regexp  # 只显示全字符合的列。-x --line-regexp  # 只显示全列符合的列。-y # 此参数效果跟“-i”相同。-o # 只输出文件中匹配到的部分。-m <num> --max-count=<num> # 找到num行结果后停止查找,用来限制匹配行数
复制代码

规则表达式

^    # 锚定行的开始 如:'^grep'匹配所有以grep开头的行。    $    # 锚定行的结束 如:'grep$' 匹配所有以grep结尾的行。.    # 匹配一个非换行符的字符 如:'gr.p'匹配gr后接一个任意字符,然后是p。    *    # 匹配零个或多个先前字符 如:'*grep'匹配所有一个或多个空格后紧跟grep的行。    .*   # 一起用代表任意字符。   []   # 匹配一个指定范围内的字符,如'[Gg]rep'匹配Grep和grep。    [^]  # 匹配一个不在指定范围内的字符,如:'[^A-Z]rep' 匹配不包含 A-Z 中的字母开头,紧跟 rep 的行\(..\)  # 标记匹配字符,如'\(love\)',love被标记为1。    \<      # 锚定单词的开始,如:'\<grep'匹配包含以grep开头的单词的行。    \>      # 锚定单词的结束,如'grep\>'匹配包含以grep结尾的单词的行。    x\{m\}  # 重复字符x,m次,如:'0\{5\}'匹配包含5个o的行。    x\{m,\}   # 重复字符x,至少m次,如:'o\{5,\}'匹配至少有5个o的行。    x\{m,n\}  # 重复字符x,至少m次,不多于n次,如:'o\{5,10\}'匹配5--10个o的行。   \w    # 匹配文字和数字字符,也就是[A-Za-z0-9],如:'G\w*p'匹配以G后跟零个或多个文字或数字字符,然后是p。   \W    # \w的反置形式,匹配一个或多个非单词字符,如点号句号等。   \b    # 单词锁定符,如: '\bgrep\b'只匹配grep。  
复制代码

grep 命令常见用法

在文件中搜索一个单词,命令会返回一个包含 “match_pattern” 的文本行:


grep match_pattern file_namegrep "match_pattern" file_name
复制代码


在多个文件中查找:


grep "match_pattern" file_1 file_2 file_3 ...
复制代码


输出除之外的所有行 -v 选项:


grep -v "match_pattern" file_name
复制代码


标记匹配颜色 --color=auto 选项:


grep "match_pattern" file_name --color=auto
复制代码


使用正则表达式 -E 选项:


grep -E "[1-9]+"# 或egrep "[1-9]+"
复制代码


使用正则表达式 -P 选项:


grep -P "(\d{3}\-){2}\d{4}" file_name
复制代码


只输出文件中匹配到的部分 -o 选项:


echo this is a test line. | grep -o -E "[a-z]+\."line.
echo this is a test line. | egrep -o "[a-z]+\."line.
复制代码


统计文件或者文本中包含匹配字符串的行数 -c 选项:


grep -c "text" file_name
复制代码


搜索命令行历史记录中 输入过 git 命令的记录:


history | grep git
复制代码


输出包含匹配字符串的行数 -n 选项:


grep "text" -n file_name# 或cat file_name | grep "text" -n
#多个文件grep "text" -n file_1 file_2
复制代码


打印样式匹配所位于的字符或字节偏移:


echo gun is not unix | grep -b -o "not"7:not#一行中字符串的字符偏移是从该行的第一个字符开始计算,起始值为0。选项  **-b -o**  一般总是配合使用。
复制代码


搜索多个文件并查找匹配文本在哪些文件中:


grep -l "text" file1 file2 file3...
复制代码

grep 递归搜索文件

在多级目录中对文本进行递归搜索:


grep "text" . -r -n# .表示当前目录。
复制代码


忽略匹配样式中的字符大小写:


echo "hello world" | grep -i "HELLO"# hello
复制代码


选项 -e 制动多个匹配样式:


echo this is a text line | grep -e "is" -e "line" -oisisline
#也可以使用 **-f** 选项来匹配多个样式,在样式文件中逐行写出需要匹配的字符。cat patfileaaabbb
echo aaa bbb ccc ddd eee | grep -f patfile -o
复制代码


在 grep 搜索结果中包括或者排除指定文件:


# 只在目录中所有的.php和.html文件中递归搜索字符"main()"grep "main()" . -r --include *.{php,html}
# 在搜索结果中排除所有README文件grep "main()" . -r --exclude "README"
# 在搜索结果中排除filelist文件列表里的文件grep "main()" . -r --exclude-from filelist
复制代码


使用 0 值字节后缀的 grep 与 xargs:


# 测试文件:echo "aaa" > file1echo "bbb" > file2echo "aaa" > file3
grep "aaa" file* -lZ | xargs -0 rm
# 执行后会删除file1和file3,grep输出用-Z选项来指定以0值字节作为终结符文件名(\0),xargs -0 读取输入并用0值字节终结符分隔文件名,然后删除匹配文件,-Z通常和-l结合使用。
复制代码


发布于: 刚刚阅读数: 7
用户头像

坚果

关注

此间若无火炬,我便是唯一的光 2020-10-25 加入

公众号:“大前端之旅”,OpenHarmony布道师,润和软件鸿蒙KOL,InfoQ签约作者,电子发烧友鸿蒙KOL

评论

发布
暂无评论
文本搜索工具ack与grep_Linux_坚果_InfoQ写作社区