热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

git深入理解(九):git分支管理与应用

此处以github平台为例。branch查看分支的情况,前面带*号的就是当前分支gitbranch创建分支gitbranch分支名删除分支gitbranch-

此处以github平台为例。

branch

查看分支的情况,前面带*号的就是当前分支
git branch

创建分支
git branch 分支名

删除分支
git branch -d 分支名

checkout

切换当前分支到指定分支
git checkout 分支名

创建分支并切换到创建的分支
git checkout -b 分支名

merge

合并某分支的内容到当前分支
git merge 分支名

如果两个分支同时进行了同一个文件的修改和提交,在merge时就会产生冲突,首先要手动打开文件解决冲突,再提交。

查看分支合并图
git log --graph

红色和绿色的虚线分别代表两个分支。

git merge 的原理:
我们知道分支文件中存储的是当前分支的最新一次提交的commitID,也就是版本号,而每一个版本号对应的 objects 文件中都存储着 parent 版本号(首次提交没有 parent),以此将版本串起来,每一个分支都有自己的串。

比如将分支A合并到分支B,其实就意味着使分支B和分支A的内容完全一致,那么要达到这个目的最快的方式就是将分支A中的最新的commitID复制到分支B的文件中,这样分支B也就拥有了分支A的串,就达到了合并的效果。

如果在合并的时候出现了冲突,需要手动编辑解决冲突,然后提交一个commit,但是分支A上还是之前的内容。

创建 b2 分支
git checkout -b b2
修改 t1.txt 文件,并提交回到 master 分支,修改 t1.txt,并提交分别查看 log
$ git log -3 --pretty=oneline
2a09fbe5794df384dab0cf0a95751a969a5ae057 (HEAD -> master) yy
4052f8d1f053d8b486673b5e1c79c3720ff2316f del
0d3d0d430b2fb9f5788cde925319f718eaddd61d add$ git log b2 -3 --pretty=oneline
837a61a1e32de4bf25ced7ff6b8dc2820968100c (b2) xx
4052f8d1f053d8b486673b5e1c79c3720ff2316f del
0d3d0d430b2fb9f5788cde925319f718eaddd61d add可见是在 4052f8d1f053d8b486673b5e1c79c3720ff2316f 时产生的分支,之后各自有一个提交。试图将 b2 合并到 master
$ git merge b2
Auto-merging t1.txt
CONFLICT (content): Merge conflict in t1.txt
Automatic merge failed; fix conflicts and then commit the result.手动修复 master 冲突文件,然后提交
$ git add .
$ git commit -m 'fix'
[master 19bf978] fix再次查看 log
$ git log -3 --pretty=oneline
19bf97852267733011f5f4bf5793d4a22224f71b (HEAD -> master) fix
2a09fbe5794df384dab0cf0a95751a969a5ae057 yy
837a61a1e32de4bf25ced7ff6b8dc2820968100c (b2) xx$ git log b2 -3 --pretty=oneline
837a61a1e32de4bf25ced7ff6b8dc2820968100c (b2) xx
4052f8d1f053d8b486673b5e1c79c3720ff2316f del
0d3d0d430b2fb9f5788cde925319f718eaddd61d add可以看到分支 b2 并没有什么变化,而分支 master 变化较大,首先 b2 的 837a61a1e3 插入到了 master 的 2a09fbe579 的前面去了
(那是因为 b2 比 master 先提交的);其次 master 多出了一个新提交 19bf978522。此时,对于 b2,如果不需要了则可以删掉这个本地分支;如果还有用那么就应该使 b2 同步,防止以后出现差错。
$ git checkout b2
$ git merge master
Updating 837a61a..19bf978
Fast-forwardt1.txt | 2 +-1 file changed, 1 insertion(+), 1 deletion(-)将 master 同步到 b2,并不会有冲突发生,那是因为 master 最新的提交晚于 b2 最新的提交,Git 认为可以直接覆盖。所以,当你准备
将“早期”代码合并到“晚期”代码中时才可能出现冲突,反之则不会。最后再查看一次 log
$ git log master -6 --pretty=oneline
19bf97852267733011f5f4bf5793d4a22224f71b (HEAD -> b2, master) fix
2a09fbe5794df384dab0cf0a95751a969a5ae057 yy
837a61a1e32de4bf25ced7ff6b8dc2820968100c xx
4052f8d1f053d8b486673b5e1c79c3720ff2316f del
0d3d0d430b2fb9f5788cde925319f718eaddd61d add
71243b890c66751d8b32d547f0a0f6150775903c rm$ git log b2 -6 --pretty=oneline
19bf97852267733011f5f4bf5793d4a22224f71b (HEAD -> b2, master) fix
2a09fbe5794df384dab0cf0a95751a969a5ae057 yy
837a61a1e32de4bf25ced7ff6b8dc2820968100c xx
4052f8d1f053d8b486673b5e1c79c3720ff2316f del
0d3d0d430b2fb9f5788cde925319f718eaddd61d add
71243b890c66751d8b32d547f0a0f6150775903c rm

