标签 Vim 下的文章

也谈使用分支

近期在为一个新项目作版本库规划,并策划一些即将应用于该项目的版本控制和发布流程的Rules。借此机会我也花上一些时间对我们之前的版本控制和发布流程进行一下反思,也翻看了一些书籍(比如《版本控制之道-使用subversion》、社区自由图书《Subversion与版本控制》等),了解一下Best Practice是什么样子的,同时也纠正一下我之前理解不正确的地方。

我们这些年来一直在使用CVS/Subversion这些版本控制系统进行源码管理,但若干年来似乎没有什么改善,仍然在走最初的路子进行版本控制和发布,这也导致我们仍旧挣扎在繁乱的版本库中。未经精心策划过的版本库组织及发布管理流程还在一定程度上降低了团队的工作效率。

年初曾协调开发团队和测试团队共同制定了一些版本管理和发布流程的改善措施,目前收到了一定效果,但是我个人觉得与最佳状态相比,我们还是有差距的。差距主要还是体现在对VCS系统的几个概念拿捏的不够好,比如说分支。

现在的程序员已经足够幸福了,前人经过几十年的实践给我们留下了版本控制的一些最佳实践,你在介绍CVS、SVN或GIT等VCS工具的书籍中都能读得到。之前没有静下心来好好读一读这类的书籍真是有些遗憾,走了许多弯路。

版本控制工具应用的核心应该是对分支、标签的把握,这其实也是最难的,它可不仅仅是简单命令的使用,而是体现着对软件开发流程的整体把握能力。

如何使用分支等高级概念取决于你对软件生命周期的规划,如果一个软件产品极其简单,简单到发布一个版本后就不需要增加feature和Bugfix了,那你根本不需要用到分支,甚至标签都可省去。但实际生活中我们面对的是一个复杂的世界,多数软件产品都会有一个较长的生命周期,在这个生命周期里程序员们需要为它增添feature,为它Fix bug,这样分支等概念就能派上用场,可以帮助你更佳清晰的管理版本以及提升开发和测试的效率。

Subversion引入了“TTB(Trunk-Tags-Branches)”的实践,其核心也是分支的使用。资料中多将分支分成两种类型,一种称为”Release Branch(发行分支)”,另外一种是”Feature Branch(特性分支)”。我们还是结合一个例子来说明两者的差异吧。

一般一个项目在初始阶段都会有相应的负责人对产品的版本进行规划,比如对与testproj这个项目来说,规划如下:
version 1.0.0
    – feature#1
    – feature#2
    – feature#3
version 2.0.0
    – feature#4
    – feature#5
version 3.0.0
    – feature#6
    – feature#7

规划版本号采用标准的[major.minor.revision]格式。版本库组织采用标准的TTB形式,即:
testproj(root)
    – trunk
    – branches
    – tags
之后大家就一起在trunk上编码,测试,提交,乐此不疲^_^。直到有一天开发leader宣布feature#1~#3编码+UT的工作基本OK了。下面的工作将会出现岔路口,一组人要继续在trunk上开发feature#4~#5,另外一组人则要继续执著于feature#1~#3直到1.0.0版本最终发布。分支此时该出面解决问题了。不过如何创建分支、创建什么类型的分支,出现了两种意见:
1、创建”发布分支,Release Branch”
                                        ——- Rel_1.0.0(Tag)
                                        |
         —–rel_branch_1.0 —————————->
         |
—————————————————————> trunk
如上图所示,在leader宣布feature#1~#3基本OK时创建了一个分支rel_branch_1.0,一组人将转移到这个分支上开发,另外一组将继续在thunk上开发后续features。QA对rel_branch_1.0进行严酷的测试,测试完毕后打Tag发布Rel_1.0.0正式版本。也就是说发布版是基于Branch的,且该branch会与trunk长时间并行开发一段时间(甚至可能很长),并得到足够支持。比如后续在trunk上或其他分支上发现的bug需要merge到该分支上,并在适当时刻发布补丁版本,比如Rel_1.0.1等。甚至可继续在此分支上继续增加新Feature,形成1.1.0等发布版本。

