-
A1. 附录 A:在其他环境中使用 Git
- A1.1 图形界面
- A1.2 在 Visual Studio 中使用 Git
- A1.3 在 Visual Studio Code 中使用 Git
- A1.4 在 IntelliJ / PyCharm / WebStorm / PhpStorm / RubyMine 中使用 Git
- A1.5 在 Sublime Text 中使用 Git
- A1.6 在 Bash 中使用 Git
- A1.7 在 Zsh 中使用 Git
- A1.8 在 PowerShell 中使用 Git
- A1.9 总结
-
A2. 附录 B:在你的应用程序中嵌入 Git
-
A3. 附录 C:Git 命令
7.5 Git 工具 - 搜索
搜索
对于任何大小的代码库,你经常需要找到一个函数在哪里被调用或定义,或者显示一个方法的历史。 Git 提供了一些有用的工具,可以快速轻松地查找存储在其数据库中的代码和提交。 我们将介绍其中的一些。
Git Grep
Git 自带一个名为 grep
的命令,允许你轻松地搜索任何已提交的树、工作目录,甚至索引中的字符串或正则表达式。 对于以下示例,我们将搜索 Git 本身的源代码。
默认情况下,git grep
将搜索你工作目录中的文件。 作为第一个变体,你可以使用 -n
或 --line-number
选项来打印 Git 找到匹配项的行号
$ git grep -n gmtime_r
compat/gmtime.c:3:#undef gmtime_r
compat/gmtime.c:8: return git_gmtime_r(timep, &result);
compat/gmtime.c:11:struct tm *git_gmtime_r(const time_t *timep, struct tm *result)
compat/gmtime.c:16: ret = gmtime_r(timep, result);
compat/mingw.c:826:struct tm *gmtime_r(const time_t *timep, struct tm *result)
compat/mingw.h:206:struct tm *gmtime_r(const time_t *timep, struct tm *result);
date.c:482: if (gmtime_r(&now, &now_tm))
date.c:545: if (gmtime_r(&time, tm)) {
date.c:758: /* gmtime_r() in match_digit() may have clobbered it */
git-compat-util.h:1138:struct tm *git_gmtime_r(const time_t *, struct tm *);
git-compat-util.h:1140:#define gmtime_r git_gmtime_r
除了上面显示的基本搜索之外,git grep
还支持大量其他有趣的选项。
例如,你可以要求 git grep
通过仅显示哪些文件包含搜索字符串以及每个文件中存在多少个匹配项来总结输出,而不是打印所有匹配项,使用 -c
或 --count
选项
$ git grep --count gmtime_r
compat/gmtime.c:4
compat/mingw.c:1
compat/mingw.h:1
date.c:3
git-compat-util.h:2
如果你对搜索字符串的*上下文*感兴趣,你可以使用 -p
或 --show-function
选项显示每个匹配字符串的封闭方法或函数
$ git grep -p gmtime_r *.c
date.c=static int match_multi_number(timestamp_t num, char c, const char *date,
date.c: if (gmtime_r(&now, &now_tm))
date.c=static int match_digit(const char *date, struct tm *tm, int *offset, int *tm_gmt)
date.c: if (gmtime_r(&time, tm)) {
date.c=int parse_date_basic(const char *date, timestamp_t *timestamp, int *offset)
date.c: /* gmtime_r() in match_digit() may have clobbered it */
正如你所看到的,gmtime_r
例程是从 date.c
文件中的 match_multi_number
和 match_digit
函数调用的(显示的第三个匹配项仅表示出现在注释中的字符串)。
你还可以使用 --and
标志搜索字符串的复杂组合,该标志确保多个匹配项必须出现在同一行文本中。例如,让我们查找任何定义常量且其名称包含子字符串“LINK”或“BUF_MAX”的行,特别是在由标签 v1.8.0
表示的旧版本 Git 代码库中(我们将使用 --break
和 --heading
选项,它们有助于将输出分成更易于阅读的格式)
$ git grep --break --heading \
-n -e '#define' --and \( -e LINK -e BUF_MAX \) v1.8.0
v1.8.0:builtin/index-pack.c
62:#define FLAG_LINK (1u<<20)
v1.8.0:cache.h
73:#define S_IFGITLINK 0160000
74:#define S_ISGITLINK(m) (((m) & S_IFMT) == S_IFGITLINK)
v1.8.0:environment.c
54:#define OBJECT_CREATION_MODE OBJECT_CREATION_USES_HARDLINKS
v1.8.0:strbuf.c
326:#define STRBUF_MAXLINK (2*PATH_MAX)
v1.8.0:symlinks.c
53:#define FL_SYMLINK (1 << 2)
v1.8.0:zlib.c
30:/* #define ZLIB_BUF_MAX ((uInt)-1) */
31:#define ZLIB_BUF_MAX ((uInt) 1024 * 1024 * 1024) /* 1GB */
与普通的搜索命令(如 grep
和 ack
)相比,git grep
命令有几个优点。首先,它非常快,其次,你可以搜索 Git 中的任何树,而不仅仅是工作目录。正如我们在上面的例子中看到的,我们搜索了旧版本 Git 源代码中的术语,而不是当前检出的版本。
Git 日志搜索
也许你寻找的不是术语存在于何处,而是何时存在或被引入。 git log
命令有许多强大的工具,可以通过消息的内容甚至它们引入的差异的内容来查找特定的提交。
例如,如果我们想找出 ZLIB_BUF_MAX
常量最初是什么时候引入的,我们可以使用 -S
选项(通常被称为 Git “镐头”选项)来告诉 Git 只显示那些更改了该字符串出现次数的提交。
$ git log -S ZLIB_BUF_MAX --oneline
e01503b zlib: allow feeding more than 4GB in one go
ef49a7a zlib: zlib can only process 4GB at a time
如果我们查看这些提交的差异,我们可以看到在 ef49a7a
中引入了该常量,而在 e01503b
中对其进行了修改。
如果需要更具体,你可以提供一个正则表达式,使用 -G
选项进行搜索。
行日志搜索
另一个相当高级的日志搜索是行历史搜索,它非常有用。只需运行带有 -L
选项的 git log
,它将向你展示代码库中函数或代码行的历史记录。
例如,如果我们想查看对 zlib.c
文件中的函数 git_deflate_bound
所做的每一个更改,我们可以运行 git log -L :git_deflate_bound:zlib.c
。这将尝试确定该函数的边界,然后查看历史记录,并向我们展示自该函数首次创建以来对其所做的每一个更改,以一系列补丁的形式。
$ git log -L :git_deflate_bound:zlib.c
commit ef49a7a0126d64359c974b4b3b71d7ad42ee3bca
Author: Junio C Hamano <gitster@pobox.com>
Date: Fri Jun 10 11:52:15 2011 -0700
zlib: zlib can only process 4GB at a time
diff --git a/zlib.c b/zlib.c
--- a/zlib.c
+++ b/zlib.c
@@ -85,5 +130,5 @@
-unsigned long git_deflate_bound(z_streamp strm, unsigned long size)
+unsigned long git_deflate_bound(git_zstream *strm, unsigned long size)
{
- return deflateBound(strm, size);
+ return deflateBound(&strm->z, size);
}
commit 225a6f1068f71723a910e8565db4e252b3ca21fa
Author: Junio C Hamano <gitster@pobox.com>
Date: Fri Jun 10 11:18:17 2011 -0700
zlib: wrap deflateBound() too
diff --git a/zlib.c b/zlib.c
--- a/zlib.c
+++ b/zlib.c
@@ -81,0 +85,5 @@
+unsigned long git_deflate_bound(z_streamp strm, unsigned long size)
+{
+ return deflateBound(strm, size);
+}
+
如果 Git 无法找出如何匹配你的编程语言中的函数或方法,你也可以向其提供正则表达式(或 regex)。例如,这与上面的示例效果相同:git log -L '/unsigned long git_deflate_bound/',/^}/:zlib.c
。你也可以给它一个行范围或一个单行行号,你会得到相同类型的输出。