默认情况下Git会使用Fast-forward的方式来merge,这样的好处是快,当然也有不好的地方,比如

$ git log master -5 --pretty=oneline
f3e1527aa1b7a44d4a34efcd15cbe48f3148ed1e (HEAD -> master) qqq
4a573a96f86a404fe22745ffd71c5341f636438d (b5) 555
fd0ef1ad99851b0871b6544eab1c5dd051c63882 444
069ec4544846898724d7c8cf8de6f8e0dcaaf408 merge from b3
31d8a05af468dc14dd5d1cfbdd8e8b6ca3b3db0b 333现在你可以清楚的看到 555 是从 b5 merge 来的,但是如果我把 b5 分支删掉了
$ git branch -d b5
Deleted branch b5 (was 4a573a9).再次查看,你就不知道 555 是来自 merge,因此 fast-forward 模式会丢掉分支信息。
$ git log master -5 --pretty=oneline
f3e1527aa1b7a44d4a34efcd15cbe48f3148ed1e (HEAD -> master) qqq
4a573a96f86a404fe22745ffd71c5341f636438d 555
fd0ef1ad99851b0871b6544eab1c5dd051c63882 444
069ec4544846898724d7c8cf8de6f8e0dcaaf408 merge from b3
31d8a05af468dc14dd5d1cfbdd8e8b6ca3b3db0b 333鉴于此,在 merge 的时候加上 --no-ff 参数即可避免掉这个问题,他会产生一个新的commit。
$ git merge --no-ff b3 -m 'merge from b3'

操作示例

当前目录有 aaa.txt, bbb.txt ,且是干净状态。
当前为master分支。

创建分支
git branch test-branch

切换分支
git checkout test-branch

当前目录具有和master分支一样的结构,可见 test-branch 分支会继承当时创建它的分支的状态。

编辑文件
vi ccc.txt
提交
git add .
git commit -m ‘add ccc’
该分支第一次提交
git push -u origin test-branch
在Gitlab服务器查看我们推送的分支,它会自动创建分支 test-branch

此时 test-branch 分支有ccc.txt文件,而master分支是没有的。

切换到master
git checkout master

目录下没有ccc.txt文件。

合并 test-branch 分支到master分支,首先要pull到最新版本。
git pull origin master
git merge test-branch
git push origin master
在Gitlab服务器查看master

删除所创建的分支
git push origin --delete test-branch
git branch
发现 test-branch 依然存在本地,而远程仓库不存在此分支了。
git branch -d test-branch
git branch
发现 test-branch 已经不存在了

可见git的多分支是共用一个working copy,当前是那个分支就显示那个分支的内容。

push到线上

将新建的分支以及分支代码提交到线上,第一次Push的时候,需要将本地分支和远端origin绑定,使用it push origin test_branch -u,这样远端会新建一个分支并接受此次提交。后面再提交就只需要git push即可

测试合并冲突

git branch test-branch
git checkout test-branch
git push -u origin test-branch
修改ccc.txt文件
git add .
git commit -m ‘modify ccc 2’
git push origin test-branch
git checkout master
修改ccc.txt文件
git status
git merge test-branch

error: Your local changes to the following files would be overwritten by merge:ccc.txtPlease commit your changes or stash them before you merge.Aborting

显然,如果有改动未提交,是不让merge的。
git add .
git commit -m ‘modify ccc 3’
git merge test-branch

Auto-merging ccc.txtCONFLICT (content): Merge conflict in ccc.txtAutomatic merge failed; fix conflicts and then commit the result

此时test-branch分支已经合并过来了,只是出现了冲突,打开ccc.txt解决冲突内容,然后提交。
git add .
git commit -m ‘resolve conflict’
git push origin master
git status
git log --graph

之前svn解决玩冲突还需要执行resolved命令,而git则不需要,直接提交即可。

多人开发分支设计

master developtestingtask1task2

master 主干,线上处于此分支,有权限控制
develop 过度分支
testing 测试专用分支
task1 任务1
task2 任务1

1、管理员在master分支创建develop,并负责将develop合并到master。

2、开发的时候在develop分支创建task1,在task1中开发,将task1合并到develop,为了减少冲突应经常将develop合并到task1,将task1合并到develop操作。

3、测试的时候去到测试服务器将develop合并到testing。

4、发布的时候去到线上服务器将develop合并到master。

如何修复线上Bug

比如你正在 task1 上开发一个功能,而且开发了一段时间了,这个时候突然线上报出一个bug,你需要立即修复,这个时候应该先保护好现场(使用 git stash 命令),不然切换分支后你的心该滴血了;然后从 master 创建一个分支bug-100,为什么不能从 develop 创建呢?因为在多人开发模式下 develop 分支上经常会有一些暂时不需要同步到 master 上的提交。