2、创建”特性分支,Feature Branch”

         ———Feature_branch_2.0 ——-
         |                                              |
—————————————————————> trunk
                 |
                 ——- Rel_1.0.0(Tag)
特性分支与发布分支恰相反,如上图所示,一组人继续在trunk上fix bug,直到Rel_1.0.0版本发布;其他人建立一个Feature_branch_2.0的特性分支,在该分支上开发新Features,并定期从trunk上merge trunk上的bugfix。Feature_branch_2.0开发完毕后,将分支代码merge回trunk,之后QA对trunk的最新代码进行测试,直到完毕后发布Rel_2.0.0,此后Feature_branch_2.0这个特性分支已经不再需要,它的生命周期也到此为止,可删除之。

两种类型分支如何选择呢?其实实际软件开发中两种类型分支是你中有我,我中有你的。如果从宏观来看,其实更多还要看你的产品的特点和规划,如果类似GCC这样的产品,“发布分支”必不可少。比如我曾用过的GCC的版本有2.95.x、3.2.x、3.3.x、3.4.x等,GCC对于每组[major,minor]都要有一个“发布分支”,至少是用来fix bug的,在3.2上发现的bug要同步回2.95,发布2.95新补丁,除非GNU宣布对GCC 2.95不再支持。如果一个产品的bugfix不需要回溯到以前的版本,而是一直采用trunk上的最新版本,那特性分支则更适合,这样也避免了在N多发布分支上频繁merge代码的情况了。但是特性分支不宜建多,最初我也考虑是否每人建立一个关于自己任务的分支,后来发现这样会给后期merge代码带来诸多不便。

发布分支上如果要新增个性化feature,就好比发布分支变成了”trunk”,后期也可以采用feature分支的形式来开发,保持了”trunk”上代码的稳定、可用;当然如果不太在意这点,你大可直接在该”trunk”上直接开发。

目前我所在的产品线采用的就是发布分支和特性分支结合的方式,不过产品的电信行业背景决定了产品需求多变、客户间需求差异较大,增加了我们的版本发布管理复杂性。单独在该分支上采用“特性分支”的开发流程已经不能满足需要了,因为我们有时候需要回溯到某个早期发布版上fix bug或增加feature,到了在发布分支上再建立发布分支的时候了。但为了减少后期分支数多、在各个分支上同步代码工作量大且较难跟踪管理的情况,我们采取了一些措施。比如我们目前采取的策略是尽量减少版本的个数、减少发布的次数(达到减少发布分支个数的目的),将两个版本之间交付间隔拉长(版本多为我们自己策划),将不同客户的需求转化为产品的通用功能进行统一管理,减少因版本功能差异带来的版本管理上复杂性。当然了也不能少了与客户沟通和协调,说服客户接受将其提出的个性化功能纳入到某个大版本中统一发布。

工具和概念都有了,一切还在于人的规划。简单和清晰是我们应该在版本规划、控制和管理中追求的目标,这样才易于理解、易于执行,减少不必要的工作,毕竟不是所有人都是这方面的专家。

重新定制VIM

这周五工作状态实在不好,也许是工作得有些疲劳的缘故。没有了心思工作,那莫不如利用这些时间读读书。在存储电子书的目录中左翻翻右看看,发现了那本久违了的中文版VIM手册,我决定索性打开温习一下,拣一拣那些已经生疏了的但却极其实用的命令。

下班前400多页的手册居然被我走马观花的浏览完了,其间将遇到的觉得实用的且以前不知道的或不常用的命令记录了下来,一共有50多项,其中不乏令我大呼过瘾的能显著提升工作效率的命令^_^。

VIM自从使用以来一直也未曾系统的挖掘过,之前在plugin上下过一些工夫,比如taglist,cscope等。特别是VIM的定制文件vimrc没有系统的整理过,这次重温VIM手册,让我有了重新定制VIM的想法。

定制VIM,即编制适合你自己的.vimrc文件,这个文件不需要你一切都从头开始,网上流传着很多极具参考价值的资料,比如说比较出名的被网友戏称为"史上最强vim配置文件"的amix vimrc,你完全可以照猫画虎的全部搬过来直接使用,但最好是认真读一读其中的内容,了解每一项配置背后的原理,不懂的就对比一下manual,这样印象也深刻些。

