#什么是grep:
grep (Globally search a Regular Expression and Print)
正则表达式全局搜索并打印信息,是一种强大的文本搜索工具,
#grep常用命令
查看grep的命令大全
android@tesdt-Precision-T1700:~$ grep --help
用法: grep [选项]... PATTERN [FILE]...
在每个 FILE 或是标准输入中查找 PATTERN。
默认的 PATTERN 是一个基本正则表达式(缩写为 BRE)。
例如: grep -i 'hello world' menu.h main.c
正则表达式选择与解释:
-E, --extended-regexp PATTERN 是一个可扩展的正则表达式(缩写为 ERE)
-F, --fixed-strings PATTERN 是一组由断行符分隔的定长字符串。
-G, --basic-regexp PATTERN 是一个基本正则表达式(缩写为 BRE)
-P, --perl-regexp PATTERN 是一个 Perl 正则表达式
-e, --regexp=PATTERN 用 PATTERN 来进行匹配操作
-f, --file=FILE 从 FILE 中取得 PATTERN
-i, --ignore-case 忽略大小写
-w, --word-regexp 强制 PATTERN 仅完全匹配字词
-x, --line-regexp 强制 PATTERN 仅完全匹配一行
-z, --null-data 一个 0 字节的数据行,但不是空行
杂项:
-s, --no-messages 不显示错误信息
-v, --invert-match 选中不匹配的行
-V, --version 显示版本信息并退出
--help 显示此帮助并退出
--mmap 忽略向后兼容性
输出控制:
-m, --max-count=NUM NUM 次匹配后停止
-b, --byte-offset 输出的同时打印字节偏移
-n, --line-number 输出的同时打印行号
--line-buffered 每行输出清空
-H, --with-filename 为每一匹配项打印文件名
-h, --no-filename 输出时不显示文件名前缀
--label=LABEL 将LABEL 作为标准输入文件名前缀
-o, --only-matching 只显示一行中匹配PATTERN 的部分
-q, --quiet, --silent 不显示所有输出
--binary-files=TYPE 假定二进制文件的TYPE 类型;
TYPE 可以是`binary', `text', 或`without-match'
-a, --text 等同于 --binary-files=text
-I 等同于 --binary-files=without-match
-d, --directories=ACTION 操作目录的方式;
ACTION 可以是`read', `recurse',或`skip'
-D, --devices=ACTION 操作设备、先入先出队列、套接字的方式;
ACTION 可以是`read'或`skip'
-R, -r, --recursive 等同于 --directories=recurse
--include=FILE_PATTERN 只查找匹配FILE_PATTERN 的文件
--exclude=FILE_PATTERN 跳过匹配FILE_PATTERN 的文件和目录
--exclude-from=FILE 跳过所有除FILE 以外的文件
--exclude-dir=PATTERN 跳过所有匹配PATTERN 的目录。
-L, --files-without-match 只打印不匹配FILEs 的文件名
-l, --files-with-matches 只打印匹配FILES 的文件名
-c, --count 只打印每个FILE 中的匹配行数目
-T, --initial-tab 行首tabs 分隔(如有必要)
-Z, --null 在FILE 文件最后打印空字符
文件控制:
-B, --before-context=NUM 打印以文本起始的NUM 行
-A, --after-context=NUM 打印以文本结尾的NUM 行
-C, --context=NUM 打印输出文本NUM 行
-NUM 等同于 --context=NUM
--color[=WHEN],
--colour[=WHEN] 使用标志高亮匹配字串;
WHEN 可以是`always', `never'或`auto'
-U, --binary 不要清除行尾的CR 字符(MSDOS 模式)
-u, --unix-byte-offsets 当CR 字符不存在,报告字节偏移(MSDOS 模式)
‘egrep’即‘grep -E’。‘fgrep’即‘grep -F’。
直接使用‘egrep’或是‘fgrep’均已不可行了。
不带 FILE 参数,或是 FILE 为 -,将读取标准输入。如果少于两个 FILE 参数
就要默认使用 -h 参数。如果选中任意一行,那退出状态为 0,否则为 1;
如果有错误产生,且未指定 -q 参数,那退出状态为 2。
请将错误报告给: bug-grep@gnu.org
GNU Grep 主页: <http://www.gnu.org/software/grep/>
GNU 软件的通用帮助: <http://www.gnu.org/gethelp/>
一些说明:
##grep的基本使用方法:
用法: grep [选项]... PATTERN [FILE]...
在每个 FILE 中按照选项的规定查找 PATTERN。
默认的 PATTERN 是一个基本正则表达式。
例如: grep -i 'hello world' menu.h main.c
在menu.h main.c文件中忽略大小写查找hello world。
##常用的选项说明
-i, --ignore-case 忽略大小写
-w, --word-regexp 强制 PATTERN 仅完全匹配字词
-n, --line-number 输出的同时打印行号
-R, -r, --recursive 等同于 --directories=recurse
--include=FILE_PATTERN 只查找匹配FILE_PATTERN 的文件
--exclude=FILE_PATTERN 跳过匹配FILE_PATTERN 的文件和目录
--exclude-from=FILE 跳过所有除FILE 以外的文件
--exclude-dir=PATTERN 跳过所有匹配PATTERN 的目录。
##常用RE(正则表达式)
正则表达式 | 说明 |
---|---|
\ | 忽略正则表达式中特殊字符的原有含义 |
^ | 开始字符 |
$ | 结束字符 |
< | 从匹配正则表达式的行开始 |
\> | 到匹配正则表达式的行结束 |
[ ] | 单个字符;如[A] 即A符合要求 |
[ - ] | 范围 ;如[A-Z]即A,B,C一直到Z都符合要求 |
. | 任意一个字符 |
? | 单个字符 至多1次 0次或1次 |
* | 所有字符,长度可以为0 1次或多次 |
+ | 1次或多次 |
{n} | 正好出现n次 |
{n,} | n次以上 |
{n,m} | 出现n到m次 |
| | 或 |
#grep 个人经验
个人经验,要想高效的使用grep来,有二个关键:
- 一是使用好grep的选项:
特别是对于海量的代码开发,比如手机开发(哈哈,懂的自然懂) ,要想在海量代码中找到自己想的字符串,这个需要许多技巧的。比如,指定更小的搜索的路径,跳过指定的搜索路径,指定搜索的文件类型,跳过指定的文件类型,等等。
总之,一句话:“请尽量缩小搜索范围”**
- 二是使用好正则表达式
正则表达式,是一个比较灵活的字符串匹配工具,许多时候确实是功能强大,正则表达式说起来有点玄,其实我们只要掌握一些基本的几个,就可以应用绝大多数的操作了。
#grep常用样例
- 在当前目录下查找字符串"what"
grep -rn "what" ./
- 在当前目录下的vendor和device目录下,忽略大小写完全匹配what
grep -rniw "what" vendor/ device/
- 在./LINUX/android/device/目录下的.mk文件下查找字符串"what"
grep -rn --include *.mk "what" ./LINUX/android/device/
- 在当前目录下除out目录之外的mk文件中搜索字符串"what"
grep -rn --include *.mk --exclude-dir out "what" ./
- 在packages/apps目录下所有除strings.xml的xml文件中查找"what"
忽略大小写,输出行号,文件夹查找
grep -nir --include=*.xml --exclude=strings.xml "what" packages/apps/
- 在packages/apps/Settings目录下所有除strings.xml的xml文件中查找"1.3"
忽略大小写,输出行号,文件夹查找
grep -rni --include *.xml --exclude strings.xml "1\.3" ./packages/apps/Settings/
- 在java文件中查找中文字符串
grep -rni --include=*.java --exclude-dir={out*,cts,sdk,V1*} -P ".*[\x{4e00}-\x{9fa5}]+" | tee chine_log_003.txt
- 通过管道过滤ls -l输出的内容,只显示以a开头的行
$ ls -l | grep "^a"
其实,我们要查看当前目录下的以a开关的内容,还有另外的命令:
find ./ -name "^a"
- grep not 操作
在packages/apps/Gallery2/Android.mk文件中,查找不能与optional匹配的行
grep -v "optional" packages/apps/Gallery2/Android.mk
- grep and 操作
在目录./packages/apps/目录下的mk格式的文件中查找匹配contact和com.android的字符串
grep -rn --include *.mk "contact" ./packages/apps/ | grep "com.android"
- grep or操作
在目录./packages/apps/目录下的mk格式的文件中查找匹配contact或com.android的字符串
grep -rn --include *.mk -e "contact" -e "com.android" ./packages/apps/
或者:
grep -rn --include *.mk -E "contact|com.android" ./packages/apps/
- 查看whatsapp应用中报错,权限,失败等相关日志信息:
grep -rni "whatsapp" logcat_2017_1010_1457.log | grep -i -E "err|Permission|denial|fail|androidruntime"
- 查找ip地址127.0.0.1 (其中包含特殊字符 ‘.’)
grep "127\.0\.0\.1" filename
- 只输出行中"id=xxx"的部分
grep -o 'id=[0-9]+' data.doc
下面是一些与grep相关联使用的命令:
- 管道 |
把一个命令的输出传递给另外一个命令为输入,管道是让用户将不同程序间的标准输入、标准输出连接起来,而不单单是重定向到文件
查找文件file.doc
ls | grep file.doc
在frameworks目录下的java文件中查找hello字符串
find ./frameworks/ -name "*.java" | xargs grep -rn "hello"
- 文件重定向 >
重定向标准输出到文件,覆盖文件
将frameworks目录下的java文件中查找hello字符串输出到./cqtest.txt文件中
find ./frameworks/ -name “*.java” | xargs grep “hello” > ./cqtest.txt
- xargs 命令
如果后面的程序必须要命令行参数,而不接受管道传参数,就要使用xargs
在frameworks目录下的java文件中查找hello字符串
find ./frameworks/ -name "*.java" | xargs grep -rn "hello"
- android开发优化后的命令:
排除了.git,lib,.gradle,.idea,captures,out,prebuilts文件夹下的文件,也排除了类似png,jpg,jar,so文件,查找ro.hwui.texture_cache_size,大大加快了查找效率
grep -rnwi --exclude-dir={.git,lib,.gradle,.idea,captures,out,prebuilts} --exclude={*.png,*.jpg,*.jar,*.so} "ro.hwui.texture_cache_size" ./