先吹一波
git
是一個非常好用的版本管理工具隘道,自從用上了git症歇,感覺其他神馬都是浮云有木有捞烟!
然而,git有時候也會給你出點(diǎn)難題当船。
踩坑過程
git
的一個不足题画,導(dǎo)致我們多出了計劃之外的工作量。
公司項(xiàng)目組件化重構(gòu)德频,原來的項(xiàng)目結(jié)構(gòu)是:
| -- MyProject
| -- gradle
| -- app
| -- build.gradle
| -- src
| -- build.gradle
| -- settings.gradle
組件化之后苍息,是這樣的:
| -- MyProject
| -- gradle
| -- layer_business
| -- app
| -- build.gradle
| -- src
| -- device
| -- build.gradle
| -- src
| -- transaction
| -- user
| -- ...
| -- layer_business_component
| -- comp_share
| -- comp_version
| -- ...
| -- comp_photo
| -- layer_business_base
| -- base_ui
| -- build.gradle
| -- src
| -- ...
| -- build.gradle
| -- settings.gradle
原來代碼全都堆在app module
里面,后來移動放到了./layer_business/app里面壹置,進(jìn)一步又抽取分到各個子module里面竞思。
而在這個過程中,業(yè)務(wù)開發(fā)還在舊的代碼分支上進(jìn)行钞护。由于重構(gòu)和業(yè)務(wù)開發(fā)是不同的兩撥人盖喷,且考慮到此重構(gòu)階段代碼改動較大,而業(yè)務(wù)開發(fā)又時間緊迫难咕,所以在不同的分支進(jìn)行课梳,計劃重構(gòu)階段性目標(biāo)完成后合并業(yè)務(wù)分支的代碼進(jìn)來。
這樣就出現(xiàn)了這種情況:
兩個分支余佃,一個分支對某些文件進(jìn)行了修改暮刃,而另一個分支對同樣這些文件移動了位置并且做了修改
那么這樣還能不能成功合并?
不能爆土。
合并之后的結(jié)果是:
| -- MyProject
| -- gradle
| -- app
| -- build.gradle
| -- src
| -- layer_business
| -- app
| -- build.gradle
| -- src
| -- device
| -- build.gradle
| -- src
| -- transaction
| -- user
| -- ...
| -- layer_business_component
| -- comp_share
| -- comp_version
| -- ...
| -- comp_photo
| -- layer_business_base
| -- base_ui
| -- build.gradle
| -- src
| -- ...
| -- build.gradle
| -- settings.gradle
你沒看錯椭懊,有兩個app module存在:
./app/
./layer_business/app/
點(diǎn)擊進(jìn)去看,發(fā)現(xiàn)兩個分支的代碼并沒有合并步势,兩邊修改的文件氧猬,同時存在。
原因分析
其實(shí)也不能叫分析坏瘩,因?yàn)樽詈笠矝]明確找到文檔有此類說明盅抚。以下純屬猜測:
merge
之后,git
無法對此文件內(nèi)容進(jìn)行合并并且給出沖突定位桑腮,為什么呢泉哈?因?yàn)?code>git沒有我們想象的那么聰明,它不知道破讨,如果把這兩個文件合并,那么對于文件位置奕纫,以那邊的修改為準(zhǔn)提陶?它到底是該移到新目錄,還是該留在原位置匹层?所以git
索性不合并了隙笆,直接告訴你锌蓄,對于主動發(fā)起合并的那個分支來說,這邊做了修改撑柔,那邊做了刪除瘸爽,怎么合并,你自己最清楚铅忿,所以你自己來剪决!
起初我也不信偉大的git怎么會有這種問題呢,于是我動手驗(yàn)證了下:
驗(yàn)證問題
不信的同學(xué)可以跟著我進(jìn)行一番驗(yàn)證:
$ mkdir gitest
$ cd gitest/
$ git init
Initialized empty Git repository in /Users/leonhe/repos/gitest/.git/
$ mkdir dir1
$ touch dir1/T.java
$ echo "initiation">dir1/T.java
$ git add .
$ git commit -m "initiation"
[master (root-commit) 28aac23] initiation
1 file changed, 1 insertion(+)
create mode 100644 dir1/T.java
$ git checkout -b develop
Switched to a new branch 'develop'
$ mkdir dir2
$ mv dir1/T.java dir2
$ echo "working on develop">>dir2/T.java
$ git add .
$ git commit -m "working on develop"
[develop 76e545e] working on develop
2 files changed, 2 insertions(+), 1 deletion(-)
delete mode 100644 dir1/T.java
create mode 100644 dir2/T.java
$ git checkout master
Switched to branch 'master'
$ echo "working on master">>dir1/T.java
$ git add .
$ git commit -m "working on master"
[master 16ecbc8] working on master
1 file changed, 1 insertion(+)
$ git merge develop
CONFLICT (modify/delete): dir1/T.java deleted in develop and modified in HEAD. Version HEAD of dir1/T.java left in tree.
Automatic merge failed; fix conflicts and then commit the result.
$ git status
On branch master
You have unmerged paths.
(fix conflicts and run "git commit")
(use "git merge --abort" to abort the merge)
Changes to be committed:
new file: dir2/T.java
Unmerged paths:
(use "git add/rm <file>..." as appropriate to mark resolution)
deleted by them: dir1/T.java
$
上面的操作就是:
- 初始化
git
倉庫 - 在
master
分支上創(chuàng)建dir1目錄檀训,并在dir1下創(chuàng)建T.java文件柑潦。 - 編輯T.java后,commit
- 切出一個
develop
分支峻凫,然后創(chuàng)建dir2目錄 - 將dir1下的T.java文件移動到dir2下
- 編輯dir2/T.java并提交到develop分支
- 切回master
- 編輯dir1/T.java渗鬼,commit
- merge
結(jié)果就是
CONFLICT (modify/delete): dir1/T.java deleted in develop and modified in HEAD. Version HEAD of dir1/T.java left in tree.
Automatic merge failed; fix conflicts and then commit the result.
再執(zhí)行git status
查看詳細(xì)狀態(tài)
$ git status
On branch master
You have unmerged paths.
(fix conflicts and run "git commit")
(use "git merge --abort" to abort the merge)
Changes to be committed:
new file: dir2/T.java
Unmerged paths:
(use "git add/rm <file>..." as appropriate to mark resolution)
deleted by them: dir1/T.java
$
git 提示,dir2/T.java是develop
分支新加的荧琼,而我們master
上的dir1/T.java文件被他們刪掉了譬胎。
那么此時的文件目錄結(jié)構(gòu)是怎樣的呢?
| -- gitest
| -- dir1
| -- T.java
| -- dir2
| -- T.java
merge
后命锄,兩個分支的T.java文件同時存在银择!
那么此時的兩個T.java文件里是否有兩個分支同時做的修改導(dǎo)致的沖突記錄呢?
先看看dir1/T.java:
$ cat dir2/T.java
initiation
working on master
沒有累舷,再看看dir2/T.java:
$ cat dir2/T.java
initiation
working on develop
并沒有任何沖突浩考!
那么對于這次merge
,我們只能手動去合并了嗎被盈?至少我在網(wǎng)上找了一圈析孽,沒有找到好的辦法。有的話只怎,請告訴我袜瞬,謝謝??!
小結(jié)
當(dāng)預(yù)計會出現(xiàn)這代碼的改動時身堡,一定要盡量頻繁地進(jìn)行兩邊同時進(jìn)行的分支的合并邓尤,如果等到兩邊都改動很大了的時候才來合并,那就等著吧CONFLICT (modify/delete)贴谎。汞扎。。??