Git读书笔记
git入门
简单写一下读后感。
我看完的参考书籍,建议先看译本,再看英文原版加深理解:
书名 | Pro Git | Version Control with Git |
---|---|---|
作者 | Scott Chacon & Ben Straub | Jon Loeliger & Matthew McCullogh |
下载地址 | 在线查看 |
本地仓库操作
先熟练掌握本地仓库的操作才比较容易理解远程仓库
初次运行git前配置
git config --global user.name "YOUR_NAME"
git config --global user.email "YOUR_EMAIL"
若需要提交GPG签名
git config --global user.signingkey <public GPG key>
# <public GPG key>是你的公钥id,可以从gpg --list-keys查看
若要检查你的git配置,打开~/.gitconfig文件即可。这是我的配置文件:
[user]
name = lixingcong
email = lixingcong@xxx.xxx
signingkey = xxx # GPG公钥id(可选)
[push]
default = simple #使用新版的推送模式
[alias]
ls = log -1 HEAD # 别名,快捷命令
ad = add *
cm = commit -am
lg = log --pretty=format:"%Cred%h%Creset-%cD-%Cgreen%s" --graph
lg1 = log --pretty=oneline
st = status -s
ps = stash
[credential]
helper = cache #凭证缓存
[help]
autocorrect = 15 #自动更正错误
也可以运行查看当前的configuration
git config list
开始工作!
初始化一个文件夹为git仓库:
mk /tmp/test && cd /tmp/test
git init
工作区与暂存区
你在资源管理器里面看到的文件是“工作区”,你通过git add跟踪的文件存放的缓存位置是“暂存区”。
当你的“工作区”与“暂存区”中的文件是一致的,可以称为“干净”的。反之,称为“脏的”。
查看当前的工作区与暂存区命令
git status
这个命令要尽量多使用,可以了解当前目录的情况。
执行status后,绿色的表示已经跟踪并且存放到暂存区(可以随时commit)。红色的表示尚未跟踪,使用
git add FILE-NAME
可以将该文件加入暂存区,生成一个缓存。(准确来说是一个object)
每次修改玩完一个文件,都需要加入暂存区。
看看当前目录下与暂存区有什么区别?
git diff
关于git diff,主要有三种使用方法:
git diff #工作区中文件更新了哪些内容
git diff HEAD #工作区与HEAD中内容对比
git diff --cached #暂存区将要添加到commit的内容
HEAD指针可以理解为最近一次的提交。
提交
通过add跟踪后并使用status确认工作区是干净的,可以进行提交。‘
这里的提交,不是指提交到github上面,而是提交到本地的git仓库。
因为每一个文件夹下都有一个隐藏的.git文件夹,存放提交的历史记录。
git commit -m "YOUR-COMMENT"
若文件file1已经被跟踪,后来又修改file1,可以使用-add选项,不需另外敲add跟踪
git commit -am "YOUR-COMMENT"
若提交后,发现文件A中某句话写错了,修改文件A后,可以追加提交:覆盖上一次提交。建议书写your-comment跟上一次提交一致。
git commit -am "YOUR-COMMENT" --amend
打标签
提交确认无误可以进行打标签,方便后期管理
git tag -a v1.0 -m "YOUR-COMMENT-ON-THIS-TAG"
如果已经有GPG key,可以对tag进行GPG签名。参数-s
git tag -a -s v2.0 -m "SIGN-MY-TAG"
验证tag签名
git -v v2.0
默认情况下,git push 并不会把标签传送到远端服务器上,只有通过显式命令才能分享标签到远端仓库。其命令格式如同推送分支:
git push origin [tagname]
注意:打标签有两种方式,一种称之为轻量级“lightweight”,另一种是正常的“annotated”。不带任何参数运行git tag就是轻量级,加上-a
参数就是annotated
标签 | 命令 | 特点 | 用途 |
---|---|---|---|
lightweight | git tag v1.0 |
不含任何tagger信息,仅仅是将某次commit作为tag | 不建议使用,但可以暂时作为二分查找bisect的临时标签 |
annotated | git tag -a v1.0 |
完整的标签,含tagger所有信息:文字信息,作者,日期,GPG签名值 | 建议使用,作为正常发布的release tag |
摘录一段关于lightweight标签的博客文章
So, how does one create a tag? Just git tag v1.0.0 right? WRONG. You usually don't want to do that. In fact, some suggest that this command does the wrong thing by default. Without arguments, git tag creates a "lightweight" tag that is basically a branch that never moves.
Lightweight tags are still useful though, perhaps for marking a known good (or bad) version, or a bunch of commits you may need to use in the future. Nevertheless, you probably don't want to push these kinds of tags.
上述意思就是不鼓励我们使用lightweight标签,不鼓励push到上游。
创建忽略提交名单
功能就是防止提交上去。忽略.o这种c语言中间文件例:
echo *.o > ./.gitignore
这个ignore文件支持通配符,参考pro-git.pdf
查看提交历史
基本命令:
git log
使用图的方式,比较直观看到当前更新了的分支。
git log --pretty=format:"%s--%h--%ci" --graph
查看上一次的提交
git log -1 HEAD
查看某个文件的提交历史
git log FILE-NAME
当然,可以按时间进行筛选结果,参考pro-git.pdf
使用二分搜索查找bug出处,详细看pro-git.pdf:
git bisect start
git bisect bad
git reset v1.4
git bisect good
查看某个文件在某次Commit的完整内容(比如前2个提交)
git show master~2:src/main.cpp
相对提交名
有时候使用相对提交名,以确定【相对本次提交】的【前某次提交】。
尖角符号^n表示本次提交的来自第n个祖先(纵向,相当于爸爸/叔叔/伯伯)
波浪号~n表示当前提交的前第n代(横向,相当于爸爸/爷爷/曾祖父)
恢复到某次提交
使用reset 命令,能把暂存区恢复到任意一次commit。
使用soft模式,将head指针指向新的提交,暂存区和工作区的内容不变。对当前工作具有最小的影响。
git reset --soft YOUR-COMMIT-SHA1
使用mixed模式,不改变工作区,只恢复暂存区。
因为默认reset就是mixed模式,所以这个命令省略mixed
git reset YOUR-COMMIT-SHA1
若使用--hard具有更大的威力,其破坏性更大,恢复也更彻底。
git reset --hard YOUR-COMMIT-SHA1
对比一下威力:
选项 | soft | mixed | hard |
---|---|---|---|
恢复HEAD指针 | yes | yes | yes |
恢复暂存区 | - | yes | yes |
恢复工作区 | - | - | yes |
目录下文件移动与重命名
不建议使用system-commands,使用git专用的命令:
git rm FILE-NAME
git mv FILE1 FILE2
分支操作
使用分支是git的一大特色,加速开发。
查看分支:
git branch
新建分支:
git branch YOUR-BRANCH-NAME
切换分支:
git checkout YOUR-BRANCH-NAME
合并分支:
git merge ANOTHER-BRANCH-NAME
合并时候,通常会遇到冲突,平和冲突完毕,commit一次即可。
删除本地分支:
git branch -D [BRANCH_NAME]
保存进度:
有时候需要切换紧急的分支,而又不想提交当前的分支。可以将其暂存下来。
如果不暂存,会提示切换分支失败。
保存的地方类似一个栈的数据结构。“入栈”->“出栈”
git stash save
保存完毕,工作区是干净的,可以进行切换分支
恢复进度:
从紧急分支切换回来,需要恢复进度:
git stash pop
变基:参考pro-git.pdf,说白了就是改祖宗。
Youtube上一段几分钟的视频,讲得非常清楚。请自备梯子观看!
干货来了:
花了一小时,我详细地学习了台湾同胞写的一篇非常精彩的博客《Git-rebase 小筆記》,终于知道怎么用rebase了,功能非常强大!!可以实现分支之间的变基、单分支的历史commit的改写(改commit message,commit代码内容,变换commit顺序等),诸多功能。。说不清楚啊!!
基本命令:分支test_rebse的base-commit变为master
git checkout test_rebse
git rebase master
交互式变基:可以单分支的改写历史commit
git rebase -i COMMIT_SHA_ID_TO_CHANGE_START_FROM
删除远程分支:
git push origin :[BRANCH_NAME]
远程仓库操作
主要是在本机实现一下远程的模拟。
制作一个远程仓库
mkdir /tmp/remote_sample && cd /tmp/remote_sample
git init
git remote add origin /tmp/remote_sample
git config receive.denyCurrentBranch warn #允许从别的地方推送
echo hello world > test
git add test && git commit -m "INIT_COMMENT"
就可以简单制作一个服务器了,有趣的是只运行在自己电脑上!
添加一个合作者,如Tony,新建一个Tony分支,以防止推送到master主线:
git checkout -b Tony
当然,可以添加更多的参与者。这个remote仓库只需适时进行多分支合并到master。
多人开发某个分支会导致某些合并问题并造成覆盖。不建议多人公用一个master分支。
master分支作为最终产品发布的主线。
克隆,修改,提交一条龙
模拟一下github的克隆:
mkdir /tmp/local_test && cd /tmp/local_test
git clone /tmp/remote_sample
cd remote_sample
假如我是tony,我的原则:只在自己的分支进行工作。
为了不影响其他人分支,先切换自己的分支
git checkout Tony
想要进行提交,必须先跟上游服务器的HEAD一致。先从远处拉!
git pull
提示Already up-to-date 就把自己的Tony专属分支拿回来了。
若多人开发这个分支,在pull时候,会遇到冲突,也许你克隆仓库后没有及时跟上分支的步伐。
强烈建议一人一个分支。
修改过程模拟:
echo hello world from Tony! > test
按照正常步骤add , commit提交。最后推送给远处:
git push
remote端收到push后合并
当remote仓库管理者得知你推送后,考虑是否把tony跟master分支合并。
模拟一下合并:
cd /tmp/remote_sample
git checkout master
git merge Tony
修正冲突,合并。master分支变为新master,称为Fast-Forward,表示前进了一步
好奇,是否remote的Tony分支跟Tony的文件一致呢?
切换一下Tony分支看看:
git checkout Tony
git status
什么!有未提交的文件!这是什么鬼!
原因很简单,暂存区是tony提交的,在【Tony分支暂存区】test文件内容是Hello world form tony。
而在【remote工作区】的test文件内容是hello world。
因此造成一种“别人都改了,怎么我变化没有呢?”的错觉。实际上缓存区已经是Tony最新的副本了。
HEAD指针还停留在原来地方,落后于Tony分支。
证明一下:查看从工作区准备commit的内容
git diff --cached
可以看到,当前工作区居然想把Tony写的那句from tony删掉了。
因此,要想保持工作区与分支是一致的,使用reset更新当前工作区。
(不可逆的工作区清倒)
git reset --hard SHA1-OF-TONY-LAST-PUSH
Github常用操作
Fork
在以前,“fork”是一个贬义词,指的是某个人使开源项目向不同的方向发展,或者创建一个竞争项目,使得原项目的贡献者分裂。
在 GitHub,“fork”指的是你自己的空间中创建的项目副本,这个副本允许你以一种更开放的方式对其进行修改。
Pull Request
GitHub 设计了一个以合并请求为中心的特殊合作流程。不管你是在一个紧密的团队中使用单独的版本库,或者使用许多的“Fork”来为一个由陌生人组成的国际企业或网络做出贡献,这种合作流程都能应付。
流程通常如下:
- Fork
- 从 master 分支中创建一个新分支
- 提交一些修改来改进项目
- 将这个分支推送push到 GitHub 上
- 创建一个合并请求Pull Request
- 讨论(通过网页或者邮件),根据实际情况继续修改
- 项目的拥有者合并或关闭你的合并请求
:eyes:
Wiki
wiki页面的制作。
有待了解的知识点
列个表,自己还不懂的:
- 以补丁形式的Merge
- Git服务器搭建
未完成:配图片。实际的pul request操作