<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    jasmine214--love

    只有當你的內心總是充滿快樂、美好的愿望和寧靜時,你才能擁有強壯的體魄和明朗、快樂或者寧靜的面容。
    posts - 731, comments - 60, trackbacks - 0, articles - 0

    一個成功的Git分支模型

    Posted on 2011-02-22 11:22 幻海藍夢 閱讀(1261) 評論(1)  編輯  收藏 所屬分類: 配置管理

    本文中我會展示一種開發(fā)模型,一年前該模型就已經(jīng)被我用在所有的項目中(包括工作中的項目和私有項目),結果是非常成功的。我早就想為此寫點東西,可直到現(xiàn)在才有時間。本文不會講述任何項目的細節(jié),只會涉及到分支策略和發(fā)布管理。

    本文使用Git作為所有源碼的版本控制工具。

    為什么是Git?

    要全面了解Git與其它集中式版本控制系統(tǒng)相比的優(yōu)劣,可以參考這個頁面。 這方面的爭論可謂是硝煙彌漫。作為一個開發(fā)者,所有這些工具中我最鐘情于Git。Git的的確確改變了人們考慮合并及分支的方式。在我之前所處的經(jīng)典 CVS/Subversion世界中,合并/分支總是被認為是有點可怕的事情(“小心合并沖突,丫會惡心到你”),因此你只應偶爾干這種事情。

    但有了Git,這類事情就變得非常簡單,分支及合并甚至被認為是你日常版本控制操作的核心之一。例如,在CVS/Subversion的中,分支及合并往往在后面的章節(jié)才被介紹(針對高級用戶),但在每一本Git的書中,該內容已經(jīng)在前3章中介紹(基礎)。

    簡單及易重復性帶來的好處就是,分支及合并變得不再可怕。版本控制工具本該幫助我們方便的進行和分支及合并操作。

    簡單介紹下工具后,讓我們來看開發(fā)模型。我講介紹的模型本質上只是一組步驟,每個團隊成員都必須遵循這些步驟以形成一個可靠管理的軟件開發(fā)過程。

    去中心化但仍保持中心化

    在這個分支模型中我們使用的,且被證實工作得很好的倉庫配置,其核心是一個中心“真理”倉庫。注意只有該倉庫才被認為是中心庫(由于Git是 DVCS [分布式版本控制系統(tǒng)],在技術層面沒有中心庫這一東西)。之后我們用origin指代該倉庫,因為大多數(shù)Git用戶都熟悉這個名稱。

    每個開發(fā)者都對origin做push和pull操作。不過除了這種中心化的push-pull關系外,每個開發(fā)者還可以從其他開發(fā)者或者小組處 pull變更。例如,可能兩個或更多的開發(fā)者一起開發(fā)一個大的特性,在往origin永久性的push工作代碼之前,他們之間可以執(zhí)行一些去中心化的操 作。在上圖中,分別有Alice和Bob、Alice和David、Clair和David這些小組。

    從技術上來說,這僅僅是Alice定義一個Git remote,名字為bob,指向Bob的倉庫,反過來也一樣。

    主要分支

    此開發(fā)模型的核心主要受現(xiàn)有的模型啟發(fā)。中心倉庫包含了兩個主要分支,這兩個分支的壽命是無限的:

    • master
    • develop

    每個Git用于都應該熟悉origin上的master分支。與master分支平行存在的,是另外一個名為develop的分支。

    我們認為origin/develop分支上的HEAD源碼反映了開發(fā)過程中最新的提交變更。有人會稱之為“集成分支”。該分支是自動化每日構建的代碼源。

    當develop分支上的源碼到達一個穩(wěn)定的狀態(tài)時,就可以發(fā)布版本。所有develop上的變更都應該以某種方式合并回master分支,并且使用發(fā)布版本號打上標簽。稍后我們會討論具體操作細節(jié)。

    因此,每次有變化被合并到master分支時,根據(jù)定義這就是一次新的產(chǎn)品版本發(fā)布。我們趨向于嚴格遵守該規(guī)范,所以理論上來說,每次master有提交時,我們都可以使用一個Git鉤子(hook)腳本來自動構建并部署軟件至產(chǎn)品環(huán)境服務器。

    支持性分支

    緊接著主要分支master和develop,我們的開發(fā)模型使用多種支持性分支來幫助團隊成員間實現(xiàn)并行開發(fā)、追蹤產(chǎn)品特性、準備產(chǎn)品版本發(fā)布、以及快速修復產(chǎn)品問題。與主要分支不同的是,這些分支的壽命是有限的,它們最終都會被刪除。

    我們會用到的分支有這幾類:

    • 特性分支(feature branch)
    • 發(fā)布分支(release branch)
    • 熱補丁分支(hotfix branch)

    上述每種分支都有特定的用途,它們各自關于源自什么分支、合并回什么分支,都有嚴格的規(guī)定。稍后我們逐個進行介紹。

    從技術角度來說,這些分支一點都不“特殊”。分支按照我們對其的使用方式進行分類。技術角度它們都一樣是平常的Git分支。

    特性分支

    可能的分支來源:develop
    必須合并回:develop
    分支命令約定:任何除master, develop, release-*, 或 hotfix-*以外的名稱

    特性分支(有時也被稱作topic分支)是用來為下一發(fā)布版本開發(fā)新特性。當開始開發(fā)一個特性的時候,該特性會成為哪個發(fā)布版本的一部分,往往還不 知道。特性分支的重點是,只要特性還在開發(fā),該分支就會一直存在,不過它最終會被合并回develop分支(將該特性加入到發(fā)布版本中),或者被丟棄(如 果試驗的結果令人失望)。

    特性分支往往只存在于開發(fā)者的倉庫中,而不會出現(xiàn)在origin。

    創(chuàng)建一個特性分支

    開始開發(fā)新特性的時候,從develop分支創(chuàng)建特性分支。

    $ git checkout -b myfeature develop
    Switch to a new branch “myfeature”

    合并完成的特性回develop

    完成的特性應該被合并回develop分支以將特性加入到下一個發(fā)布版本中:

    $ git checkout develop
    Switch to branch ‘develop’
    $ git merge –no-ff myfeature
    Updating ea1b82a..05e9557
    (Summary of changes)
    $ git branch -d myfeature
    Deleted branch myfeature (was 05e9557).
    $ git push origin develop

    上述代碼中的–no-ff標記會使合并永遠創(chuàng)建一個新的commit對象,即使該合并能以fast-forward的方式進行。這么做可以避免丟失特性分支存在的歷史信息,同時也能清晰的展現(xiàn)一組commit一起構成一個特性。比較下面的圖:

    在第2張圖中,已經(jīng)無法一眼從Git歷史中看到哪些commit對象構成了一個特性——你需要閱讀日志以獲得該信息。在這種情況下,回退(revert)整個特性(一組commit)就會比較麻煩,而如果使用了–no-diff就會簡單很多。

    是的,這么做會造成一些(空的)commit對象,但這么做是利大于弊的。

    可惜的是,我沒能找到方法讓–no-diff成為默認的git merge行為參數(shù),但其實應該這么做。

    發(fā)布分支

    可能的分支來源:develop
    必須合并回:develop和master
    分支命名約定:release-*

    發(fā)布分支為準備新的產(chǎn)品版本發(fā)布做支持。它允許你在最后時刻檢查所有的細節(jié)。此外,它還允許你修復小bug以及準備版本發(fā)布的元數(shù)據(jù)(例如版本號,構建日期等等)。在發(fā)布分支做這些事情之后,develop分支就會顯得比較干凈,也方便為下一大版本發(fā)布接受特性。

    從develop分支創(chuàng)建發(fā)布分支的時間通常是develop分支(差不多)能反映新版本所期望狀態(tài)的時候。至少說,這是時候版本發(fā)布所計劃的特性 都已經(jīng)合并回了develop分支。而未來其它版本發(fā)布計劃的特性則不應該合并,它們必須等到當前的版本分支創(chuàng)建好之后才能合并。

    正是在發(fā)布分支創(chuàng)建的時候,對應的版本發(fā)布才獲得一個版本號——不能更早。在該時刻之前,develop分支反映的是“下一版本”的相關變更,但不知道這“下一版本”到底會成為0.3還是1.0,直到發(fā)布分支被創(chuàng)建。版本號是在發(fā)布分支創(chuàng)建時,基于項目版本號規(guī)則確定的。

    創(chuàng)建一個發(fā)布分支

    發(fā)布分支從develop分支創(chuàng)建。例如,假設1.1.5是當前的產(chǎn)品版本,同時我們即將發(fā)布下個版本。develop分支的狀態(tài)已經(jīng)是準備好“下 一版本”發(fā)布了,我們也決定下個版本是1.2(而不是1.1.6或者2.0)。因此我們創(chuàng)建發(fā)布分支,并且為其賦予一個能體現(xiàn)新版本號的名稱:

    $ git checkout -b releases-1.2 develop
    Switched to a new branch “release-1.2”
    $ ./bump-version.sh 1.2
    Files modified successfully. version bumped to 1.2.
    $ git commit -a -m “Bumped version number to 1.2”
    [release-1.2 74d9424] Bumped version number to 1.2
    1 files changed. 1 insertions(+). 1 deletions(-)

    創(chuàng)建新分支并轉到該分支之后,我們設定版本號。這里的bump-version.sh是一個虛構的shell腳本,它修改一些本地工作區(qū)的文件以體現(xiàn)新的版本號。(當然這也可以手動完成——這里只是說要有一些文件變更)接著,提交新版本號。

    新的發(fā)布分支可能存在一段時間,直到該版本明確對外交付。這段時間內,該分支上可能會有一些bug的修復(而不是在develop分支上)。在該分支上添加新特性是嚴格禁止的。新特性必須合并到develop分支,然后等待下一個版本發(fā)布。

    結束一個特性分支

    當特性分支達到一個可以正式發(fā)布的狀態(tài)時,我們就需要執(zhí)行一些操作。首先,將發(fā)布分支合并至master(記住,我們之前定義master分支上的 每一個commit都對應一個新版本)。接著,master分支上的commit必須被打上標簽(tag),以方便將來尋找歷史版本。最后,發(fā)布分支上的 變更需要合并回develop,這樣將來的版本也能包含相關的bug修復。

    前兩步在Git中的操作:

    $ git checkout master
    Switched to branch ‘master’
    $ git merge –no-ff release-1.2
    Merge made by recursive.
    (Summary of changes)
    $ git tag -a 1.2

    現(xiàn)在版本發(fā)布完成了,而且為未來的查閱提供了標簽。

    提醒:你可能同時也會想要用 -s 或者 -u <key> 來對標簽進行簽名。

    為了能保留發(fā)布分支上的變更,我們還需要將分支合并回develop。在Git中:

    $ git checkout develop
    Switched to branch ‘develop’
    $ git merge –no-ff release-1.2
    Merge made by recursive.
    (Summary of changes)

    這一操作可能會導致合并沖突(可能性還很大,因為我們改變了版本號)。如果發(fā)現(xiàn),則修復之并提交。

    現(xiàn)在工作才算真正完成了,最后一步是刪除發(fā)布分支,因為我們已不再需要它:

    $ git branch -d release-1.2
    Deleted branch release-1.2 (was ff452fe).

    熱補丁分支

    可能的分支來源:master
    必須合并回:develop和master
    分支命名約定:hotfix-*

    熱補丁分支和發(fā)布分支十分類似,它的目的也是發(fā)布一個新的產(chǎn)品版本,盡管是不在計劃中的版本發(fā)布。當產(chǎn)品版本發(fā)現(xiàn)未預期的問題的時候,就需要理解著 手處理,這個時候就要用到熱補丁分支。當產(chǎn)品版本的重大bug需要立即解決的時候,我們從對應版本的標簽創(chuàng)建出一個熱補丁分支。

    使用熱補丁分支的主要作用是(develop分支上的)團隊成員可以繼續(xù)工作,而另外的人可以在熱補丁分支上進行快速的產(chǎn)品bug修復。

    創(chuàng)建一個熱補丁分支

    熱補丁分支從master分支創(chuàng)建。例如,假設1.2是當前正在被使用的產(chǎn)品版本,由于一個嚴重的bug,產(chǎn)品引起了很多問題。同時,develop分支還處于不穩(wěn)定狀態(tài),無法發(fā)布新的版本。這時我們可以創(chuàng)建一個熱補丁分支,并在該分支上修復問題:

    $ git checkout -b hotfix-1.2.1 master
    Switched to a new branch “hotfix-1.2.1″
    $ ./bump-version.sh 1.2.1
    Files modified successfully, version bumped to 1.2.1.
    $ git commit -a -m “Bumped version number to 1.2.1″
    [hotfix-1.2.1 41e61bb] Bumped version number to 1.2.1
    1 files changed, 1 insertions(+), 1 deletions(-)

    不要忘了在創(chuàng)建熱補丁分之后設定一個新的版本號!

    然后,修復bug并使用一個或者多個單獨的commit提交。

    $ git commit -m “Fixed severe production problem”
    [hotfix-1.2.1 abbe5d6] Fixed severe production problem
    5 files changed, 32 insertions(+), 17 deletions(-)

    結束一個熱補丁分支

    修復完成后,熱補丁分支需要合并回master,但同時它還需要被合并回develop,這樣相關的修復代碼才會同時被包含在下個版本中。這與我們完成發(fā)布分支很類似。

    首先,更新master分支并打上標簽。

    $ git checkout master
    Switched to branch ‘master’
    $ git merge –no-ff hotfix-1.2.1
    Merge made by recursive.
    (Summary of changes)
    $ git tag -a 1.2.1

    提醒:你可能同時也會想要用 -s 或者 -u <key> 來對標簽進行簽名。

    接著,將修復代碼合并到develop:

    $ git checkout develop
    Switched to branch ‘develop’
    $ git merge –no-ff hotfix-1.2.1
    Merge made by recursive.
    (Summary of changes)

    這里還有個例外情況,如果這個時候有發(fā)布分支存在,熱補丁分支的變更則應該合并至發(fā)布分支,而不是develop。將熱補丁合并到發(fā)布分支,也意味著當發(fā)布分支結束的時候,變更最終會被合并到develop。(如果develop上的開發(fā)工作急需熱補丁并無法等待發(fā)布分支完成,這時你也已經(jīng)可以安全地將熱補丁合并到develop分支。)

    最后,刪除臨時的熱補丁分支:

    $ git branch -d hotfix-1.2.1
    Deleted branch hotfix-1.2.1 (was abbe5d6).

    小結

    雖然這個分支模型中沒有什么特別新鮮的東西,但本文起始處的“全景圖”事實上在我們的項目中起到了非常大的作用。它幫助建立了優(yōu)雅的,易理解的概念模型,使得團隊成員能夠快速建立并理解一個公用的分支和發(fā)布過程。

    我同時也提供了一個該圖對應的高質量PDF版本。你可以打印出來并掛在墻上,隨時參考。

    原文:http://www.juvenxu.com/tag/%E5%88%86%E6%94%AF/

    Feedback

    # re: 一個成功的Git分支模型  回復  更多評論   

    2014-09-29 16:13 by 4332weizi
    我想知道博文里面的圖是用什么工具畫出來的,知道的同志們告知一下,謝謝
    主站蜘蛛池模板: 欧美大尺寸SUV免费| 日韩精品亚洲专区在线观看| 日韩精品内射视频免费观看 | 成人免费网站在线观看| 国产精品亚洲二区在线观看| 亚洲综合色一区二区三区小说| 相泽南亚洲一区二区在线播放| 午夜免费福利片观看| 日本一线a视频免费观看| 亚洲国产成人久久精品影视| 色九月亚洲综合网| 亚洲国产成人久久综合碰| 91嫩草亚洲精品| 18禁在线无遮挡免费观看网站| 国产又大又黑又粗免费视频| 亚洲依依成人精品| 最近中文字幕免费mv视频8| 亚洲第一成年网站视频| 欧洲一级毛片免费| 亚洲91av视频| 最近免费中文字幕中文高清| mm1313亚洲精品无码又大又粗| 一级做受视频免费是看美女| 国产公开免费人成视频| 亚洲精品123区在线观看| 国产a视频精品免费观看| 亚洲综合色婷婷在线观看| 久久久久久国产精品免费免费男同 | 女人18毛片水最多免费观看| 亚洲国产人成在线观看69网站 | 最近免费中文字幕大全视频 | 大地资源中文在线观看免费版| 久久精品国产亚洲| 中文字幕久精品免费视频| 亚洲特级aaaaaa毛片| 国产精品免费一级在线观看| 久久国产精品免费一区| 亚洲日韩中文在线精品第一| a毛片全部免费播放| 成人区精品一区二区不卡亚洲| 亚洲综合精品网站|