Git裸仓库和本地仓库

通常我们会用 git init 命令来将我们所在的目录转换为一个 Git 本地仓库或者初始化一个新的空仓库。

image-20220409091857475

1、用法

  • 将当前目录转换为一个本地仓库
git init

这个命令执行后会在本地生成一个 .git 的文件夹,用来追踪仓库的所有变更。效果如下:

image-20220409084837739

  • 指定某个目录成为本地仓库
git init <repo>

这个命令执行后, 将创建一个名为repo且只包含 .git 子文件夹的空目录。效果如下:

image-20220409085121136

  • 指定某个目录成为中心仓库(裸仓库)
git init --bare <repo.git> 

这个命令执行后,将在本地创建一个名为 repo 的文件夹, 里面包含着 Git 的基本目录, 我们一般会将这个文件夹命名为后面加 .git 的形式,如 repo.git (这也是为什么我们从 GitHub clone 仓库的时候,地址都是 xxx.git 这样的形式的原因)。效果如下:

image-20220409085437726

详细说一下使用 --bare 参数的含义,使用 --bare 参数初始化的仓库,我们一般称之为裸仓库, 因为这样创建的仓库并不包含 工作区 , 也就是说,我们并不能在这个目录下执行我们一般使用的 Git 命令。

2、对比

我们来对比一下直接使用 git init 创建的仓库和加了 --bare 参数的两个仓库。 我们直接看两个仓库的的 config 文件中的内容:

  • 直接 git init 创建的仓库:
$ cat ../repo/.git/config
[core]
    repositoryformatversion = 0
    filemode = true
    bare = false
    logallrefupdates = true
    ignorecase = true
    precomposeunicode = true
  • 加了 --bare 创建的裸仓库:
$ cat ../repo_bare.git/config
[core]
    repositoryformatversion = 0
    filemode = true
    bare = true
    ignorecase = true
    precomposeunicode = true

可以看到最直观的差异在于 bare 配置项是否为 true 。此外不加 --bare 创建的本地仓库配置中有一项 logallrefupdates = true , 作用根据名字就可以看出来, 记录所有的 ref (引用) 更新, 关于 ref 的部分之后有时间可以再写,这个配置可以理解为是 Git 的一道防线。

3、功能差异

我们可以使用最简单的例子演示一下。

(1)创建本地仓库和裸仓库
# 直接创建本地仓库
hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test
$ git init repo_local
Initialized empty Git repository in C:/Users/hg/Desktop/test/repo_local/.git/

# 创建裸仓库
hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test
$ git init --bare repo_bare
Initialized empty Git repository in C:/Users/hg/Desktop/test/repo_bare/


(2)分别 clone 两个仓库
hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test
$ git clone repo_local c1
Cloning into 'c1'...
warning: You appear to have cloned an empty repository.
done.

hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test
$ git clone repo_bare c2
Cloning into 'c2'...
warning: You appear to have cloned an empty repository.
done.

(3)分别到2个仓库进行模拟提交并push
hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test
$ cd c1/

hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test/c1 (master)
$ touch test

hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test/c1 (master)
$ git add -A

hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test/c1 (master)
$ git commit -m"test commit"
[master (root-commit) eca8cdb] test commit
 1 file changed, 0 insertions(+), 0 deletions(-)
 create mode 100644 test

hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test/c1 (master)
$ git push origin master
Counting objects: 3, done.
Writing objects: 100% (3/3), 209 bytes | 52.00 KiB/s, done.
Total 3 (delta 0), reused 0 (delta 0)
remote: error: refusing to update checked out branch: refs/heads/master
remote: error: By default, updating the current branch in a non-bare repository
remote: is denied, because it will make the index and work tree inconsistent
remote: with what you pushed, and will require 'git reset --hard' to match
remote: the work tree to HEAD.
remote:
remote: You can set the 'receive.denyCurrentBranch' configuration variable
remote: to 'ignore' or 'warn' in the remote repository to allow pushing into
remote: its current branch; however, this is not recommended unless you
remote: arranged to update its work tree to match what you pushed in some
remote: other way.
remote:
remote: To squelch this message and still keep the default behaviour, set
remote: 'receive.denyCurrentBranch' configuration variable to 'refuse'.
To C:/Users/hg/Desktop/test/repo_local
 ! [remote rejected] master -> master (branch is currently checked out)
error: failed to push some refs to 'C:/Users/hg/Desktop/test/repo_local'


# 进入 c2 仓库重复执行
hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test
$ cd c2/

hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test/c2 (master)
$ touch test

hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test/c2 (master)
$ git add -A

hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test/c2 (master)
$ git commit -m"test commit"
[master (root-commit) 2cbc92c] test commit
 1 file changed, 0 insertions(+), 0 deletions(-)
 create mode 100644 test

hg@LAPTOP-G8TUFE0T MINGW64 ~/Desktop/test/c2 (master)
$ git push origin master
Counting objects: 3, done.
Writing objects: 100% (3/3), 209 bytes | 104.00 KiB/s, done.
Total 3 (delta 0), reused 0 (delta 0)
To C:/Users/hg/Desktop/test/repo_bare
 * [new branch]      master -> master

4、总结

从裸仓库 clone 下来的本地仓库可以进行正常的 push 操作, 但是从一般仓库 clone 下来的本地仓库却不行。 这也正是裸仓库存在的意义。 裸仓库一般情况下是作为远端的中心仓库而存在的,裸仓库可以直接作为服务器仓库供各开发者push、pull数据,实现数据共享和同步,不保存文件,只保存历史提交的版本信息。

参考文章

https://segmentfault.com/a/1190000007686496

Logo

腾讯云面向开发者汇聚海量精品云计算使用和开发经验,营造开放的云计算技术生态圈。

更多推荐