Git概述 | ① 什么是版本管理系统 | ② Git和SVN的区别 |
③ Git的四个组成部分 | ④ Git中文件的几个状态 | |
⑤ Git中的四类对象 | ||
Git下载安装配置 | ||
Git本地基本操做 | ① 配置「git config」 | ② 获取帮助「git help」 |
③ 建立Git仓库「git init」 | ||
④ 添加文件到暂存区「git add」 | ||
⑤ 让Git不跟踪特定文件「.gitignore文件」 | ||
⑥ 暂存区内容提交到本地仓库「git commit」 | ||
⑦ 查看工做区与暂存区状态「git status」 | ||
⑧ 内容变化,差别对比「git diff」 | ||
⑨ 查看历史提交记录「git log」 | ||
⑩ 查看某个文件的改动记录「git blame」 | ||
⑪ 设置Git命令别名「git config --global alias」 | ||
⑫ 为重要提交打标签「git tag」 | ||
Git文件恢复与版本回退 | ① 文件恢复,未add「git checkout」 | |
② 文件恢复,已add未commit「git reset HEAD」 | ||
③ 版本回退,已commit「git reset --hard」 | ||
④ 查看输入过的指令记录「git reflog」 | ||
⑤ 撤销某次提交「git revert」 | ||
⑥ 查看某次提交的修改内容「git show」 | ||
⑦ 查看分支最新commit的Hash值「git rev-parse」 | ||
⑧ 找回丢失对象的最后一点但愿「git fsck」 | ||
Git本地分支 | ① 分支的概念 | ② 建立其余分支的缘由 |
③ 一个简单的分支管理策略 | ||
④ 分支的建立于切换「git branch」 | ||
⑤ 分支合并「git merge」 VS 「git rebase」 | ||
⑥ 解决合并冲突 | ⑦ 删除分支 | |
⑧ 恢复误删分支「git log --branches」 | ⑨ 分支重命名 | |
⑩ 切换分支时暂存未commit的更改「git stash」 | ||
⑪ 把commit从一个分支挪到另外一个分支「git cherry-pick」 | ||
Git远程仓库 | ① 远程仓库概述 | |
② 本地仓库与远程仓库创建关联「git remote」 | ||
③ 推送本地仓库到远程仓库「git push」 | ||
④ 克隆远程仓库「git clone」 | ||
⑤ 同步远程仓库更新「git fetch」VS「git pull」 | ||
⑥ git push 时的unrelated history问题 | ||
⑦ SSH Key避免每次push重复输入帐号密码 | ||
Git工做流 | ① 集中式工做流 | ② 功能分支工做流 |
③ Gitflow工做流 | ④ Forking工做流 | |
⑤ Pull Request工做流 | ||
其余杂项 | ① 为开源项目贡献代码 | ② SourceTree使用详解 |
了解Git相关的概念,有助于后续命令的掌握~html
VCS(Version Control System),一种用于记录一个或多个文件内容变化 历史,以便未来能对特定版本的历史记录进行查看,更改,备份还原的系统。 能够简单类比为「游戏存档」,打Boss前存下档,没过关,从新读档; 分支剧情,想体验不一样选择触发的不一样剧情,能够存多个档, 想玩哪一个读哪一个。linux
VCS通常分为下述三类:git
使用简单的数据库来记录文件的历史更新差别,好比RCS。github
![]()
用一个服务器来保存全部文件的修订版本,协同工做的人链接这个服务器, 获取或提交文件更新,好比SVN。算法
![]()
这种协同方式有个两个明显的缺点: 1.「须要联网」:同步和推送更新速度受带宽限制,内网还好,外网可能会有点慢了(大文件); 2.「依赖中央服务器」:每一个人的本地只有之前所同步的版本,若是服务器宕(dang)机了,谁都没法获取或提交更新。shell
每一个用户拥有完整的提交历史,支持离线提交更改,查看历史提交记录等。中央服务器更多的只是用做更改合并,同步的工具,好比Git。数据库
![]()
从根本上来讲,Git是一个内存寻址的文件系统,根据文件的Hash值来定位文件。 这个40位的Hash值使用SHA1算法生成,由两部分拼接: header = "<type>" + content.length + "\0" (参数依次为:对象类型,数据字节长度,空字节(用于分隔header与content) hash = sha1(header+content),这里的拼接是二进制级别的拼接,而非字符串拼接。vim
Git和SVN除了上面说的联网需求不一样外,还有「存储差别」:windows
SVN关心:文件内容的具体差别;而Git关心:文件总体是否发生改变。 SVN每次提交记录的是:「哪些文件进行了修改,修改了哪些行的哪些内容」。缓存
![]()
如图,Version 2中记录的是文件A和C的变化,而Version 3中记录文件C的变化,以此类推; 而Git中,并不保存这些先后变换的差别数据,而是保存整个缓存区中的全部文件, 又称快照,「有变化的文件保存,没变化的文件不保存,而是对上次保存的快照作一个连接」, 由于这种不一样的保存方式,使得Git切换分支的速度比SVN快上很多。
![]()
固然SVN也有它的优势,好比「权限控制」,能够设定每一个帐户的读写权限,而Git中 则没有响应的权限控制。至于用哪一个的,仍是看公司要求吧~
简单说下Git的四个组成部分:
接下来讲下这几个部分是如何协同工做的:
工做区与暂存区:工做区更改,经过git add命令能够把更改提交到暂存区; 也能够git checkout命令使用暂存区内容覆盖当前的工做区的内容。
暂存区与本地仓库:能够经过git commit命令把暂存区的内容提交到本地仓库, 每次commit都会生成一个快照,快照使用Hash值编号。能够经过git reset Hash值, 把某个快照还原到暂存区中。
工做区和本地仓库:经过git checkout 快照编号,直接把某个快照还原到工做区中。
本地仓库和远程仓库:能够经过git push命令把commit推送到远程仓库,多人协做的 时候可能还须要进行一些冲突处理;还有经过git clone拉取某个远程仓库的项目到本地, 或经过git fetch拉取远程仓库的最新内容,检查后决定是否合并到本地仓库中。
工做区和远程仓库:这里二者的协做通常是git pull,即把远程主机的最新内容拉取下来后直接合并。
按照大类划分,能够分为两种状态:Tracked(已跟踪)和Untracked(未跟踪), 依据是:「该文件是否已加入版本控制」?
文件状态变化周期流程图:
流程简述:
假设某个项目已加入Git版本控制系统
Untracked
状态;Tracked
状态又或者说 此时这个文件已经被版本控制系统所跟踪,并且他处于Staged
**(暂存)状态;Unmodified
(未修改)状态;Modified
**(修改)状态;在Git系统中有四种类型的对象,几乎全部的Git操做都是在这四种对象上进行的,依次为: Blob
(块)对象,Tree
(树)对象,Commit
(提交)对象,Tag
(标签)对象。 前三者的关系如图所示:
接着咱们来详解的讲解这四类对象:
① 块对象(Blob)
一块二进制数据,「仅存放文件内容」,不包括文件名、权限等信息。Git会根据文件内容计算 出一个Hash值,以这个Hash值做为文件索引保存起来。意味着,相同文件内容的文件,只会保存 一个,即共享同一个Blob对象。可使用:
git hash-object 文件名
来计算文件内容的Hash值。 若是你知道已经添加到Git中的某个文件的hash值,还能够经过git cat-file hash值
来读取数据 对象,可选参数:-p
(查看Git对象内容) ,-t
(查看Git对象类型),示例以下:
② 树对象(Tree)
保存一个或多个块对象的引用,每次commit对应一个树对象,这里生成一个commit, 而后调用**
git ls-tree Hash值
** 查看树对象的内容:
利用上面的 git cat-file -p hash值
来查看blob块的具体内容:
除了保存块对象的引用外,树对象还能够引用「其余树对象」,从而构成一个「目录层次结构」。 新建一个test目录,复制一个1.txt文件到这个路径下,提交一个commit,而后查看树对象的内容:
能够指向了另外一个tree对象,这个tree对象指向另外一个1.txt文件,树对象解决了文件名的问题。 而对于提交的人、时间、说明信息等,咱们还须要经过提交对象进行了解。
③ 提交对象(Commit)
保存树对象的Hash值,父Commit的Hash值,提交做者、时间、说明信息。 一样可使用**
git cat-file
**命令查看commit对象:
④ 标签对象(Tag)
通常会对某次重要的commit加TAG,以示重要,分为两种状况:
.git/refs/tags/标签名
,里面保存TAG对象的引用。这里为咱们上面的两个commit一次打上两种标签,而后看下具体的结果:
sudo apt-get install git
安装便可。brew install git
进行安装。安装完后,使用Git还须要进行环境的配置,配置信息保存在gitconfig文件中,有三种级别:
C:\Program Files\Git\mingw64\etc\gitconfig
, 不一样的系统可能不同,你能够经过:git config -e --system
,底部能够找到配置文件的路径:
C:/Users/当前用户/.gitconfig
, 一样能够采用上面的:git config -e --global
查看配置文件的位置。项目路径/.git/config
配置生效优先级:local > global > system,经常使用命令:
# 配置
git config --global user.name "用户名" # 配置用户名
git config --global user.email "用户邮箱" # 配置邮箱
git config --global core.editor 编辑器 # 配置编辑器,模式使用vi或者vim
# 查看配置
git config --global user.name # 查看配置的用户名
git config --global user.email # 查看配置的邮箱
# 查看全部配置列表
git config --global --list # 查看全局设置相关参数列表
git config --local --list # 查看本地设置相关参数列表
git config --system --list # 查看系统配置参数列表
git config --list # 查看全部Git的配置(全局+本地+系统)
复制代码
除了命令行的方式外,你还能够直接去编辑对应的配置文件。
git help 命令 # 查看某个git命令的介绍,用法
git 命令 --help # 另外一种写法
复制代码
git init 仓库名 # 建立一个新的带Git仓库的项目
git init # 为已存在的项目生成一个Git仓库
复制代码
git add 文件名 # 将工做区的某个文件添加到暂存区。
git add -u # 添加全部被tracked文件中被修改或删除的文件信息到暂存区,不处理untracked的文件
git add -A # 添加全部被tracked文件中被修改或删除的文件信息到暂存区,包括untracked的文件
git add . # 将当前工做区的全部文件都加入暂存区
git add -i # 进入交互界面模式,按需添加文件到缓存区
复制代码
不少人应该没用过交互界面模式,这里演示下用法:
流程简述:
当咱们使用git add命令把未标记的文件添加到缓存区后,Git就会开始跟踪这个文件。 对于一些好比:自动生成的文件,日志,临时编译文件,应用签名文件等,就不必进行跟踪了, 咱们能够编写一个**「.gitignore文件」,把不须要跟踪的文件和文件夹写上,git就不会去 跟踪这些文件了,另外:.gitignore文件与.git文件夹在同级目录下**。
若是不想本身写这个文件,能够到 github.com/github/giti… 选择对应的模板,复制粘贴。 也能够自行编写,支持简化了的真这个表达式(规范与示例模板摘自:Git王者超神之路)
*
: 匹配零个或多个任意字符[abc]
:只匹配括号内中的任意一个字符[0-9]
:表明范围,匹配0-9之间的任何字符?
:匹配任意一个字符**
:匹配任意的中间目录,例如a/*/z能够匹配:a/z,a/b/z,a/b/c/z等模板示例:
# 忽略全部以 .c结尾的文件
*.c
# 可是 stream.c 会被git追踪
!stream.c
# 只忽略当前文件夹下的TODO文件, 不包括其余文件夹下的TODO例如: subdir/TODO
/TODO
# 忽略全部在build文件夹下的文件
build/
# 忽略 doc/notes.txt, 但不包括多层下.txt例如: doc/server/arch.txt
doc/*.txt
# 忽略全部在doc目录下的.pdf文件
doc/**/*.pdf
复制代码
有一点要特别注意!!!!
配置.gitignore只对那些没有添加到版本控制系统的文件生效(未Tracked的文件)!
举个简单的例子:
有A,B两个文件,你先把他两个add了,而后在.gitignore文件中 配置了不跟踪这两个文件,可是你会发现根本不会生效。
git add A
git add B
# 配置不跟踪A和B
git add .gitignore
复制代码
因此,最好的作法就是在项目刚开始的时候,先添加.gitignore文件。 固然,即便是发生了,仍是有解决方法的,能够键入下述命令清除标 记状态,而后先添加.gitignore,再添加文件便可:
git rm -r --cached . # 清除版本控制标记,.表明全部文件,也可指定具体文件
复制代码
另外,若是你用的IDEA系列的代码编辑器,能够安装一个「.ignore」的插件,手动 勾选不须要跟踪的文件,直接生成.gitignore文件。
git commit -m "提交说明" # 将暂存区内容提交到本地仓库
git commit -a -m "提交说明" # 跳过缓存区操做,直接把工做区内容提交到本地仓库
复制代码
若是不加-m “提交说明”,git会让用你让默认编辑器(vi或vim)来编写提交说明。 除此以外,有时可能想修改上次提交的内容:提交说明,修改文件等:
# 合并暂存区和最近的一次commit,生成新的commit并替换掉老的。若是缓存区没内容,
# 利用amend能够修改上次commit的提交说明。
#
# 注:由于amend后生成的commit是一个全新的commit,旧的会被删除,因此别在公共的
# commit上使用amend!切记!!!
git commit --amend
git commit --amend --no-edit # 沿用上次commit的提交说明
复制代码
git status # 查看工做区与暂存区的当前状况
git status -s # 让结果以更简短的形式输出
复制代码
git diff # 工做区与缓存区的差别
git diff 分支名 # 工做区与某分支的差别,远程分支这样写:remotes/origin/分支名
git diff HEAD # 工做区与HEAD指针指向的内容差别
git diff 提交id 文件路径 # 工做区某文件当前版本与历史版本的差别
git diff --stage # 工做区文件与上次提交的差别(1.6 版本前用 --cached)
git diff 版本TAG # 查看从某个版本后都改动内容
git diff 分支A 分支B # 比较从分支A和分支B的差别(也支持比较两个TAG)
git diff 分支A...分支B # 比较两分支在分开后各自的改动
# 注:若是只想统计哪些文件被改动,多少行被改动,能够添加--stat参数
复制代码
git log # 查看全部commit记录(SHA-A校验和,做者名称,邮箱,提交时间,提交说明)
git log -p -次数 # 查看最近多少次的提交记录
git log --stat # 简略显示每次提交的内容更改
git log --name-only # 仅显示已修改的文件清单
git log --name-status # 显示新增,修改,删除的文件清单
git log --oneline # 让提交记录以精简的一行输出
git log –graph –all --online # 图形展现分支的合并历史
git log --author=做者 # 查询做者的提交记录(和grep同时使用要加一个--all--match参数)
git log --grep=过滤信息 # 列出提交信息中包含过滤信息的提交记录
git log -S查询内容 # 和--grep相似,S和查询内容间没有空格
git log fileName # 查看某文件的修改记录,找背锅专用
复制代码
除此以外,还能够经过 –pretty 对提交信息进行定制,好比:
更多规则与定制以下(更多可参见:Viewing the Commit History) format对应的经常使用占位符:(注:做者是指最后一次修改文件的人,提交者是提交该文件的人)
占位符 | 说明 | 占位符 | 说明 |
---|---|---|---|
%H |
提交对象(commit)的完整哈希字串 | %h |
提交对象的简短哈希字串 |
%T |
树对象(tree)的完整哈希字串 | %t |
树对象的简短哈希字串 |
%P |
父对象(parent)的完整哈希字串 | %p |
父对象的简短哈希字串 |
%an |
做者(author)的名字 | %ae |
做者的电子邮件地址 |
%ad |
做者修订日期(能够用 –date= 选项定制格式) | %ar |
按多久之前的方式显示 |
%cn |
提交者(committer)的名字 | %ce |
提交者的电子邮件地址 |
%cd |
提交日期 | %cr |
提交日期,按多久之前的方式显示 |
%s |
提交说明 |
一些其余操做:
选项 | 说明 |
---|---|
-p |
按补丁格式显示每一个更新之间的差别 |
–stat |
显示每次更新的文件修改统计信息(行数) |
–shortstat |
只显示 –stat 中最后的行数修改添加移除统计 |
–name-only |
仅在提交信息后显示已修改的文件清单 |
–name-status |
显示新增、修改、删除的文件清单 |
–abbrev-commit |
仅显示 SHA-1 的前几个字符,而非全部的 40 个字符 |
–relative-date |
使用较短的相对时间显示(好比,“2 weeks ago”) |
–graph |
显示 ASCII 图形表示的分支合并历史 |
–pretty |
格式定制,可选选项有:oneline,short,full,Fullerton和format(后跟指定格式) |
还有一些限制log输出的选项:
选项 | 说明 |
---|---|
-(n) |
仅显示最近的 n 条提交 |
–since, –after |
仅显示指定时间以后的提交。 |
–until, –before |
仅显示指定时间以前的提交。 |
–author |
仅显示指定做者相关的提交。 |
–committer |
仅显示指定提交者相关的提交。 |
–grep |
仅显示含指定关键字的提交 |
-S |
仅显示添加或移除了某个关键字的提交 |
git blame 文件名 # 查看某文件的每一行内容的做者,最新commit和提交时间
复制代码
这里为了演示,先修改一波做者用户名和邮箱,而后往1.txt中新增内容:
Tip:若是你用的IDEA系列的编译器,右键行号,选择Annotate也能够实现一样的效果。如:
在终端使用Git命令的时候,虽然能够经过按两次tab来自动补全。可是有些命令比较经常使用, 每次都要敲完就显得有些繁琐了,能够为这些命令起一个简单的别名,好比: status为st,checkout为co ; commit为ci ; branch为br等,设置示例以下:
git config --global alias.st status
复制代码
别名的设置保存在git的配置文件中:
对于某些提交,咱们能够为它打上Tag,表示此次提交很重要, 好比为一些正式发布大版本的 commit,打上TAG,当某个版本出问题了,经过TAG能够快速找到这次提交对应的Hash值, 直接切换到这次版本的代码去查找问题,比起一个个commit找省事多了。
Git中的标签分为两种:轻量级标签 和 附加标签,命令以下:
git tag 标记内容 # 轻量级标签
git tag -a 标记内容 -m "附加信息" # 附加标签
复制代码
若是想为以前某次commit打TAG,能够找出这次提交的Hash值,添加-a选项,示例以下:
git tag -a 标记内容 版本id # 好比:git tag -a v1.1 bcfed96
复制代码
另外,git push 的时候默认不会把标签推送到远程仓库,若是想把标签页推送到远程仓库,能够:
git push origin 标记内容 # 推送某标签到远程仓库
git push origin --tags # 删除全部本地仓库中不存在的TAG
复制代码
除此以外还有下述常规操做:
git checkout -b 分支名 标记内容 # 新建分支的时候打上TAG
git show 标记内容 # 查看标签对应的信息
git tag -d 标记内容 # 删除本地TAG
git push origin --delete tag 标记内容 # 删除远程TAG
复制代码
若是在工做区直接删除已经被Git Tracked的文件,暂存区中还会存在此文件:
Git告诉你,工做区的文件被删除了,你有两种可选操做:「删除缓存区文件」 或 「恢复被删文件」:
# 删除暂存区中的文件:
git rm 文件名
git commit -m "提交说明"
# 误删恢复文件(用暂存区的文件覆盖工做区的文件)
git checkout -- 文件名
# Tip:git rm 等价于 git rm --cached 文件名 + rm 文件名
# 务必注意:git checkout会抛弃当前工做区的更改!!!不可恢复!!!务必当心!!!
复制代码
若是更改已经add到暂存区中,想恢复原状,能够执行下述命令:
git reset HEAD 文件名
git checkout 文件名
复制代码
文件已经commit,想恢复未上次commit的版本或者上上次,能够:
git reset HEAD^ # 恢复成上次提交的版本
git reset HEAD^^ # 恢复成上上次提交的版本,就是多个^,以此类推或用
git reset HEAD~3 # 也能够直接~次数
git reset --hard 版本号 # git log查看到的Hash值,取前七位便可,根据版本号回退
复制代码
reset命令的做用其实就是:重置HEAD指针,让其指向另外一个commit,而这个动做可能会对 缓存区形成影响,举个例子:
原本的分支线:- A - B - C (HEAD, master),git reset B后:- A - B (HEAD, master) 解释:看不到C了,可是他仍是存在的,能够经过git reset C版本号找回,前提是 C没有被Git当作垃圾处理掉(通常是30天)。
reset提供了三个可选参数:
Git会记住你输入的每一个Git指令,好比上面的git reset 切换成一个旧的commit,而后 git log后发现新提交的记录没了,想切换回新的那次commit, 能够先调git reflog 获取新commit的Hash值,而后git reset 回去。
git reflog
复制代码
注:指令记录不会永久保存!Git会定时清理用不到的对象!!!
有时可能咱们想撤销某次提交所作的更改,可使用revert命令
git revert HEAD # 撤销最近的一个提交
git revert 提交的Hash值 # 撤销某次commit
复制代码
注意!!!
不是真的把提交给撤销了,而是生成一个新的提交来覆盖旧的提交,被撤销的提交 和新的提交记录都会保存!!!若是不信的话,你能够再键入git revert HEAD, 会发现被撤销的更改又变回来了。简单点说:「撤销的只是文件变化,提交记录依旧存在」。
git show 提交Hash值 # 查看某次commit的修改内容
复制代码
git rev-parse 分支名 # 查看分支最新commit的Hash值,也能够直接写HEAD
复制代码
由于你的某次误操做致使commit丢失,若是git reflog都找不到,你可使用git fsck,找到丢失 的对象的版本Hash值,而后恢复便可。
git fsck --lost-found
复制代码
分支并非Git对象,和轻量级的TAG对象相似,只包含对commit对象的索引。只是分支更新后, 索引会替换为最新的commit,而TAG对象建立后索引就不在变化。分支文件保存与下述两个路径:
当前项目/.git/refs/heads/
当前项目/.git/refs/remotes/
说到分支,必然会说起HEAD,它指向「当前工做的本地分支」,对应文件:当前项目/.git/HEAD
下面经过示例和图解的方式帮你们理解分支:
如法炮制,提交两次:
从上面的图中不难发现这样的规律:每次commit,master都会向前移动,指向最新提交。 这个时候可能有些童鞋会问:commit之间的箭头哪来的?或者说commit怎么串成一条线的?
答:还记得一开始介绍的commit对象吗?里面有一个parent的值,指向父commit的Hash值。
经过两个常见的场景来体会建立其余分支的必要性:
项目通常都是一步步迭代升级的,有大版本和小版本的更新: 大版本通常是改头换面的更新,好比 UI大改,架构大改,版本是: v2.0.0这样;小版本的更新通常是UI小改,Bug修复优化等,版本是: v2.0.11这样;只有一条master分支,意味着:你的分支线会 很是很是的长,假如你已经发布到了 第二个大版本,而后用户反馈第一个版本有很严重的BUG,这时候想切回第一个版本改BUG, 而后改完BUG切回第二个大版本,想一想也是够呛的。 (PS:可能你说我能够对重要的commit打tag, 而后找到这个tag 切回去,固然也行这里是想告诉你引入其余分支会给你带来的便利)
若是只有一个master分支的话,假如某次commit冲突了,而这个冲突很难解决或者解决不了, 那么整个开发就卡在这里,没法继续向后进行了。
为了解决只有一个master分支引发的问题,能够引入分支管理,最简单的一种策略以下:
在master分支上开辟一个新的develop分支,而后咱们根据功能或者业务,再在develop 分支上另外开辟其余分支,完成分支上的任务后,再将这个分支合并到develop分支上! 而后这个功能分支的任务也到此结束,能够删掉,而当发布正式版后,再把develop分支 合并到master分支上,并打上TAG。
master与develop分支都做为长期分支,而其余建立的分支做为临时性分支! 简述各个分支的划分:
git branch 分支名 # 建立分支
git branch # 查看本地分支
复制代码
咱们在master分支上建立一个develop分支,此时的版本线变成了这样:
此时虽然已经建立了develop分支,可是HEAD仍是指向master,接着咱们来切换分支:
git checkout 分支名 # 切换分支
git checkout -b 分支名 # 建立分支同时切换到这个分支
复制代码
切换到develop后,提交一次,此时的版本线:
再提交一次,而后切换为master分支,此时的版本线:
切换回master后,提交一次,此时的版本线:
行吧,讲到这里,相信各位童鞋对Git中的分支已经有所了解了。
Git中,可使用「git merge」和「git rebase」两个命令来进行分支的合并。
git merge合并分支
合并的方式分为两种:快速合并 和 普通合并,二者的区别在于: 「前者合并后看不出曾经作过合并,然后合并后的历史会有分支记录」 如图所示:
快速合并,默认,快速合并有一个前提:「当前分支的每一个提交都在另外一个分支中」, Git不建立任何新的commit,只是将当前分支指向合并进来的分支。下面演示下快速合并, 执行git reset 切换到第四次commit,而后执行git merge develop合并master分支。
普通合并,添加**–no-ff**参数表示禁用快速合并。
另外有时会有这样的场景:合并的分支中有不少commit记录是无需在分支中体现的,一个commit 就够了。能够借助**--squash**参数来压缩提交,示例以下:
附:git merge的经常使用参数:
git merge -ff # 快速合并,默认参数
git merge -ff-only # 只有快速合并的状况才合并
git merge --no-ff # 不使用快速合并
git merge -n 分支名 # 合并分支,不会在合并后显示合并先后的不一样状态
git merge -stat 分支名 # 合并分支,合并结束后显示合并先后的不一样状态
git merge -e 分支名 # 合并分支,合并前调用编辑器,可自行编写commit
复制代码
Tips: git-merge除了用来合并分支外,拉取远程仓库更新时也可用到(git fetch + git merge)
git reabse合并分支
rebase(衍合,变基),网上不少教程写得很高深莫测,其实并无那么复杂, 只是这种合并会让树整洁,易于跟踪。以上面4中的结果为例,先把master分支 和develop分支重置到最新的commit。
先走一波前面的merge合并方式:
接着再试试rebase合并方式:
Git会把每一个提交都取消掉,并把他们临时保存为补丁,好比通过一些冲突解决,生成新的commit, 旧的commit会被丢弃,还会被git的gc回收,这样的结果就是一条直线的树。
在分支的合并的时候,并非每次都能直接合并的,有时会遇到合并冲突,特别是在多人协做的时候。 出现合并冲突后,须要解决完冲突,才能继续合并。
举个简单的例子,A和B在master分支上开辟出两个分支来完成相关的功能, A作完了,把本身的分支合并到master分支,此时master分支向前移动了几回commit, 接着B也完成了他的功能,想把本身分支合并到master分支,若是改动的文件和和A改动 的文件相同的话,此时就会合并失败,而后须要处理完冲突,才可以继续合并!
接下来咱们来简单的模拟合并冲突,先来试试merge:
merge分支后处理冲突
如图,合并完A分支后合并B分支出现了冲突,接着键入:git status查看冲突的文件:
能够看到未合并的两个文件,1.txt和2.txt,打开其中一个文件:
<<< 和 >>>包裹着的就是冲突内容,保留本身想要的内容,处理完后删掉<<<和>>>,修改完后:
2.txt文件也如法炮制,接着add,而后commit便可,合并结束。
rebase分支后处理冲突
如图,A合并成功,在合并B的时候,出现了合并冲突,有三个可选的操做:
git rebase --continue # 处理完冲突后,继续处理下一个补丁
git rebase --abort # 放弃全部的冲突处理,恢复rebase前的状况
git rebase --skip # 跳过当前的补丁,处理下一个补丁,不建议使用,补丁部分的commit会丢失!
复制代码
键入git status查看冲突文件:
接着处理1.txt文件中的冲突,解决完成后,先键入git add,接着键入git rebase --continue 处理下一个冲突:
处理接下来的冲突,直到没有冲突为止:
能够看到使用rebase合并,最后的分支线是一条直线。另外,使用rebase合并中途出差错, 可使用git rebase --abort恢复rebase前的状态。
合并完的分支,基本没什么用了,可使用下述命令删除:
git branch -d 分支名 # 删除分支,分支上有未提交更改是不能删除的
git branch -D 分支名 # 强行删除分支,尽管这个分支上有未提交的更改
复制代码
两步:找出被删分支最新的commit的Hash值,而后恢复分支:
git log --branches="被删除的分支名" # 找到被删分支最新的commit版本号
git branch 分支名 版本号(前七位便可) # 恢复被删分支
复制代码
有时咱们可能在某个分支上正编写着代码,而后有一些突发的状况,须要 咱们暂时切换到 其余分支上,好比要紧急修复bug,或者切换分支给同事 review代码,此时若是直接切换 分支是会提示切换失败的,由于这个分支 上作的更改尚未提交,你能够直接add后commit, 而后再切换,不过咱们习惯写完某个功能再提交,咱们想:
先暂存这个分支上的改动,切去其余分支上搞完事,而后回来继续 继续在以前的改动上写代码。
那么可使用:
git stash # 保存当前的改动
复制代码
而后放心的切换分支,而后再切换回来,接着使用:
git stash apply # 恢复保存改动
复制代码
另外有一点必定要注意!!!能够stash多个改动!!若是你切换到另外一个分支 又stash了,而后切换回来stash apply是恢复成另外一个分支的stash!!! 若是你这样stash了屡次的话,我建议你先键入:
git stash list # 查看stash列表
复制代码
找到本身想恢复的那个
好比这里恢复的应该是master上的stash,可使用下述命令进行恢复:
git stash apply stash@{1}
复制代码
git branch -m 老分支名 新分支名 # 分支重命名
复制代码
有时咱们可能须要把某个分支上的一次commit放到另外一个分支上,此时可使用git cherry-pick, 好比下面这样两个分支:
master分支:A -> B -> C feature分支:a -> b
如今想把feature分支上的b,放到master的后,能够这样操做:
在实际开发过程当中,基本都是团队协做的形式进行,即多人一块儿负责同一个项目,那如何共享同一份代码并进行管理呢?能够用到「Git远程仓库」。能够本身搭建,或选择专业的代码托管平台,好比:Github,Git@OSC,GitCafe,GitLab,coding.net,gitc,BitBucket,Geakit,Douban CODE 等。固然,若是有条件的话,确定是本身搭建的爽一些,可控,还能够作一些订制(集成编译,机器人提醒等),简单点的能够试试「Gogs」,可玩性更高的能够试试「GitLab」。
在Github上新建了一个项目仓库,会生成对应的仓库连接,如:
键入下述命令进行关联:
git remote add origin 远程仓库地址
复制代码
接着可键入下述命令查看关联状况:
git remote # 列出已经存在的远程分支
git remote -v # 查看远程仓库的地址
复制代码
创建完关联后,咱们可使用git push命令把本地更改推送到远程仓库
git push -u origin master
复制代码
-u参数:做为第一次提交使用,做用是把本地master分支和远程master分支关联起来(设置默认远程主机),后续提交不须要这个参数!
另外,若是想修改远程仓库地址,可经过下述命令:
# 直接修改远程仓库地址
git remote set-url origin 远程仓库地址
# 也能够先删除origin后再添加
git remote rm origin # 删除仓库关联
git remote add origin 远程仓库地址 # 添加仓库关联
复制代码
你还能够直接修改「.git文件夹中的config文件」,直接替换圈住位置内容便可:
还有一点:「origin」并非固定的东西,只是后面「仓库地址的一个别名」!!能够写成其余的东西,而后你也能够设置多个仓库关联,用不一样的别名标志,好比:
git remote add github https://github.com/coder-pig/SimpleTea.git
git remote add osc git@git.oschina.net:coder-pig/SimpleTea.git
复制代码
把项目推送到远程仓库后,其余开发者就能够经过git clone命令把项目克隆到本地
git clone 仓库地址 # 克隆项目到当前文件夹下
git clone 仓库地址 目录名 # 克隆项目到特定目录下
# 注:git clone命令只会创建master分支,若是想克隆特定远程分支,可在克隆后:
git checkout -t origin/dev
# 该命令等同于
git checkout -b dev origin/dev
# 除此以外,还能够:
git fetch origin 远程分支:本地分支 # 会在本地新建分支,但不会自动切换,还需checkout
git branch --set-upstream 本地分支 远程分支 # 创建本地分支与远程分支的连接
复制代码
获取远程仓库更新的方法有两种:fetch 和 pull,简要讲解下二者的区别:
仅仅只是从远处服务器获取到最新版本到本地,假如你不去合并(merge),本地工做空间是不会发生变化的!好比:在Github上建立一个README.md文件,而后调 git fetch 去获取远程仓库的更新。
一步到位,pull = fetch + merge,好比:一样修改Github上的README.md 文件,而后git pull 同步远程仓库的更新:
区别显而易见,使用git fetch会更安全一些,毕竟merge的时候,查看更新的状况,再决定是否进行合并。
在Github建立新项目后,在repo处建立了README.md或其余文件,而后关联本地仓库,push时会报错: Push rejected: Push to origin/master was rejected,而后提示你pull一下,当你pull时又会报错: 「refusing to merge unrelated histories」,缘由是两个仓库不一样致使的,可以使用下述命令解决:
git pull origin master --allow-unrelated-histories
复制代码
除此以外,你还能够粗暴一点,直接用本地仓库「强制覆盖远程仓库」,可是 慎用!!!若是出问题了,只能看下其余人的电脑中是否有原始的本地仓库进行还原!!!
git push -f origin # 慎用!!!
复制代码
私有项目,使用Https协议pull或push,都须要验证帐号和密码,有点繁琐,若是想避免这种重复输入的状况,能够考虑使用SSH协议。SSH,Secureshell(安全外壳协议),专为远程登录会话与其余网络服务提供安全性的协议,而SSH传输的数据是能够通过压缩的,能够加快传输的速度,出于安全性与速度,优先考虑使用SSH协议,而SSH的安全验证规则又分为基于密码和基于密钥两种!这里使用的是第二种,即在本地建立一对密钥「公钥(id_rsa.pub)和私钥(id_rsa)」而后把公钥内容贴到远程仓库设置中的ssh keys中,从而创建本地与远程的认证关系。配置SSH Key的流程以下:
执行完ssh-keygen那个指令后,后面依次要你输入文件名 直接回车 → 会生成两个默认的秘钥文件,接着提示输入密码, 直接回车 → 若是这里你输入密码了的话,那么push的时候你仍是须要输入密码,接着又输多一次密码 直接回车 → 出现最下面的这串东西就说明ssh key已经建立成功了! 接着能够用编辑器打开id_rsa.pub文件或者键入下述命令复制内容:
clip <id_rsa.pub
复制代码
打开Github,点击头像,选择:Settings,而后点击左侧SSH Keys,而后New SSH Key
而后Github会给你发来一个提示建立了一个新ssh key的邮件,无视就好,接下来咱们能够键入:
**ssh -T git@github.com**
复制代码
而后若是上面设置过密码则须要输入密码,不然直接输入yes而后一直按回车就好!,最后出现Hi xxx那句话就说明ssh key配置成功了!
其余远程仓库配置方法相似,另外若是想一个电脑管理多个SSH-Key,可移步至:
关于Git工做流,Github上有一篇图文并茂写得很好的文章,就不细说了,只是简单介绍下,更多详情可见:《Git Workflows and Tutorials》
相似于SVN,不过只有一条master分支,而后一群人就在这条分支上嗨,好比有小A和小B:
和集中式分部流相比只是分支再不是只有master,而是根据功能开辟新的分支而已,示例以下:
- 1.小A要开发新功能,git branch -b new-feature 开辟新分支
注:这里的仓库管理者是拥有仓库管理权限的人
其实就是功能分支工做流作了一些规范而已,大概流程参见上面「一个简单的分支管理策略」
分布式工做流,每一个开发者都拥有本身独立的仓库,为开源项目贡献代码经常使用,把项目fork到本身的远程仓库,完成相应更改,而后pull request到源仓库,源仓库管理者能够决定是否合并。
和Forking工做流相似,Pull Requests是Bitbucket上方便开发者之间协做的功能
你能够Clone别人的开源项目,在看别人代码的时候,以为做者某些地方写得很差,写错,或者你有更好的想法,在本地修改后,想把修改push推送到开源项目上,是没法直接Push推送更改的。参与开源项目的方式有两种:
方法一: 是让做者把你加为写做者,添加协做者流程: 点击仓库的Settings → Collaborators 而后输入想添加的人的用户名或者邮箱,点击 添加便可。
方法二: 点击Fork按钮,把这个项目fork到本身的帐号下,而后Clone到本地,而后作你想作的修改,commit提交,而后push到本身帐号里的仓库,而后打开开源项目,点击,而后新建一个「pull request」,接着设置本身的仓库为源仓库,设置源分支,目标仓库与目标分支,而后还有pull request的标题和描述信息,填写完毕后,肯定。 这个时候开源项目的做者就会收到一个pullrequest的请求,由他来进行审核,做者审查完代码以为没问题的话,他能够点击一下merge按钮便可将这个pull request合并到本身的项目中,假如做者发现了你代码中还有些bug,他能够经过Pull Request跟你说明,要修复了xxBUG才容许合并,那么你再修改下BUG,提交,更改后的提交会进入Pull Request,而后做者再审核这样!
Tips:o(╯□╰)o假如做者不关闭或者merge你的这个Pull Request,你能够一直commit骚扰主项目…
命令行虽酷炫可装逼,可是有时用图形化工具仍是能提升很多效率的,安利个巨好用的Git图形化工具SourceTree,官网下载地址:www.sourcetreeapp.com/,网上教程满天飞,笔者也不粘贴复制了,找到个写得还行的,有兴趣可移步至:《用SourceTree轻松Git项目图解》。
后面有新的会更新,待续...
参考文献与更多Git学习资料: