forked from dcxy/learngit
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathStromNote.txt
335 lines (306 loc) · 17.1 KB
/
StromNote.txt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
摘自廖雪峰的git教程(Thanks Liao)
===============================================================================
创建版本库:
版本库又名仓库repository,其实就是一个可以被git管理的目录
$ mkdir foo 选择一个合适的地方建一个文件夹
$ cd foo 进入这个文件夹
$ git init 通过git init命令把当前目录建立成为git仓库
建好之后这个目录下回多一个.git目录,这个目录就是git用来跟踪管理版本库的
其实也可以使用非控文件夹做git仓库
把文件添加进仓库
在foo目录下新建一个readme.txt
$ git add readme.txt 把文件添加到仓库
$ git commit -m "..." 把文件提交到仓库,并添加说明
可以多次add再统一commit
-------------------------------------------------------------------------------
时光穿梭:
版本回退:
使用git statue命令能实时掌握当前git仓库的状态
$ git statue
如果你想看看文件被改动了哪些内容,就需要使用git diff命令
$ git diff
这里的git diff其实是比较的已经add的和未add的文件之间的差别
git会给我们的工作保存快照,这个快照就是commit,所以我们可以随时回到过去
使用git log命令来查看历史记录
$ git log
使用--pretty=oneline参数可以减少log的输出,使他的每个记录只输出一行
$ git log --pretty=oneline
每条记录都会有一个commit id,这条id就表示了这条记录
git用HEAD表示当前版本,HEAD^表示上一个版本,以此类推HEAD^^...HEAD~100
然后就可以使用如下命令回到过去了
$ git reset --hard HEAD^
$ git reset --hard <commit id> 这里commit id只打前几位能区分开来就好
使用了git reset命令,HEAD就指向指定的版本,再使用log命令将只能获得当前
HEAD版本之前的版本
所以我们要使用下面的命令,git记录了我们的每次操作
$ git reflog
使用这个命令找到比当前版本新的版本的commit id就可以reset回去了
总之:log命令可以查看提交历史,reflog可以查看命令历史
工作区和暂存区:
工作区(work directory)就是我们在电脑里能看到的目录,比复foo文件夹
版本库(repository)工作区有一个隐藏目录.git,它不属于工作区,而是git的版本库
git的版本库里存在了很多东西,其中最重要的就是:
称为stage(或者叫index)的暂存区
git为我们创建的第一个分支master
指向master的指针HEAD
git add就是把文件添加到暂存区
git commit就是把暂存区文件提交到当前分支
因为只有默认创建的master分支,所以commit就是往master分支提交
因为每次commit的都是暂存区的内容,所以没有被add的内容不会被commit
使用如下命令可以比较工作区文件和暂存区文件的异同
$ git diff -- file
撤销修改:
使用如下命令可以撤销工作区的修改
$ git checkout -- file
这条命令实际上是把工作区的文件恢复到暂存区的状态
即使工作区的状态恢复到上次add,或commit时候的状态
使用下面的命令可以撤销add到暂存区的操作
$ git reset HEAD file
这条命令是在提交到暂存区之后又后悔了,使用它使文件恢复到版本库的状态
即使工作区的状态恢复到上次commit时候的状态
删除文件:
通常在工作区中删除文件,直接rm即可
这时在查看git状态时会提示你工作区中哪些文件被删除了
这时,使用如下命令删除版本库中的对应文件
$ git rm foo.txt
$ git commit
同样可以使用如下命令,找回该文件
$ git checkout -- foo.txt
-------------------------------------------------------------------------------
远程仓库:
git的远程仓库,除了可以自己搭建之外,还可以使用github,本机git和github通信
是通过ssh协议加密的,所以需要如下设置(可选):
1.创建ssh key,如果用户主目录下有.ssh文件夹,且其内有id_rsa和id_rsa.pub
两个文件可以跳过第一步了,如果没有打开shell,创建ssh key:
$ ssh-keygen -t rsa -C "your_email@example.com"
2.一切顺利就能创建成功,私钥是id_rsa,公钥是id_rsa..pub,登录github,进入
Account setting,ssh key页面,然后add ssh key,填上任意title,在key文本里
粘贴id_rsa.pub内容,点击啊add key即可完成。
添加远程仓库:
添加远程仓库先在github创建repository,然后可以克隆到本地或关联本地仓库
在git中远程仓库默认叫做origin
关联本地仓库:
回到本地的仓库,运行一下命令
$ git remote add origin git @github.com:somebody/foo.git
把本地仓库内容推送到远程仓库
$ git push -u origin master
这条命令实际上是吧当前分支master推送到远程
因为远程库是空的,所以这次推送加上-u选项,git不但把本地master分支
推送到远程的master分支,还把两者关联起来,以后的推送拉去就能简化命令
以后的推送都可以使用下面的命令来实现
$ git push origin master
从远程库中克隆:
要是从头开发的话,可以先在github创建repository,然后在克隆到本地
使用如下命令来克隆到本地
$ git clone git@github.com:somebody/foo.git
其实git给出的地址不止一个,还可以使用https://github.com/somebody/foo.git
其实git是支持多种协议的,默认的git://使用ssh,但也可以使用https等其他协议
-------------------------------------------------------------------------------
分支管理:
git无论是创建,切换和删除分支,都只需要不到1秒,无论一个文件还是一万个文件
创建与合并分支:
一开始master分支是一条线,git用master指向最新的提交,再用HEAD指向master,就
能确定当前分支,以及当前分支的提交点,每次提交master都会向前移动一小步,
当我们创建新的分支dev时,git就创建一个新的指针,指向与master相同的指针
,再把HEAD指向dev,从现在开始,对工作区的修改和提交就是针对dev分支了,在
dev上的工作完成后,就可以把dev合并到master上,其实就是把master指针指向d
ev当前的提交,这时就可以把dev分支删除了,由此可见,git的分支操作很快
使用如下命令创建并切换到dev分支,-b就表示创建并切换
$ git checkout -b dev
这条命令相当于下面两条
$ git branch dev
$ git checkout dev
然后使用git branch查看当前分支,这条命令会列出所有分支,当前分支标有*号
$ git branch
在dev上的工作和master相同,这时如果想切换会master,使用如下命令
$ git checkout master
现在把dev分支上的内容合并到master,使用如下命令
$ git merge dev
注意这个命令用于将指定分支合并到当前分支
合并时候就可以删除dev分支了,使用如下命令
$ git branch -d dev
git鼓励我们用分支来完成任务,这样不会托慢速度而且更安全
解决冲突:
分支合并也不总是一帆风顺的,有时候master分支和dev分支都在进行开发,两个分支
都进行新的commit,而且提交了相同的文件,但是内容不同,这时就不能快速合并
需要手动修改冲突文件,使用git statue可以查看冲突文件,修改后在提交就好了
使用如下命令能查看分支合并图
$ git log --graph
$ git log --graph --pretty=oneline --abbrev-commit
分支管理策略:
通常合并分支时,如果可能,git会使用Fast forward模式,但是这种模式下删除分支后,
会丢失分支信息,如果强制使用Fast forward模式,git会在merge时生成一个
commit来保存信息
策略:
1.实际开发中,master分支应该是非常稳定的,仅仅用来发布新版本,平时不
能再上面干活
2.干活都在dev分支上,dev分支是不稳定的,仅当发布新版本时候把dev分支合并到
master分支上,然后在master分支上发布1.0版本什么的
3.每个小伙伴都在dev分支上干活,每个人都有自己的分支,时不时地往dev分
支上合并就好了
git分支十分强大,在团队开发中应该充分利用
在合并分支时候,加上--no-ff参数就可以使用普通模式合并,合并后的历史有分支,
能看出曾进做过的合并,而Fast forward合并就看不出曾进做过的合并
--no-ff合并:
普通模式合并,no-ff表示禁用Fast forward
使用如下命令进行no-ff合并,因为合并生成一个commit,所以加-m参数加说明
$ git merge --no-ff -m "merge with no-ff"
bug分支:
git分支管理非常强大,每个bug都可以使用临时分支来解决,合并后就可以删除了
一个很常见的bug修复流程:
首先使用git stash将当前工作区储藏起来,以备日后恢复工作
$ git stash
这时又是干净的工作区了,若要在master上修复bug,就在master上建bug分支
$ git checkout master 切换到master分支
$ git checkout -b issue-101 创建并切换到bug分支issue-101
修复bug完成后,切换到master分支,并完成合并,然后删除bug分支
$ git checkout mater 切换到master分支
$ git merge --no-ff issue-101 -m "merge bug fix 101" 合并分支
$ git branch -d issue-101 删粗bug修复分支
切换会dev分支
$ git checkout dev
要恢复stash起来的工作区,首先使用下面的命令查看stash list
$ git stash list
1.要恢复工作区,方法一
$ git stash apply stash@{0} 恢复stash@{0}存储的工作区
$ git stash drop stash@{0} 删除stash{0}
2.要恢复工作区,方法二
$ git stash pop 恢复工作区,并删除stash
feature分支:
开发时,有时候希望添加一些新功能,但我们不希望实验性的代码把主分支弄坏
可以创建feature分支,在上面开发,完成后再合并,删粗
首先新建一个feature分支
$ git checkout -b feature-newfun 新建并切换到feature分支
开发完成后回到dev分支,准备合并
$ git checkout dev
1.如果一切顺利就可以合并并删除feature分支了
$ git merge --no-ff feature-newfun -m "add new fun"
$ git branch -d feature-newfun
2.如果这时情况有变,要取消新功能,我们就需要销毁feature分支
$ git branch -d feature-newfun
这时会销毁失败,需要强制销毁
$ git branch -D feature-newfun
至此删除feature分支完成
多人协作:
当我们从远程仓库中克隆的时候,git把本地master分支和远程的master分支关联了,
并且远程仓库的默认名称是origin
要查看远程仓库,使用如下命令
$ git remote
或者使用-v选项显示更详细的信息
$ git remote -v
上面会显示可以抓取和推送的git地址,如果没有推送权限将看不到push地址
推送分支:
推送分支时候,要指定本地分支,这样把该分支推送到远程仓库对应的分支上
$ git push orgin master
如果要退送其他分支,比如dev分支
$ git push orgin dev
但是,并不一定要把本地分支都推送到远程分支上,
master是主分支,因此要时时刻刻与远程分支同步
dev是开发分支,团队成员都要在上面工作,因此也要时刻与远程分支同步
bug分支只用在本地修复bug,没必要推送,除非老板要看你每周修复几个bug
feature分支是否推送,取决于你是否和小伙伴合作在上面开发
总之,分支是否往远程推送,取决于自己,完全可以在本地自己藏着玩
通常要合作在上面开发的分支都要推送到远程仓库
抓取分支:
使用下面的命令从远程库抓取,或者克隆本地库
$ git clone repository
从远程库中克隆下来的只有master分支,现在要在dev分支上开发,就要在本地
创建远程organ的dev分支
$ git branch -b dev orgin/dev
这时你的小伙伴就可以在他本地push到远程了
如果你的小伙伴push是碰巧你也push了相同文件呢,这时就会产生冲突
这时推送就会失败,这时就要把远程库的代码抓下来解决冲突
$ git pull
可能pull失败,因为没有本地dev分支没有和远程dev分支关联
这时使用下面的代码来关联
$ git branch --set-upstream branch-name orgin/branch-name
来下来之后,常常会有合并冲突,需要手动解决
解决后再push就好了
多人合作的工作模式常常是这样的:
首先可以使用下面的命令推送自己的修改
$ git push orgin branch-name
如果推送失败是因为远程分支比你本地的新,使用pull合并
$ git pull
如果合并有冲突,需要解决冲突,在本地提交
没有冲突或者解决掉冲突后就可以推送了
$ git push orgin branch-name
如果git提示"no tracking information",则说明本地分支没有和远程库的分支
关联,使用下面的命令来关联
$ git branch --set-upstream branch-name orgin/branch-name
-------------------------------------------------------------------------------
标签管理:
发布一个版本时,我么常常现在版本库中打一个标签,这样就唯一确定了打标签时候的
版本,将来无论什么时候,随时都可以取出打标签的版本,其实就是一个版本快照
git的标签也是一个指针,和分支很像,不同的是,分支可以移动,而标签不能移动
创建标签:
首先切换到要打标签的版本
$ git checkout master
然后使用如下的命令打标签
$ git tag v1.0
可以使用如下的命令查看所有标签
$ git tag
打标签默认是打在最近的commit上的,如果要给历史版本打标签
就要先找到历史的commit id再给它大上
$ git log --pretty=oneline --abbrev-commit
$ git tag v0.9 633834
注意标签不是按照时间排序,而是按照字母排序的
使用git tag <tagname>查看标签信息
还可以创建带说明的标签,使用-a指定标签名,-m指定说明
$ git tag -a v0.1 -m "this is a tag" 632345
还可可以使用-s用私钥签名一个标签
$ git tag -s v0.2 -m "signed version 0.2 released" f234b3
签名使用GPG签名,所以必须先安装了GunPG,并且有gpg密钥对
操作标签:
删除标签:
$ git tag -d v0.1
推送标签到远程
$ git push origin <tagname>
或者可以一次性推送所有本地标签
$ git push origin --tags
删除远程标签,要先删本地的,再删远程的
$ git tag -d v0.1
$ git push origin :refs/tags/v0.1
-------------------------------------------------------------------------------
使用GitHub:
github不仅仅是一个提供git远程仓库的网站,也是一个开源协作社区
通过github,我们既可以参与别人的开源项目,别人也可以参与我们的开源项目
在github上可以fork别人的项目到自己的帐号下,再从自己的帐号下clone
自己的开发要往自己的远程库推送,如果我们改一个bug什么的,可以在github
上发起pull request,然后就看别人是否接受了
-------------------------------------------------------------------------------
自定义git:
git有很多配置选项
如之前见到的
$ git config --global user.name "xxx"
$ git config --global user.email "xxx@example.com"
$ git config --global color.ui true
忽略特殊文件:
我们开发时常常需要把一些不想被git管理的文件添加到git管理的文件夹
比如java编译厚的.class文件
这时在git根目录新建.gitignore文件在里面写忽略规则即可
https://github.com/github/gitignore这个网站已经为我们提供了各种情况
的忽略规则,我们只要适当组合即可
配置别名:
git也有alias就像linux的alias,很好用,特别是不能补全命令的windows上
git config --global alias.st statue
一个有趣的配置:
git config --global alias.lg "log --color --graph
--pretty=format:'%Cred%h%Creset -%C(yellow)%d%Creset %s %Cgreen(%cr)
%C(bold blue)<%an>%Creset' --abbrev-commit"
配置git的时候,--global是对当前用户起作用的,不加就是对当前库起作用
每个仓库的git配置文件都放在.git/config中
每个用户的git配置文件都放在用户主目录的.gitconfig中
搭建git服务器:
1.在linux上搭建git服务器:
$ sudo apt-get install git (sudo yum install git)
$ sudo adduser git
2.创建证书登录,收集所有需要登录用户的公钥,即id_rsa.pub
放入/home/git/.ssh/authorized_keys文件里,一行一个
3.初始化git仓库,切换到自己指定的git目录执行
$ sudo git init -bare sample.git 服务器专用的初始化方式
4.把git目录owner改为git
$ sudo chown git:git sample.git
5.禁用git用户shell登录
在/etc/passwd文件中把git的登录运行文件改为/usr/bin/git-shell
这样git服务器就设置好了,可以在自己的电脑上clone了
$ git clone git@server:/srv/sample.git