Adjust `makeover` to handle newer AsciiDoc output.
[gitmagic.git] / zh_cn / multiplayer.txt
blob250baf0951042888fd50e819b1f42af199f1f5cd
1 == 多人Git ==
3 我最初在一个私人项目上使用Git,我是那个项目的唯一的开发者。在与Git分布式特性
4 有关的命令中,我只用到了 *pull* 和 *clone*,以此即可在不同地方保持项目同步。
6 后来我想用Git发布我的代码,并且包括其他贡献者的变更。我不得不学习如何管理有来
7 自世界各地的多个开发的项目,幸运的是,这是Git的长处,也可以说是其存在的理由。
9 === 我是谁? ===
11 每个提交都有一个作者姓名和电子信箱,这显示在 *git log* 里。Git使用系统默认
12 设定来填充这些信息。要设定这些信息,键入:
14   $ git config --global user.name "John Doe"
15   $ git config --global user.email johndoe@example.com
17 去掉global选项设定则只对当前仓库生效。
19 === Git在SSH, HTTP上的使用 ===
21 假设你有ssh访问权限,可以访问一个网页服务器,但上面并没有安装Git。尽管比它的
22 原生协议效率低,Git也是可以通过HTTP来进行通信的。
24 在你的帐户下,下载,编译并安装Git,并在你的网页目录里创建一个Git仓库:
26  $ GIT_DIR=proj.git git init
27  $ cd proj.git
28  $ git --bare update-server-info
29  $ cp hooks/post-update.sample hooks/post-update
31 对较老版本的Git,只拷贝还不够,你还要运行:
33  $ chmod a+x hooks/post-update
35 现在你可以通过SSH从随便哪个克隆发布你的最新版本:
37  $ git push web.server:/path/to/proj.git master
39 而随便任何人都可以通过如下命令来取得你的项目:
41  $ git clone http://web.server/proj.git
43 === Git在随便什么上的使用 ===
45 若想不依赖服务器,甚至无需网络连接来同步仓库?需要在紧急时期凑合一下?我们
46 已经看过<<makinghistory, *git fast-export* 和 *git fast-import* 可以转换
47 资源库到一个单一文件以及转回来>>。 我们可以通过任何媒介,来来回回传送这些文件
48 以同步git仓库。但一个更有效率的工具是 *git bundle* 。
50 发送者创建一个“文件包”:
52  $ git bundle create somefile HEAD
54 然后传输这个文件包, +somefile+ ,给某个其他参与者:电子邮件,优盘,一个
55 *xxd* 打印品和一个OCR扫描仪,通过电话读字节,狼烟,等等。接收者通过键入如下命
56 令从文件包获取提交:
58  $ git pull somefile
60 接收者甚至可以在一个空仓库做这个。不考虑大小, +somefile+ 可以包含整个原先
61 git仓库。
63 在较大的项目里,可以通过只打包其他仓库缺少的变更来消除存储浪费。例如,假设提交
64 ``1b6d...''是两个参与者共享的最近提交:
66  $ git bundle create somefile HEAD ^1b6d
68 如果提交频繁,人们可能很容易忘记刚发送了哪个提交。帮助页面建议使用标签解决这个问
69 题。即,在你发了一个文件包后,键入:
71  $ git tag -f lastbundle HEAD
73 并创建较新文件包,使用:
75  $ git bundle create newbundle HEAD ^lastbundle
77 === 补丁:全球货币 ===
79 补丁是变更的文本形式,易于计算机理解,人也类似。补丁可以通吃。你可以给开发者电
80 邮一个补丁,不用管他们用的什么版本控制系统,只要对方可以读电子邮件,他们就能看
81 到你的修改。类似的,在你这边,你只需要一个电子邮件帐号,而不必搭建一个在线的Git
82 仓库。
84 回想一下第一章:
86  $ git diff 1b6d > my.patch
88 输出是一个补丁,可以粘贴到电子邮件里用以讨论。在一个Git仓库,键入:
90  $ git apply < my.patch
92 来打这个补丁。
94 在更正式些的设置里,当作者名字以及或许签名应该被记录下的时候,为过去某一刻生成
95 补丁,键入:
97  $ git format-patch 1b6d
99 结果文件可以给 *git-send-email* 发送,或者手工发送。你也可以指定一个提交范围:
101  $ git format-patch 1b6d..HEAD^^
103 在接收一端,保存邮件到一个文件,然后键入:
105  $ git am < email.txt
107 这就打了补丁并创建了一个提交,其自动包含了诸如作者之类的信息。
109 使用浏览器的邮件客户端,在保存补丁为文件之前,你可能需要建一个按钮,看看邮件内
110 容原来的原始形式。
112 对基于mbox的邮件客户端有些微不同,但如果你在使用的话,你可能是那种能轻易找出
113 答案的那种人,不用读教程。
115 === 对不起,移走了 ===
117 克隆一个仓库后,运行 *git push* 或 *git pull* 将自动推到或从原先的仓库URL拉出
118 新内容。Git如何做这个呢?秘密在于同克隆一起创建的配置选项里面。让我们看一下:
120  $ git config --list
122 选项 +remote.origin.url+ 控制仓库的URL源;``origin'' 是给源仓库的昵称。和
123 ``master'' 分支的惯例一样,我们可以改变或删除这个昵称,但通常没有理由这么做。
125 如果原先仓库移走,我们可以更新URL,通过:
127  $ git config remote.origin.url git://new.url/proj.git
129 选项 +branch.master.merge+ 指定 *git pull* 里的默认远端分支。在初始克隆的时候,
130 它被设为原仓库的当前分支,因此即使原仓库之后挪到一个不同的分支,后来的
131 pull也将忠实地跟着原来的分支。
133 这个选项只使用我们初次克隆的仓库,它的值记录在选项 +branch.master.remote+
134 。如果我们从其他仓库拉入,我们必须显示指定我们想要哪个分支:
136  $ git pull git://example.com/other.git master
138 以上也解释了为什么我们较早一些push和pull的例子没有参数。
140 === 远端分支 ===
142 当你克隆了一个仓库,你也克隆了它的所有分支。你或许没有注意到这点,因为Git将它们
143 隐藏起来了:你必须明确地要求。这使得远端仓库里的分支不至于干扰你的分支,也使
144 Git对初学者稍稍容易些。
146 列出远端分支,使用:
148  $ git branch -r
150 你应该看到类似:
152  origin/HEAD
153  origin/master
154  origin/experimental
156 这显示了远端仓库的分支和HEAD,可以用在常用的Git命令里。例如,假设你已经做了
157 很多提交,并希望和最后取到的版本比较一下。你可以搜索适当的SHA1哈希值,但使用
158 下面命令更容易些:
160  $ git diff origin/HEAD
162 或你可以看看``experimental''分支都有啥:
164  $ git log origin/experimental
166 === 多远端 ===
168 假设另两个开发在同一个项目上工作,我们希望保持两个标签。我们可以同时跟踪多个
169 仓库:
171  $ git remote add other git://example.com/some_repo.git
172  $ git pull other some_branch
174 现在我们已经合并到第二个仓库的一个分支,并且我们已容易访问所有仓库的所有
175 分支。
177  $ git diff origin/experimental^ other/some_branch~5
179 但如果为了不影响自己的工作,我们只想比较他们的变更怎么办呢?换句话说,我们想
180 检查一下他们的分支,又不使他们的变更入侵我们的工作目录。这里我们并不要运行pull
181 命令,而是运行:
183  $ git fetch        # Fetch from origin, the default.
184  $ git fetch other  # Fetch from the second programmer.
186 这只是获取历史。尽管工作目录维持不变,我们可以参考任何仓库的任何分支,使用
187 一个Git命令,因为我们现在有一个本地拷贝。
189 回想一下,在幕后,一个pull是简单地一个 *fetch* 然后 *merge* 。通常,我们
190 *pull* 因为我们想在获取后合并最近提交;这个情况是一个值得注意的例外。
192 关于如何去除远端仓库,如何忽略特定分支等更多,参见 *git help remote* 。
194 === 我的喜好 ===
196 对我手头的项目,我喜欢贡献者去准备仓库,这样我可以从其中拉。一些Git伺服让你
197 点一个按钮,拥有自己的分叉项目。
199 在我获取一个树之后,我运行Git命令去浏览并检查这些变更,理想情况下这些变更组织
200 良好,描述良好。我合并这些变更,也或许做些编辑。直到满意,我才把变更推入主资
201 源库。
203 尽管我不经常收到贡献,我相信这个方法扩展性良好。参见
204 http://torvalds-family.blogspot.com/2009/06/happiness-is-warm-scm.html[ 这篇
205 来自Linus Torvalds的博客 ]
207 呆在Git的世界里比补丁文件稍更方便,因为不用我将补丁转换到Git提交。更进一步,
208 Git处理诸如作者姓名和信箱地址的细节,还有时间和日期,以及要求作者描述他们的提
209 交。