Git 如何处理符号链接?

如果我有一个符号链接文件或目录,并将其提交到 Git 存储库,那会发生什么?

我假设它将保留为符号链接,直到文件被删除为止,然后,如果您从旧版本中拉回文件,它只会创建一个普通文件。

删除文件引用时该怎么办?它只是提交悬空的链接吗?

答案

Git 只是将链接的内容(即,链接到的文件系统对象的路径)存储在 “blob” 中,就像处理普通文件一样。然后,它将名称,模式和类型(包括它是符号链接的事实)存储在表示其包含目录的树对象中。

签出包含链接的树时,无论目标文件系统对象是否存在,它都会将对象恢复为符号链接。

如果删除符号链接引用的文件,则不会以任何方式影响 Git 控制的符号链接。您将有一个悬而未决的参考。用户可以根据需要自行删除或更改链接以指向有效的内容。

您可以通过将文件添加到索引中来查看 Git 对文件的处理方式。索引就像一个预先提交。提交索引后,您可以使用git checkout将索引中的所有内容都带回到工作目录中。那么,当您向索引添加符号链接时,Git 会做什么?

为了找出答案,首先建立一个符号链接:

$ ln -s /path/referenced/by/symlink symlink

Git 尚不知道此文件。 git ls-files可让您检查索引( -s打印stat -like 输出):

$ git ls-files -s ./symlink
[nothing]

现在,通过将符号链接的内容添加到索引中,将其添加到 Git 对象存储中。将文件添加到索引时,Git 会将其内容存储在 Git 对象存储中。

$ git add ./symlink

那么,增加了什么呢?

$ git ls-files -s ./symlink
120000 1596f9db1b9610f238b78dd168ae33faa2dec15c 0       symlink

哈希是对在 Git 对象存储中创建的打包对象的引用。如果您在存储库的根目录中查找.git/objects/15/96f9db1b9610f238b78dd168ae33faa2dec15c ,则可以检查该对象。 这是 Git 存储在资源库中的文件,您以后可以检出。如果检查此文件,您会发现它很小。它不存储链接文件的内容。

(注意120000ls-files输出中列出的模式。常规文件的格式类似于100644

但是,当您从存储库中检出该对象并进入文件系统时,Git 对该对象有什么作用?这取决于core.symlinks配置。从man git-config

核心符号链接

如果为 false,则将符号链接检出为包含链接文本的小型纯文件。

因此,在存储库中使用符号链接时,签出后,您会获得一个参考完整文件系统路径的文本文件,或者是一个正确的符号链接,具体取决于core.symlinks配置的值。

无论哪种方式,符号链接引用的数据都不会存储在资源库中。

“编者注”:该帖子可能包含过时的信息。请参阅注释和有关自 1.6.1 起的 Git 更改的问题

链接目录:

重要的是要注意当目录为软链接时会发生什么。任何带有更新的 Git 拉动都将删除该链接并将其设置为普通目录。这就是我努力学习的方法。 这里这里有一些见解

之前

ls -l
 lrwxrwxrwx 1 admin adm   29 Sep 30 15:28 src/somedir -> /mnt/somedir

git add/commit/push

It remains the same

经过git pull和发现一些更新

drwxrwsr-x 2 admin adm 4096 Oct  2 05:54 src/somedir