关于
我的项目
相关阅读
无关联文章
热度排行
- [转] 宫崎骏用动漫教给我们的人生哲理,每一句都能说到心里! - (日期:[八月 24, 2013] 点击:[53,284])
- Google 网页爬虫报告无法连接站点解决办法 - (日期:[七月 20, 2014] 点击:[38,641])
- 架设Tiny Tiny RSS(TTRSS)阅读器,找回Google Reader! - (日期:[九月 27, 2013] 点击:[27,770])
- SkyDrive、DropBox和Google Drive三大公有云存储服务对比 - (日期:[六月 25, 2013] 点击:[25,579])
- 升级到至强E5440后,与i5 CPU笔记本性能对比 - (日期:[二月 18, 2014] 点击:[23,724])
- 公钥私钥加密解密数字证书数字签名详解 - (日期:[四月 19, 2014] 点击:[22,963])
- 本站建站技术合集 - (日期:[九月 20, 2013] 点击:[22,500])
- 使用OpenerDNS解决无法访问Google的问题 - (日期:[七月 5, 2014] 点击:[21,793])
- WordPress博客添加“返回顶部”按钮 - (日期:[七月 14, 2013] 点击:[21,210])
- Linux文件系统基础之inode和dentry - (日期:[三月 13, 2015] 点击:[20,170])
- 云存储中的HTTP鉴权算法分析 - (日期:[二月 7, 2014] 点击:[18,642])
- 存储基础知识之——磁盘阵列原理及操作实战 - (日期:[二月 9, 2014] 点击:[17,497])
- 精选37条强大的常用linux shell命令组合 - (日期:[九月 4, 2013] 点击:[17,429])
- DNS原理、架构和配置详解 - (日期:[九月 6, 2013] 点击:[16,807])
- Netty和Jetty的Java NIO 网络框架模型分析 - (日期:[七月 13, 2013] 点击:[16,336])
- CoreOS 初识之安装 - (日期:[十一月 16, 2014] 点击:[16,171])
- Windows与Linux文件系统互访的几种方法 - (日期:[八月 21, 2014] 点击:[15,734])
- Dijkstra算法求解最短路径分析 - (日期:[七月 12, 2014] 点击:[14,927])
- NAS解决方案实现多媒体文件共享播放 - (日期:[十二月 21, 2014] 点击:[13,924])
- 简介 - (日期:[九月 1, 2012] 点击:[13,761])
- 如何编程实现 2 + 2 = 5? - (日期:[六月 2, 2014] 点击:[13,270])
- 搭建了一个iNews程序 - (日期:[十月 15, 2013] 点击:[13,242])
- 2014年9月曝出的Bash ShellShock漏洞简析 - (日期:[九月 26, 2014] 点击:[13,140])
- 彻底解决WordPress博客垃圾评论的问题 - (日期:[八月 5, 2013] 点击:[13,091])
- 如何使用1M的内存排序100万个8位数 - (日期:[三月 27, 2014] 点击:[12,555])
- 全部日志列表 - (日期:[十一月 11, 2012] 点击:[12,335])
- 关于回调函数和this指针探讨 - (日期:[八月 24, 2014] 点击:[12,211])
- 给定一个long型常量,其值为x,给定long型变量a,要求a & x 的取值集合 - (日期:[九月 8, 2012] 点击:[11,708])
- WordPress建站必备实用插件 - (日期:[八月 7, 2014] 点击:[11,361])
- 开源好用的电子书管理服务Talebook(Calibre网络版)安装使用指南 - (日期:[四月 23, 2022] 点击:[11,291])
分类目录
文章归档
- 2024年十二月 (1)
- 2024年四月 (1)
- 2024年二月 (1)
- 2023年九月 (1)
- 2023年一月 (1)
- 2022年十月 (1)
- 2022年八月 (2)
- 2022年四月 (1)
- 2022年三月 (1)
- 2021年十二月 (2)
- 2021年十月 (2)
- 2021年九月 (1)
- 2021年八月 (1)
- 2021年五月 (1)
- 2021年三月 (2)
- 2021年一月 (2)
- 2020年十二月 (5)
- 2020年十一月 (2)
- 2020年十月 (2)
- 2020年九月 (1)
- 2020年八月 (5)
- 2020年七月 (2)
- 2019年九月 (1)
- 2018年八月 (1)
- 2018年七月 (1)
- 2018年六月 (1)
- 2018年五月 (1)
- 2018年三月 (1)
- 2018年二月 (1)
- 2018年一月 (2)
- 2017年十二月 (3)
- 2017年十月 (4)
- 2017年九月 (1)
- 2017年七月 (1)
- 2017年六月 (1)
- 2016年十二月 (1)
- 2016年十月 (1)
- 2016年九月 (1)
- 2016年七月 (2)
- 2016年六月 (1)
- 2016年二月 (3)
- 2015年十二月 (3)
- 2015年十一月 (2)
- 2015年十月 (1)
- 2015年八月 (2)
- 2015年七月 (4)
- 2015年六月 (1)
- 2015年三月 (2)
- 2015年二月 (1)
- 2015年一月 (4)
- 2014年十二月 (2)
- 2014年十一月 (2)
- 2014年十月 (5)
- 2014年九月 (8)
- 2014年八月 (11)
- 2014年七月 (17)
- 2014年六月 (7)
- 2014年五月 (15)
- 2014年四月 (16)
- 2014年三月 (14)
- 2014年二月 (5)
- 2013年十二月 (5)
- 2013年十一月 (3)
- 2013年十月 (13)
- 2013年九月 (13)
- 2013年八月 (13)
- 2013年七月 (9)
- 2013年六月 (8)
- 2013年五月 (1)
- 2013年三月 (3)
- 2013年一月 (1)
- 2012年十一月 (1)
- 2012年九月 (12)
- 2012年八月 (3)
- 2011年二月 (1)
- 2009年三月 (1)
- 2009年二月 (1)
- 2008年十一月 (1)
- 2008年六月 (1)
- 2008年四月 (1)
- 2008年三月 (1)
一些Git的基本概念和用法(rebase、HEAD、reset、master、remote、pull、push、force等)
由于之前在一个相对封闭且各方面的IT流程都相对完善的公司工作,公司内部总是习惯使用非常成熟的IT工具,不会很快引入外部的新的更高效的生产工具,因此对一些业界已经普及使用的IT工具接触总是会慢很多。除了工作时间,也没有太多机会接触外部的东西,对git的了解比较晚。本人也是从完全不懂Git(只会从github下载master分支的zip包来看源码^_^),到会用pull、push,再后面也会一些稍微高级一点的用法,如rebase、reset、合并或者修改已有commits等。在这个边学边用的过程中也遇到过一些困惑和曲解,这里系统的整理一下一些常用方法和涉及的概念,希望对新接触这个的Coder能有所帮助。
一、复制或者克隆一个库
一般而言,我们从下载或者复制一个库开始。在github等平台上,使用Fork这个动作复制远端的库到自己名下,复制出来的库跟源库之间有派生关系,可以提交MergeRequest(github上面称为PullRequest,个人觉得这个叫法让人费解,容易产生歧义。)合并代码到源库,MR审核通过后即合入,因此一般正式的版本库中是不能轻易直接push commits的。
要下载远端的代码到本地,则需要在本地git clone远端代码库。git clone可以指定版本下载。clone的过程即在本地建立了一个本地git库。同样本地的这个库与远端的库也是有派生关系,共祖先。Fork和clone后,使得一个本地库可以跟多个远端库有继承关系,因此一个本地库可以配置多个源,这个源也就是remote,每个源有一个容易记的名字,还有其地址。地址又有两种方式,https和ssh,可根据需要选用。
二、提交修改目标库
在本地,先切换到需要修改代码的分支,使用git checkout命令,可以将当前工作的代码切换到指定的分支上面。本地编辑代码后,使用git commit命令生成一条提交记录,该提交记录在本地的分支上面,如果需要同步到远端,那么需要push到远端代码库。到此为止一共提到了3个git库:源库,fork到自己名下的远端库,本地库。任意两个库之间需要增量同步代码都需要MR或者本地库可以push到远端。 通常远端库有多人在合入代码时,如果本地的修改不是基于远端当前最新版本修改的,那么push会报错,提示需要先pull远端的库到本地。此时最常见的是,执行一次pull命令,如果远端的修改与本地的修改不冲突,或者git能自动解决冲突(复杂情况下并不能保证业务逻辑不出问题。),那么此时就顺利将远端比本端多的commits合并到本地了,这个过程其实跟svn等管理工具原理是完全一样的。如果不能自动合并,那么需要手工解决冲突,此时在额外产生的解决conflict的commit中会记录两边的修改。提交后,再push即可。值得注意的是,无论是否有手动解决冲突,都会产生一条merge commit,并且每一次git pull都会产生一条,无论是否有实质性的commit合入。大多数情况下,我们掌握了git checkout、git commit、git pull、git push就够日常开发用了。
三、git rebase的用法
有时候,刚才提到的merge commit并不友好,并且是不需要的,希望代码历史有一个清晰的合入过程,那么git也提供了另一个思路解决问题。即git rebase,rebase的思想,顾名思义,就是将当前分支以目标分支为基,重新组织本地多出来的那些commits,让本地的修改永远保持在整个版本树的最上层,即always commit at the top。git的做法并不是简单的将本地的所有commits合并在一起,而是相当于在目标分支上面一一“重放”每个commit,默认情况下所有commit的时间信息会被重写,但是提交人和提交message等不会重写。rebase带来的一个问题是,可能解决冲突的次数会增多,每一个commit可能都需要解决一次。如果commits过多,那么不适合这种方式,或许直接pull远端是更好的选择,所以本地的commit不要太多,rebase要勤做,在rebase的过程中合并多余的commits。rebase之后,如果原来的本地分支已经push到远端,那么新的本地分支则跟远端的分支不在有完整的继承关系,而是从某个点就分叉了,此时可以选择将新的本地分支push到一个全新的远程分支,基于新的分支来工作。也可以考虑push -force原分支,注意push force等于强制覆盖远端分支,远端分支上面的记录将丢失。如果对rebase过程不是特别了解,或者对解决冲突的过程中的修改不放心,那么建议备份远端分支和rebase之前的本地分支。
四、高级用法,合并commits,修改指定commit
上面介绍了rebase的基本思想和用法,其实基于rebase,还有更多高级用法。比如,修改一个已经提交的commit。rebase的过程中允许交互,在开始rebase之前,git会自动分析出哪些commit是比目标端多的,一一列出,给用户选择每个commit的处理方式,有三种选择:
- 保留 pick p
- 编辑 edit e
- 合并到下一条 squash s
用的最多的是保留和合并,编辑的时候一般很少。还有一种是skip,标准的流程里面并没有此项,在git tortoise界面上提供了此选项,也非常有用。如果选择的是编辑,则在rebase的过程中,当处理到该commit的时候,会给出修改代码的机会。此时需要先将代码重置到父版本,然后将本commit的修改直接置位未提交状态,使用git reset HEAD^即可。其中Head代表当前的最新版本,^代表父版本。^和~的含义可以参考:链接。修改好后,提交commit,此时即可重新生成所有的commit信息,相当于对原commit进行了修改。