Git 提交时神秘的 create mode 100644 到底是什么?一文告诉你答案!

简介

上下文:它出现在哪里?

常见于以下命令输出中:

$ git show

输出示例:

diff --git a/src/test.txt b/src/test.txt
new file mode 100644
index 0000000..7f3e5a4
--- /dev/null
+++ b/src/test.txt
@@ -0,0 +1,2 @@
hello
world
$ git commit -m "Add new files"
[main 1a2b3c4] Add new files
 2 files changed, 15 insertions(+)
 create mode 100644 README.md
 create mode 100755 script.sh
 create mode 120000 symlink.txt

这些都表示此提交中新增了一个文件,且该文件的“mode(模式)”为 100644。

Git 的文件模式(mode)是什么?

Git 在内部保存每个文件的三类信息(在 indextree 对象中):

信息含义
mode文件类型 + 权限
SHA-1/Hash文件内容对应的 blob 对象
文件名文件路径名

所以,Git 实际上并不直接保存整个文件,而是保存 “文件内容(blob)+ mode + 文件名” 三元组。

100644 的含义分解

100644 是一个 文件模式(mode),用八进制表示。

文件类型部分(前两位)

模式前缀文件类型
100普通文件(regular file)
120符号链接(symbolic link)
160Git 子模块(gitlink)

权限部分(后三位)

模式含义
644普通文件(可读写、不可执行)
755可执行文件(脚本、二进制等)

实际权限解析(Linux 风格):

6 = rw- (所有者可读写)
4 = r-- (组可读)
4 = r-- (其他用户可读)

rw-r--r--

常见几种模式对照表

Git 模式(八进制)类型含义示例
100644普通文件普通非执行文件(rw-r–r–)源代码、配置文件
100755普通文件可执行文件(rwxr-xr-x)脚本、二进制文件
120000符号链接保存符号链接路径Linux 下的 symlink
160000gitlink子模块.gitmodules 指向的 repo
040000目录(tree 对象)存储 tree 引用非实际文件

查看文件在 Git 中的 mode

可以用以下命令查看工作区、索引和历史中对应文件的 mode

查看 index(暂存区):

git ls-files -s

示例输出:

100644 4d1f48a5e9e6b032d212d24ec8594f98639b6f08 0       README.md
100755 05a43e2a77fdab64d45c601944bcdbba05cf8cb1 0       build.sh

第一列即是文件模式。

查看 tree(提交快照):

git ls-tree HEAD

输出:

100644 blob 4d1f48a5e9e6b032d212d24ec8594f98639b6f08    README.md
100755 blob 05a43e2a77fdab64d45c601944bcdbba05cf8cb1    build.sh

Git 为什么只保存两种权限(644 与 755)

  • Git 主要在 跨平台 环境使用。

  • Windows 文件系统不支持 POSIX 权限位(如执行权限)。

  • 为了避免混乱,Git 简化为两种状态:

    • 普通文件(非执行)

    • 可执行文件(执行位)

也就是说,Git 只关心文件是否“可执行”,其余权限位不影响版本内容。

如何修改可执行标志

修改本地文件执行权限:

chmod +x script.sh

再次提交:

git add script.sh
git commit -m "make script executable"

提交后再次查看

mode change 100644 => 100755 script.sh

子模块(160000)与符号链接(120000)

模式类型存储内容备注
160000子模块存储子仓库的 commit ID不保存实际文件
120000符号链接存储符号链接目标路径不保存实际文件内容

示例(符号链接):

ln -s ../config.yml link.yml
git add link.yml
git commit -m "add symlink"
# 显示 create mode 120000 link.yml

Git 是如何在内部保存这些信息的?

Git 提交由多个对象组成:

Commit → Tree → Blob

举例:

commit对象
  ↓
tree对象(保存 mode + 文件名 + blob 引用)
  ├── 100644 README.md → blob(内容)
  ├── 100755 build.sh  → blob(内容)
  └── 040000 src/ → tree(子目录)

也就是说,100644 这一信息实际上存在 tree 对象 中,是提交快照(snapshot)的一部分。

实际操作示例

场景 1:添加普通文件
# 创建普通文本文件
echo "Hello World" > hello.txt

# 查看权限
ls -l hello.txt
# -rw-r--r-- 1 user group 12 Jan 1 10:00 hello.txt

# 添加到 Git 并提交
git add hello.txt
git commit -m "Add hello"
# create mode 100644 hello.txt
场景 2:创建可执行脚本
# 创建脚本并设为可执行
echo '#!/bin/bash\necho "Hello"' > run.sh
chmod +x run.sh

# 查看权限
ls -l run.sh
# -rwxr-xr-x 1 user group 22 Jan 1 10:00 run.sh

# 添加到 Git 并提交
git add run.sh
git commit -m "Add script"
# create mode 100755 run.sh
场景 3:创建符号链接
# 创建符号链接
ln -s target.txt link.txt

# 查看文件类型
ls -l link.txt
# lrwxrwxrwx 1 user group 10 Jan 1 10:00 link.txt -> target.txt

# 添加到 Git 并提交
git add link.txt
git commit -m "Add symlink"
# create mode 120000 link.txt

文件模式的管理

查看 Git 中的文件模式
# 查看 Git 数据库中文件的模式
git ls-tree HEAD
# 100644 blob 89abcde    README.md
# 100755 blob 2345678    script.sh
# 120000 blob 3456789    link.txt
# 040000 tree 4567890    src

# 查看特定提交的文件模式
git ls-tree 1a2b3c4
修改文件模式
# 如果文件模式不正确,可以修改文件权限后重新添加
chmod +x my-script.py
git add my-script.py
git commit -m "Fix file permissions"

# 或者使用 Git 命令更新索引中的文件模式
git update-index --chmod=+x my-script.py
忽略文件权限变化
# 告诉 Git 忽略文件权限变化(在跨平台协作时有用)
git config core.filemode false

# 检查当前配置
git config core.filemode
跨平台注意:
  • Linux/MacGit 尊重文件权限。

  • Windows:权限支持有限,默认 core.fileMode = false,模式变化不会被跟踪。

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值