修复好之后将分支 bug-100 的提交 merge 到 master (加上 --no-ff 参数),然后发布到线上,删除 bug-100 分支。

总感觉哪里还少了点啥,是的,你的 develop 分支上同样存在这个bug,而且我是在 bug-100 分支上修复的,不是在 task1 分支,所以如果不修复之,日后将 develop 上的提交同步到 master 上的时候是不是会把这个 bug 也带过去了呢?

既然如此,有人会说,这还不简单,将 master 同步到 develop 不就行了。显然不能这么干,因为 develop 上面还有一些提交没有同步到 master ,开头就提到过,这样做将会是一场悲剧。 又有人说了,那我在 develop 分支再重复修复一遍总行吧,这个当然可以。不过 Git 专门提供了一个cherry-pick命令,让我们能复制一个特定的提交到当前分支:git cherry-pick commitID并且 Git 会替当前分支做一次提交。

此时就可以回到 task1 分支上,恢复之前保存的现场,并且要从 develop 同步一次,然后接着开发了。

关于 git stash 命令可以参考 https://blog.csdn.net/raoxiaoya/article/details/111456361


推荐阅读
  • 本文详细介绍了中央电视台电影频道的节目预告,并通过专业工具分析了其加载方式,确保用户能够获取最准确的电视节目信息。 ... [详细]
  • 本文详细介绍了 Apache Jena 库中的 Txn.executeWrite 方法,通过多个实际代码示例展示了其在不同场景下的应用,帮助开发者更好地理解和使用该方法。 ... [详细]
  • 假设您当前位于主分支(版本C4),但发现某些功能已损坏,而这些功能在之前的版本C1时还能正常工作。如何将从C1到C4的所有提交复制到一个新的特性分支中,同时将主分支回滚至C1,以便能够安全地进行版本控制和持续集成部署? ... [详细]
  • Windows服务与数据库交互问题解析
    本文探讨了在Windows 10(64位)环境下开发的Windows服务,旨在定期向本地MS SQL Server (v.11)插入记录。尽管服务已成功安装并运行,但记录并未正确插入。我们将详细分析可能的原因及解决方案。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 本文详细介绍了如何构建一个高效的UI管理系统,集中处理UI页面的打开、关闭、层级管理和页面跳转等问题。通过UIManager统一管理外部切换逻辑,实现功能逻辑分散化和代码复用,支持多人协作开发。 ... [详细]
  • 本文探讨了 Objective-C 中的一些重要语法特性,包括 goto 语句、块(block)的使用、访问修饰符以及属性管理等。通过实例代码和详细解释,帮助开发者更好地理解和应用这些特性。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 深入了解 Windows 窗体中的 SplitContainer 控件
    SplitContainer 控件是 Windows 窗体中的一种复合控件,由两个可调整大小的面板和一个可移动的拆分条组成。本文将详细介绍其功能、属性以及如何通过编程方式创建复杂的用户界面。 ... [详细]
  • 本文探讨了在地理信息系统中,如何通过图层数据获取任意两条道路的交叉点坐标及其名称。文中详细介绍了实现方法和相关技术细节。 ... [详细]
  • 在 Flutter 开发过程中,开发者经常会遇到 Widget 构造函数中的可选参数 Key。对于初学者来说,理解 Key 的作用和使用场景可能是一个挑战。本文将详细探讨 Key 的概念及其应用场景,并通过实例帮助你更好地掌握这一重要工具。 ... [详细]
  • 深入理解Redis的数据结构与对象系统
    本文详细探讨了Redis中的数据结构和对象系统的实现,包括字符串、列表、集合、哈希表和有序集合等五种核心对象类型,以及它们所使用的底层数据结构。通过分析源码和相关文献,帮助读者更好地理解Redis的设计原理。 ... [详细]
  • 本文详细介绍了 SVN 中的 switch 命令及其使用方法,包括如何切换工作副本到新的 URL 和如何处理版本库迁移等情况。 ... [详细]
  • 深入理解 SQL 视图、存储过程与事务
    本文详细介绍了SQL中的视图、存储过程和事务的概念及应用。视图为用户提供了一种灵活的数据查询方式,存储过程则封装了复杂的SQL逻辑,而事务确保了数据库操作的完整性和一致性。 ... [详细]
  • XNA 3.0 游戏编程:从 XML 文件加载数据
    本文介绍如何在 XNA 3.0 游戏项目中从 XML 文件加载数据。我们将探讨如何将 XML 数据序列化为二进制文件,并通过内容管道加载到游戏中。此外,还会涉及自定义类型读取器和写入器的实现。 ... [详细]
author-avatar
Sadness_小妖_619
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有