我重新定制的vimrc基本上就是对amix vimrc的裁剪,所以这里也没什么值得列出来的内容。不过对其中的一些配置我倒想在这里说道说道:

1、的使用
在定义映射时,可使用标识,这类似一个trick,在映射转换时被变量mapleader的值替换掉。mapleader是一个特殊变量,如果mapleader没有被显示赋值,其值默认为'\';

如: nmap w :w!这个键映射,如果之前没有显式给mapleader变量赋值,那么在normal模式下,敲入'\w'即是执行将当前更新写入文件。如果之前设定let mapleader = ",",那保存文件的命令就变成了',w'。

2、vimrc更新自动生效
autocmd bufwritepost *vimrc* source ~/.vimrc
这是一个比较实用的配置,可让你即时看到修改配置后的效果,比如修改colorscheme。

3、编码选项的设定
公司的服务器环境设定的内码都是GBK,这样我们的代码源文件的内码也就都是GBK;但是Ubuntu默认内码是UTF-8,如果不做任何设置用VIM打开这些文件,势必会导致文件中的中文字符显示为乱码。关于VIM字符编码的问题曾经在一篇文章中分析过,这里不再细说。用下面的设置可以解决上面提到的问题:
" 自动识别文件的编码格式, 打开已有文件时起作用
set fileencodings=GBK,UTF-8,gb18030,ucs-bom,cp936
" 标识源文件中的数据的内码格式
set fileencoding=GBK
" 标识vim buffer中的数据编码格式
set encoding=UTF-8
这样配置有一个小问题就是对于新建的文件强制设定了采用GBK的内码。

4、比较实用的设置
map / "NORMAL模式下这个命令用起来很舒服

以下是在visual模式下自动对块文字区域加(),{},[]等的命令
vnoremap $1 `>a)`<i(
vnoremap $2 `>a]`<i[
vnoremap $3 `>a}`<i{
vnoremap $$ `>a"`<i"
vnoremap $q `>a'`<i'
vnoremap $e `>a"`<i"

source $VIMRUNTIME/ftplugin/man.vim "将光标停留在你想查Manual的Word上,normal模式下敲入'\K'即可自动查找这个word的Manual。

VIM太强大,里面存在太多的技巧和奇妙的命令,VIM manual也是常看常新,Ubuntu里的其他编辑器已经都让我卸载了,以让自己更加专注于VIM^_^。

如发现本站页面被黑,比如:挂载广告、挖矿等恶意代码,请朋友们及时联系我。十分感谢! Go语言第一课 Go语言精进之路1 Go语言精进之路2 Go语言编程指南
商务合作请联系bigwhite.cn AT aliyun.com

欢迎使用邮件订阅我的博客

输入邮箱订阅本站,只要有新文章发布,就会第一时间发送邮件通知你哦!

这里是 Tony Bai的个人Blog,欢迎访问、订阅和留言! 订阅Feed请点击上面图片

如果您觉得这里的文章对您有帮助,请扫描上方二维码进行捐赠 ,加油后的Tony Bai将会为您呈现更多精彩的文章,谢谢!

如果您希望通过微信捐赠,请用微信客户端扫描下方赞赏码:

如果您希望通过比特币或以太币捐赠,可以扫描下方二维码:

比特币:

以太币:

如果您喜欢通过微信浏览本站内容,可以扫描下方二维码,订阅本站官方微信订阅号“iamtonybai”;点击二维码,可直达本人官方微博主页^_^:
本站Powered by Digital Ocean VPS。
选择Digital Ocean VPS主机,即可获得10美元现金充值,可 免费使用两个月哟! 著名主机提供商Linode 10$优惠码:linode10,在 这里注册即可免费获 得。阿里云推荐码: 1WFZ0V立享9折!


View Tony Bai's profile on LinkedIn
DigitalOcean Referral Badge

文章

评论

  • 正在加载...

分类

标签

归档



View My Stats