亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > 綜合 > 正文

Git Submodule使用完整教程(小結)

2024-07-21 02:03:46
字體:
來源:轉載
供稿:網友

自從看了蔣鑫的《Git權威指南》之后就開始使用Git Submodule功能,團隊也都熟悉了怎么使用,多個子系統(模塊)都能及時更新到最新的公共資源,把使用的過程以及經驗和容易遇到的問題分享給大家。

Git Submodule功能剛剛開始學習可能覺得有點怪異,所以本教程把每一步的操作的命令和結果都用代碼的形式展現給大家,以便更好的理解。

1.對于公共資源各種程序員的處理方式

每個公司的系統都會有一套統一的系統風格,或者針對某一個大客戶的多個系統風格保持統一,而且如果風格改動后要同步到多個系統中;這樣的需求幾乎每個開發人員都遇到,下面看看各個層次的程序員怎么處理:

假如對于系統的風格需要幾個目錄:css、images、js。

  • 普通程序員,把最新版本的代碼逐個復制到每個項目中,如果有N個項目,那就是要復制N x 3次;如果漏掉了某個文件夾沒有復制…@(&#@#。
  • 文藝程序員,使用Git Submodule功能,執行:git submodule update,然后沖一杯咖啡悠哉的享受著。

引用一段《Git權威指南》的話: 項目的版本庫在某些情況蝦需要引用其他版本庫中的文件,例如公司積累了一套常用的函數庫,被多個項目調用,顯然這個函數庫的代碼不能直接放到某個項目的代碼中,而是要獨立為一個代碼庫,那么其他項目要調用公共函數庫該如何處理呢?分別把公共函數庫的文件拷貝到各自的項目中會造成冗余,丟棄了公共函數庫的維護歷史,這顯然不是好的方法。

2.開始學習Git Submodule

“工欲善其事,必先利其器”!

既然文藝程序員那么輕松就搞定了,那我們就把過程一一道來。

說明:本例采用兩個項目以及兩個公共類庫演示對submodule的操作。因為在一寫資料或者書上的例子都是一個項目對應1~N個lib,但是實際應用往往并不是這么簡單。

2.1 創建Git Submodule測試項目

2.1.1 準備環境

➜ henryyan@hy-hp ~ pwd/home/henryyanmkdir -p submd/repos

創建需要的本地倉庫:

cd ~/submd/reposgit --git-dir=lib1.git init --baregit --git-dir=lib2.git init --baregit --git-dir=project1.git init --baregit --git-dir=project2.git init --bare

初始化工作區:

mkdir ~/submd/wscd ~/submd/ws

2.1.2 初始化項目

初始化project1:

➜ henryyan@hy-hp ~/submd/ws git clone ../repos/project1.git Cloning into project1...done.warning: You appear to have cloned an empty repository. ➜ henryyan@hy-hp ~/submd/ws cd project1➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) echo "project1" > project-infos.txt➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ lsproject-infos.txt ➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git add project-infos.txt ➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git status# On branch master## Initial commit## Changes to be committed:# (use "git rm --cached <file>..." to unstage)## new file: project-infos.txt#➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git commit -m "init project1"[master (root-commit) 473a2e2] init project1 1 files changed, 1 insertions(+), 0 deletions(-) create mode 100644 project-infos.txt➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) git push origin masterCounting objects: 3, done.Writing objects: 100% (3/3), 232 bytes, done.Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.To /home/henryyan/submd/ws/../repos/project1.git * [new branch]  master -> master</file>

初始化project2:

➜ henryyan@hy-hp ~/submd/ws/project1 cd ..➜ henryyan@hy-hp ~/submd/ws git clone ../repos/project2.git Cloning into project2...done.warning: You appear to have cloned an empty repository. ➜ henryyan@hy-hp ~/submd/ws cd project2➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) echo "project2" > project-infos.txt➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ lsproject-infos.txt ➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ git add project-infos.txt ➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ git status# On branch master## Initial commit## Changes to be committed:# (use "git rm --cached <file>..." to unstage)## new file: project-infos.txt#➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ git commit -m "init project2"[master (root-commit) 473a2e2] init project2 1 files changed, 1 insertions(+), 0 deletions(-) create mode 100644 project-infos.txt➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) git push origin masterCounting objects: 3, done.Writing objects: 100% (3/3), 232 bytes, done.Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.To /home/henryyan/submd/ws/../repos/project2.git * [new branch]  master -> master</file>

2.1.3 初始化公共類庫

初始化公共類庫lib1:

➜ henryyan@hy-hp ~/submd/ws git clone ../repos/lib1.git Cloning into lib1...done.warning: You appear to have cloned an empty repository.➜ henryyan@hy-hp ~/submd/ws cd lib1 ➜ henryyan@hy-hp ~/submd/ws/lib1 git:(master) echo "I'm lib1." > lib1-features➜ henryyan@hy-hp ~/submd/ws/lib1 git:(master) ✗ git add lib1-features ➜ henryyan@hy-hp ~/submd/ws/lib1 git:(master) ✗ git commit -m "init lib1"[master (root-commit) c22aff8] init lib1 1 files changed, 1 insertions(+), 0 deletions(-) create mode 100644 lib1-features➜ henryyan@hy-hp ~/submd/ws/lib1 git:(master) git push origin masterCounting objects: 3, done.Writing objects: 100% (3/3), 227 bytes, done.Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.To /home/henryyan/submd/ws/../repos/lib1.git * [new branch]  master -> master

初始化公共類庫lib2:

➜ henryyan@hy-hp ~/submd/ws/lib1 git:(master) cd ..➜ henryyan@hy-hp ~/submd/ws git clone ../repos/lib2.git Cloning into lib2...done.warning: You appear to have cloned an empty repository.➜ henryyan@hy-hp ~/submd/ws cd lib2 ➜ henryyan@hy-hp ~/submd/ws/lib2 git:(master) echo "I'm lib2." > lib2-features➜ henryyan@hy-hp ~/submd/ws/lib2 git:(master) ✗ git add lib2-features ➜ henryyan@hy-hp ~/submd/ws/lib2 git:(master) ✗ git commit -m "init lib2"[master (root-commit) c22aff8] init lib2 1 files changed, 1 insertions(+), 0 deletions(-) create mode 100644 lib2-features➜ henryyan@hy-hp ~/submd/ws/lib2 git:(master) git push origin masterCounting objects: 3, done.Writing objects: 100% (3/3), 227 bytes, done.Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.To /home/henryyan/submd/ws/../repos/lib2.git * [new branch]  master -> master

2.2 為主項目添加Submodules

2.2.1 為project1添加lib1和lib2

➜ henryyan@hy-hp ~/submd/ws/lib2 git:(master) cd ../project1➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) lsproject-infos.txt➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) git submodule add ~/submd/repos/lib1.git libs/lib1Cloning into libs/lib1...done.➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git submodule add ~/submd/repos/lib2.git libs/lib2Cloning into libs/lib2...done.➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ lslibs project-infos.txt➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ ls libs lib1 lib2 ➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git status # On branch master# Changes to be committed:# (use "git reset HEAD <file>..." to unstage)## new file: .gitmodules# new file: libs/lib1# new file: libs/lib2# # 查看一下公共類庫的內容 ➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) cat libs/lib1/lib1-featuresI'm lib1.➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) cat libs/lib2/lib2-featuresI'm lib2.</file>

好了,到目前為止我們已經使用git submodule add命令為project1成功添加了兩個公共類庫(lib1、lib2),查看了當前的狀態發現添加了一個新文件(.gitmodules)和兩個文件夾(libs/lib1、libs/lib2);那么新增的.gitmodules文件是做什么用的呢?我們查看一下文件內容便知曉了:

n@hy-hp ~/submd/ws/project1 git:(master) ✗ cat .gitmodules [submodule "libs/lib1"] path = libs/lib1 url = /home/henryyan/submd/repos/lib1.git[submodule "libs/lib2"] path = libs/lib2 url = /home/henryyan/submd/repos/lib2.git

原來如此,.gitmodules記錄了每個submodule的引用信息,知道在當前項目的位置以及倉庫的所在。

好的,我們現在把更改提交到倉庫。

➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git commit -a -m "add submodules[lib1,lib2] to project1"[master 7157977] add submodules[lib1,lib2] to project1 3 files changed, 8 insertions(+), 0 deletions(-) create mode 100644 .gitmodules create mode 160000 libs/lib1 create mode 160000 libs/lib2 ➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) git pushCounting objects: 5, done.Delta compression using up to 2 threads.Compressing objects: 100% (4/4), done.Writing objects: 100% (4/4), 491 bytes, done.Total 4 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (4/4), done.To /home/henryyan/submd/ws/../repos/project1.git 45cbbcb..7157977 master -> master

假如你是第一次引入公共類庫的開發人員,那么項目組的其他成員怎么Clone帶有Submodule的項目呢,下面我們再clone一個項目講解如何操作。

2.3 Clone帶有Submodule的倉庫

模擬開發人員B……

➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) cd ~/submd/ws➜ henryyan@hy-hp ~/submd/ws git clone ../repos/project1.git project1-bCloning into project1-b...done.➜ henryyan@hy-hp ~/submd/ws cd project1-b ➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) git submodule -c22aff85be91eca442734dcb07115ffe526b13a1 libs/lib1-7290dce0062bd77df1d83b27dd3fa3f25a836b54 libs/lib2

看到submodules的狀態是hash碼和文件目錄,但是注意前面有一個減號:-,含義是該子模塊還沒有檢出。

OK,檢出project1-b的submodules……

➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) git submodule initSubmodule 'libs/lib1' (/home/henryyan/submd/repos/lib1.git) registered for path 'libs/lib1'Submodule 'libs/lib2' (/home/henryyan/submd/repos/lib2.git) registered for path 'libs/lib2'➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) git submodule updateCloning into libs/lib1...done.Submodule path 'libs/lib1': checked out 'c22aff85be91eca442734dcb07115ffe526b13a1'Cloning into libs/lib2...done.Submodule path 'libs/lib2': checked out '7290dce0062bd77df1d83b27dd3fa3f25a836b54'

讀者可以查看:.git/config文件的內容,最下面有submodule的注冊信息!

驗證一下類庫的文件是否存在:

➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) cat libs/lib1/lib1-features libs/lib2/lib2-featuresI'm lib1.I'm lib2.

上面的兩個命令(git submodule init & update)其實可以簡化,后面會講到!

2.3 修改Submodule

我們在開發人員B的項目上修改Submodule的內容。

先看一下當前Submodule的狀態:

➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) cd libs/lib1➜ henryyan@hy-hp ~/submd/ws/project1-b/libs/lib1 git status# Not currently on any branch.nothing to commit (working directory clean)

為什么是Not currently on any branch呢?不是應該默認在master分支嗎?別急,一一解答!

Git對于Submodule有特殊的處理方式,在一個主項目中引入了Submodule其實Git做了3件事情:

  • 記錄引用的倉庫
  • 記錄主項目中Submodules的目錄位置
  • 記錄引用Submodule的commit id

在project1中push之后其實就是更新了引用的commit id,然后project1-b在clone的時候獲取到了submodule的commit id,然后當執行git submodule update的時候git就根據gitlink獲取submodule的commit id,最后獲取submodule的文件,所以clone之后不在任何分支上;但是master分支的commit id和HEAD保持一致。

查看~/submd/ws/project1-b/libs/lib1的引用信息:

➜ henryyan@hy-hp ~/submd/ws/project1-b/libs/lib1 cat .git/HEADc22aff85be91eca442734dcb07115ffe526b13a1➜ henryyan@hy-hp ~/submd/ws/project1-b/libs/lib1 cat .git/refs/heads/master    c22aff85be91eca442734dcb07115ffe526b13a1

現在我們要修改lib1的文件需要先切換到master分支:

➜ henryyan@hy-hp ~/submd/ws/project1-b/libs/lib1 git checkout masterSwitched to branch 'master'➜ henryyan@hy-hp ~/submd/ws/project1-b/libs/lib1 git:(master) echo "add by developer B" >> lib1-features➜ henryyan@hy-hp ~/submd/ws/project1-b/libs/lib1 git:(master) ✗ git commit -a -m "update lib1-features by developer B"[master 36ad12d] update lib1-features by developer B 1 files changed, 1 insertions(+), 0 deletions(-)

在主項目中修改Submodule提交到倉庫稍微繁瑣一點,在git push之前我們先看看project1-b狀態:

➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ git status# On branch master# Changes not staged for commit:# (use "git add <file>..." to update what will be committed)# (use "git checkout -- <file>..." to discard changes in working directory)## modified: libs/lib1 (new commits)#no changes added to commit (use "git add" and/or "git commit -a")</file></file>

libs/lib1 (new commits)狀態表示libs/lib1有新的提交,這個比較特殊,看看project1-b的狀態:

➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ git diffdiff --git a/libs/lib1 b/libs/lib1index c22aff8..36ad12d 160000--- a/libs/lib1+++ b/libs/lib1@@ -1 +1 @@-Subproject commit c22aff85be91eca442734dcb07115ffe526b13a1+Subproject commit 36ad12d40d8a41a4a88a64add27bd57cf56c9de2

從狀態中可以看出libs/lib1的commit id由原來的c22aff85be91eca442734dcb07115ffe526b13a1更改為36ad12d40d8a41a4a88a64add27bd57cf56c9de2

注意:如果現在執行了git submodule update操作那么libs/lib1的commit id又會還原到c22aff85be91eca442734dcb07115ffe526b13a1,

這樣的話剛剛的修改是不是就丟死了呢?不會,因為修改已經提交到了master分支,只要再git checkout master就可以了。
現在可以把libs/lib1的修改提交到倉庫了:

➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ cd libs/lib1➜ henryyan@hy-hp ~/submd/ws/project1-b/libs/lib1 git:(master) git pushCounting objects: 5, done.Writing objects: 100% (3/3), 300 bytes, done.Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.To /home/henryyan/submd/repos/lib1.git c22aff8..36ad12d master -> master

現在僅僅只完成了一步,下一步要提交project1-b引用submodule的commit id:

➜ henryyan@hy-hp ~/submd/ws/project1-b/libs/lib1 git:(master) cd ~/submd/ws/project1-b➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ git add -u➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ git commit -m "update libs/lib1 to lastest commit id"[master c96838a] update libs/lib1 to lastest commit id 1 files changed, 1 insertions(+), 1 deletions(-)➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) git pushCounting objects: 5, done.Delta compression using up to 2 threads.Compressing objects: 100% (3/3), done.Writing objects: 100% (3/3), 395 bytes, done.Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.To /home/henryyan/submd/ws/../repos/project1.git 7157977..c96838a master -> master

OK,大功高成,我們完成了Submodule的修改并把libs/lib1的最新commit id提交到了倉庫。

接下來要看看project1怎么獲取submodule了。

2.4 更新主項目的Submodules

好的,讓我們先進入project1目錄同步倉庫:

➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) cd ../project1➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) git pullremote: Counting objects: 5, done.remote: Compressing objects: 100% (3/3), done.remote: Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.From /home/henryyan/submd/ws/../repos/project1 7157977..c96838a master  -> origin/masterUpdating 7157977..c96838aFast-forward libs/lib1 | 2 +- 1 files changed, 1 insertions(+), 1 deletions(-)➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git status# On branch master# Changes not staged for commit:# (use "git add <file>..." to update what will be committed)# (use "git checkout -- <file>..." to discard changes in working directory)## modified: libs/lib1 (new commits)#no changes added to commit (use "git add" and/or "git commit -a")</file></file>

我們運行了git pull命令和git status獲取了最新的倉庫源碼,然后看到了狀態時modified,這是為什么呢?

我們用git diff比較一下不同:

➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git diffdiff --git a/libs/lib1 b/libs/lib1index 36ad12d..c22aff8 160000--- a/libs/lib1+++ b/libs/lib1@@ -1 +1 @@-Subproject commit 36ad12d40d8a41a4a88a64add27bd57cf56c9de2+Subproject commit c22aff85be91eca442734dcb07115ffe526b13a1從diff的結果分析出來時因為submodule的commit id更改了,我們前面剛剛講了要在主項目更新submodule的內容首先要提交submdoule的內容,然后再更新主項目中引用的submodulecommit id;現在我們看到的不同就是因為剛剛更改了project1-b的submodule commit id;好的,我來學習一下怎么更新project1的公共類庫。follow me…… ➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git submodule update➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git status# On branch master# Changes not staged for commit:# (use "git add <file>..." to update what will be committed)# (use "git checkout -- <file>..." to discard changes in working directory)## modified: libs/lib1 (new commits)#no changes added to commit (use "git add" and/or "git commit -a")</file></file>

泥馬,為什么沒有更新?git submodule update命令不是更新子模塊倉庫的嗎?

別急,先聽我解釋;因為子模塊是在project1中引入的,git submodule add ~/submd/repos/lib1.git libs/lib1命令的結果,操作之后git只是把lib1的內容clone到了project1中,但是沒有在倉庫注冊,證據如下:

➜ henryyan@hy-hp ~/submd2/ws/project1 git:(master) ✗ cat .git/config[core] repositoryformatversion = 0 filemode = true bare = false logallrefupdates = true[remote "origin"] fetch = +refs/heads/*:refs/remotes/origin/* url = /home/henryyan/submd/ws/../repos/project1.git[branch "master"] remote = origin merge = refs/heads/master

我們說過git submodule init就是在.git/config中注冊子模塊的信息,下面我們試試注冊之后再更新子模塊:

➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git submodule initSubmodule 'libs/lib1' (/home/henryyan/submd/repos/lib1.git) registered for path 'libs/lib1'Submodule 'libs/lib2' (/home/henryyan/submd/repos/lib2.git) registered for path 'libs/lib2'➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git submodule updateremote: Counting objects: 5, done.remote: Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.From /home/henryyan/submd/repos/lib1 c22aff8..36ad12d master  -> origin/masterSubmodule path 'libs/lib1': checked out '36ad12d40d8a41a4a88a64add27bd57cf56c9de2' ➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) cat .git/config[core] repositoryformatversion = 0 filemode = true bare = false logallrefupdates = true[remote "origin"] fetch = +refs/heads/*:refs/remotes/origin/* url = /home/henryyan/submd/ws/../repos/project1.git[branch "master"] remote = origin merge = refs/heads/master[submodule "libs/lib1"] url = /home/henryyan/submd/repos/lib1.git[submodule "libs/lib2"] url = /home/henryyan/submd/repos/lib2.git ➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) cat libs/lib1/lib1-featuresI'm lib1.add by developer B

上面的結果足以證明剛剛的推斷,所以記得當需要更新子模塊的內容時請先確保已經運行過git submodule init。

2.5 為project2添加lib1和lib2

這個操作對于讀到這里的你來說應該是輕車熟路了,action:

➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) cd ~/submd/ws/project2➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) git submodule add ~/submd/repos/lib1.git libs/lib1Cloning into libs/lib1...done.➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ git submodule add ~/submd/repos/lib2.git libs/lib2zsh: correct 'libs/lib2' to 'libs/lib1' [nyae]? nCloning into libs/lib2...done.➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ lslibs project-infos.txt➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ git submodule initSubmodule 'libs/lib1' (/home/henryyan/submd/repos/lib1.git) registered for path 'libs/lib1'Submodule 'libs/lib2' (/home/henryyan/submd/repos/lib2.git) registered for path 'libs/lib2' ➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ git status# On branch master# Changes to be committed:# (use "git reset HEAD <file>..." to unstage)## new file: .gitmodules# new file: libs/lib1# new file: libs/lib2#➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ git commit -a -m "add lib1 and lib2"[master 8dc697f] add lib1 and lib2 3 files changed, 8 insertions(+), 0 deletions(-) create mode 100644 .gitmodules create mode 160000 libs/lib1 create mode 160000 libs/lib2➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) git pushCounting objects: 5, done.Delta compression using up to 2 threads.Compressing objects: 100% (4/4), done.Writing objects: 100% (4/4), 471 bytes, done.Total 4 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (4/4), done.To /home/henryyan/submd/ws/../repos/project2.git 6e15c68..8dc697f master -> master </file>

我們依次執行了添加submodule并commit和push到倉庫,此階段任務完成。

2.6 修改lib1和lib2并同步到project1和project2

假如開發人員C同時負責project1和project2,有可能在修改project1的某個功能的時候發現lib1或者lib2的某個組件有bug需要修復,這個需求多模塊和大型系統中經常遇到,我們應該怎么解決呢?

假如我的需求如下:

在lib1中添加一個文件:README,用來描述lib1的功能

在lib2中的lib2-features文件中添加一寫文字:學習Git submodule的修改并同步功能

2.6.1 在lib1中添加一個文件:README

➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) cd libs/lib1➜ henryyan@hy-hp ~/submd/ws/project2/libs/lib1 git:(master) echo "lib1 readme contents" > README➜ henryyan@hy-hp ~/submd/ws/project2/libs/lib1 git:(master) ✗ git add README ➜ henryyan@hy-hp ~/submd/ws/project2/libs/lib1 git:(master) ✗ git commit -m "add file README"[master 8c666d8] add file README 1 files changed, 1 insertions(+), 0 deletions(-) create mode 100644 README➜ henryyan@hy-hp ~/submd/ws/project2/libs/lib1 git:(master) git pushCounting objects: 4, done.Delta compression using up to 2 threads.Compressing objects: 100% (2/2), done.Writing objects: 100% (3/3), 310 bytes, done.Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.To /home/henryyan/submd/repos/lib1.git 36ad12d..8c666d8 master -> master

前面提到過現在僅僅只完成了一部分,我們需要在project2中再更新lib1的commit id:

➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ git status # On branch master# Changes not staged for commit:# (use "git add <file>..." to update what will be committed)# (use "git checkout -- <file>..." to discard changes in working directory)## modified: libs/lib1 (new commits)#no changes added to commit (use "git add" and/or "git commit -a")➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ git add libs/lib1➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ git commit -m "update lib1 to lastest commit id"[master ce1f3ba] update lib1 to lastest commit id 1 files changed, 1 insertions(+), 1 deletions(-)</file></file>

我們暫時不push到倉庫,等待和lib2的修改一起push。

2.6.2 在lib2中的lib2-features文件添加文字

➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) cd libs/lib2➜ henryyan@hy-hp ~/submd/ws/project2/libs/lib2 git:(master) echo "學習Git submodule的修改并同步功能" >> lib2-features ➜ henryyan@hy-hp ~/submd/ws/project2/libs/lib2 git:(master) ✗ git add lib2-features ➜ henryyan@hy-hp ~/submd/ws/project2/libs/lib2 git:(master) ✗ git commit -m "添加文字:學習Git submodule的修改并同步功能"[master e372b21] 添加文字:學習Git submodule的修改并同步功能 1 files changed, 1 insertions(+), 0 deletions(-)➜ henryyan@hy-hp ~/submd/ws/project2/libs/lib2 git:(master) git pushCounting objects: 5, done.Delta compression using up to 2 threads.Compressing objects: 100% (2/2), done.Writing objects: 100% (3/3), 376 bytes, done.Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.To /home/henryyan/submd/repos/lib2.git 7290dce..e372b21 master -> master ➜ henryyan@hy-hp ~/submd/ws/project2/libs/lib2 git:(master) echo "學習Git submodule的修改并同步功能" >> lib2-features ➜ henryyan@hy-hp ~/submd/ws/project2/libs/lib2 git:(master) ✗ git add lib2-features ➜ henryyan@hy-hp ~/submd/ws/project2/libs/lib2 git:(master) ✗ git commit -m "添加文字:學習Git submodule的修改并同步功能"[master e372b21] 添加文字:學習Git submodule的修改并同步功能 1 files changed, 1 insertions(+), 0 deletions(-)➜ henryyan@hy-hp ~/submd/ws/project2/libs/lib2 git:(master) git pushCounting objects: 5, done.Delta compression using up to 2 threads.Compressing objects: 100% (2/2), done.Writing objects: 100% (3/3), 376 bytes, done.Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.To /home/henryyan/submd/repos/lib2.git 7290dce..e372b21 master -> master➜ henryyan@hy-hp ~/submd/ws/project2/libs/lib2 git:(master) cd -~/submd/ws/project2➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ git status # On branch master# Your branch is ahead of 'origin/master' by 1 commit.## Changes not staged for commit:# (use "git add <file>..." to update what will be committed)# (use "git checkout -- <file>..." to discard changes in working directory)## modified: libs/lib2 (new commits)#no changes added to commit (use "git add" and/or "git commit -a")➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ git add libs/lib2 ➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) ✗ git commit -m "update lib2 to lastest commit id"[master df344c5] update lib2 to lastest commit id 1 files changed, 1 insertions(+), 1 deletions(-)➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) git status # On branch master# Your branch is ahead of 'origin/master' by 2 commits.#nothing to commit (working directory clean)➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) git pushCounting objects: 8, done.Delta compression using up to 2 threads.Compressing objects: 100% (6/6), done.Writing objects: 100% (6/6), 776 bytes, done.Total 6 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (6/6), done.To /home/henryyan/submd/ws/../repos/project2.git 8dc697f..df344c5 master -> master</file></file>

2.7 同步project2的lib1和lib2的修改到project1

現在project2已經享受到了最新的代碼帶來的快樂,那么既然project1和project2屬于同一個風格,或者調用同一個功能,要讓這兩個(可能幾十個)項目保持一致。

➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) cd ../project1➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) git pullAlready up-to-date.

看看上面的結果對嗎?為什么lib1和lib2更新了但是沒有顯示new commits呢?說到這里我記得剛剛開始學習的時候真得要暈死了,Git跟我玩捉迷藏游戲,為什么我明明提交了但是從project1更新不到任何改動呢?

幫大家分析一下問題,不過在分析之前先看看當前(project1和project2)的submodule狀態:

# project2 的狀態,也就是我們剛剛修改后的狀態➜ henryyan@hy-hp ~/submd/ws/project2 git:(master) git submodule  8c666d86531513dd1aebdf235f142adbac72c035 libs/lib1 (heads/master) e372b21dffa611802c282278ec916b5418acebc2 libs/lib2 (heads/master) # project1 的狀態,等待更新submodules➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) git submodule  36ad12d40d8a41a4a88a64add27bd57cf56c9de2 libs/lib1 (remotes/origin/HEAD) 7290dce0062bd77df1d83b27dd3fa3f25a836b54 libs/lib2 (heads/master)

 兩個項目有兩個區別:

  • commit id各不相同
  • libs/lib1所處的分支不同

2.7.1 更新project1的lib1和lib2改動

我們還記得剛剛在project2中修改的時候把lib1和lib2都切換到了master分支,目前project1中的lib1不在任何分支,我們先切換到master分支:

➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) cd libs/lib1➜ henryyan@hy-hp ~/submd/ws/project1/libs/lib1 git checkout masterPrevious HEAD position was 36ad12d... update lib1-features by developer BSwitched to branch 'master'Your branch is behind 'origin/master' by 1 commit, and can be fast-forwarded.➜ henryyan@hy-hp ~/submd/ws/project1/libs/lib1 git:(master) git pullremote: Counting objects: 4, done.remote: Compressing objects: 100% (2/2), done.remote: Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.From /home/henryyan/submd/repos/lib1 36ad12d..8c666d8 master  -> origin/masterUpdating c22aff8..8c666d8Fast-forward README  | 1 + lib1-features | 1 + 2 files changed, 2 insertions(+), 0 deletions(-) create mode 100644 README➜ henryyan@hy-hp ~/submd/ws/project1/libs/lib1 git:(master) 

果不其然,我們看到了剛剛在project2中修改的內容,同步到了project1中,當然現在更新了project1的lib1,commit id也會隨之變動:

➜ henryyan@hy-hp ~/submd/ws/project1/libs/lib1 git:(master) cd ../../➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git status # On branch master# Changes not staged for commit:# (use "git add <file>..." to update what will be committed)# (use "git checkout -- <file>..." to discard changes in working directory)## modified: libs/lib1 (new commits)#no changes added to commit (use "git add" and/or "git commit -a")➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git diffdiff --git a/libs/lib1 b/libs/lib1index 36ad12d..8c666d8 160000--- a/libs/lib1+++ b/libs/lib1@@ -1 +1 @@-Subproject commit 36ad12d40d8a41a4a88a64add27bd57cf56c9de2+Subproject commit 8c666d86531513dd1aebdf235f142adbac72c035</file></file>

現在最新的commit id和project2目前的狀態一致,說明真的同步了;好的,現在可以使用相同的辦法更新lib2了:

➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ cd libs/lib2➜ henryyan@hy-hp ~/submd/ws/project1/libs/lib2 git:(master) git pullremote: Counting objects: 5, done.remote: Compressing objects: 100% (2/2), done.remote: Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.From /home/henryyan/submd/repos/lib2 7290dce..e372b21 master  -> origin/masterUpdating 7290dce..e372b21Fast-forward lib2-features | 1 + 1 files changed, 1 insertions(+), 0 deletions(-)

2.7.2 更新project1的submodule引用

在2.7.1中我們更新了project1的lib1和lib2的最新版本,現在要把最新的commit id保存到project1中以保持最新的引用。

➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git status# On branch master# Changes not staged for commit:# (use "git add <file>..." to update what will be committed)# (use "git checkout -- <file>..." to discard changes in working directory)## modified: libs/lib1 (new commits)# modified: libs/lib2 (new commits)#no changes added to commit (use "git add" and/or "git commit -a")➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) ✗ git commit -a -m "update lib1 and lib2 commit id to new version"[master 8fcca50] update lib1 and lib2 commit id to new version 2 files changed, 2 insertions(+), 2 deletions(-)➜ henryyan@hy-hp ~/submd/ws/project1 git:(master) git pushCounting objects: 5, done.Delta compression using up to 2 threads.Compressing objects: 100% (3/3), done.Writing objects: 100% (3/3), 397 bytes, done.Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.To /home/henryyan/submd/ws/../repos/project1.git c96838a..8fcca50 master -> master</file></file>

2.8 更新project1-b項目的子模塊(使用腳本)

➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) git pullremote: Counting objects: 5, done.remote: Compressing objects: 100% (3/3), done.remote: Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.From /home/henryyan/submd/ws/../repos/project1 c96838a..8fcca50 master  -> origin/masterUpdating c96838a..8fcca50Fast-forward libs/lib1 | 2 +- libs/lib2 | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-)➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ git status# On branch master# Changes not staged for commit:# (use "git add <file>..." to update what will be committed)# (use "git checkout -- <file>..." to discard changes in working directory)## modified: libs/lib1 (new commits)# modified: libs/lib2 (new commits)#no changes added to commit (use "git add" and/or "git commit -a")</file></file>

Git提示lib1和lib2有更新內容,這個判斷的依據來源于submodule commit id的引用。

現在怎么更新呢?難道還是像project1中那樣進入子模塊的目錄然后git checkout master,接著git pull?

而且現在僅僅才兩個子模塊、兩個項目,如果在真實的項目中使用的話可能幾個到幾十個不等,再加上N個submodule,自己算一下要怎么更新多少個submodules?

例如筆者現在做的一個項目有5個web模塊,每個web模塊引用公共的css、js、images、jsp資源,這樣就有20個submodule需要更新?。?!

工欲善其事,必先利其器,寫一個腳本代替手動任務。

2.8.1 牛刀小試

➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ grep path .gitmodules | awk '{ print $3 }' > /tmp/study-git-submodule-dirs➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ cat /tmp/study-git-submodule-dirs libs/lib1 libs/lib2

我們通過分析.gitmodules文件得出子模塊的路徑,然后就可以根據這些路徑進行更新。

2.8.2 上路

➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ mkdir bin➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ vi bin/update-submodules.sh

把下面的腳本復制到bin/update-submodules.sh中:

#!/bin/bashgrep path .gitmodules | awk '{ print $3 }' > /tmp/study-git-submodule-dirs # readwhile read LINEdo echo $LINE (cd ./$LINE && git checkout master && git pull)done < /tmp/study-git-submodule-dirs

稍微解釋一下上面的腳本執行過程:

  • 首先把子模塊的路徑寫入到文件/tmp/study-git-submodule-dirs中;
  • 然后讀取文件中的子模塊路徑,依次切換到master分支(修改都是在master分支上進行的),最后更新最近改動。

2.8.3 2013-01-19更新

網友@紫煌給出了更好的辦法,一個命令就可以代替上面的bin/update-submodules.sh的功能:

git submodule foreach git pull

此命令也腳本一樣,循環進入(enter)每個子模塊的目錄,然后執行foreach后面的命令。

該后面的命令可以任意的,例如 git submodule foreach ls -l 可以列出每個子模塊的文件列表

2.8.3 體驗工具帶來的便利

➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ git submodule +36ad12d40d8a41a4a88a64add27bd57cf56c9de2 libs/lib1 (heads/master)+7290dce0062bd77df1d83b27dd3fa3f25a836b54 libs/lib2 (heads/master) # 添加執行權限➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ chmod +x ./bin/update-submodules.sh➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ ./bin/update-submodules.sh libs/lib1Already on 'master'remote: Counting objects: 4, done.remote: Compressing objects: 100% (2/2), done.remote: Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.From /home/henryyan/submd/repos/lib1  36ad12d..8c666d8 master   -> origin/masterUpdating 36ad12d..8c666d8Fast-forward README |  1 + 1 files changed, 1 insertions(+), 0 deletions(-) create mode 100644 READMElibs/lib2Switched to branch 'master'remote: Counting objects: 5, done.remote: Compressing objects: 100% (2/2), done.remote: Total 3 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (3/3), done.From /home/henryyan/submd/repos/lib2  7290dce..e372b21 master   -> origin/masterUpdating 7290dce..e372b21Fast-forward lib2-features |  1 + 1 files changed, 1 insertions(+), 0 deletions(-)➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ git submodule  8c666d86531513dd1aebdf235f142adbac72c035 libs/lib1 (heads/master) e372b21dffa611802c282278ec916b5418acebc2 libs/lib2 (heads/master)  ➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ git status # On branch master# Untracked files:#  (use "git add <file>..." to include in what will be committed)##  bin/nothing added to commit but untracked files present (use "git add" to track)</file>

更新之后的兩個變化:

  • git submodule的結果和project2的submodule commit id保持一致;
  • project1-b不再提示new commits了。

現在可以把工具添加到倉庫了,當然你可以很驕傲的分享給其他項目組的同事。

➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ git add bin/update-submodules.sh ➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) ✗ git commit -m "添加自動更新submodule的快捷腳本^_^"[master 756e788] 添加自動更新submodule的快捷腳本^_^ 1 files changed, 9 insertions(+), 0 deletions(-) create mode 100755 bin/update-submodules.sh➜ henryyan@hy-hp ~/submd/ws/project1-b git:(master) git pushCounting objects: 5, done.Delta compression using up to 2 threads.Compressing objects: 100% (3/3), done.Writing objects: 100% (4/4), 625 bytes, done.Total 4 (delta 0), reused 0 (delta 0)Unpacking objects: 100% (4/4), done.To /home/henryyan/submd/ws/../repos/project1.git  8fcca50..756e788 master -> master

2.9 新進員工加入團隊,一次性Clone項目和Submodules

一般人使用的時候都是使用如下命令:

git clone /path/to/repos/foo.gitgit submodule initgit submodule update

新員工不耐煩了,嘴上不說但是心里想:怎么那么麻煩?

上面的命令簡直弱暴了,直接一行命令搞定:

git clone --recursive /path/to/repos/foo.git

亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

亚洲激情视频网| 欧美性受xxxx黑人猛交| 亚洲国产精品久久久久秋霞蜜臀| 中文字幕v亚洲ⅴv天堂| 国外成人在线播放| 日产日韩在线亚洲欧美| 日韩免费观看视频| 一区国产精品视频| 色www亚洲国产张柏芝| 国产精品成人在线| 97成人精品区在线播放| 亚洲无av在线中文字幕| 亚洲欧美国产精品久久久久久久| 91九色综合久久| 91精品久久久久久久久青青| 欧美日本高清视频| 欧美精品18videosex性欧美| 中文字幕亚洲色图| www亚洲欧美| 欧美亚洲成人精品| 奇门遁甲1982国语版免费观看高清| 黑人巨大精品欧美一区二区| 国产精品无av码在线观看| 欧美日韩中文字幕日韩欧美| 国产日韩在线看| 91精品视频观看| 欧美人与物videos| 国产亚洲欧洲在线| 日本aⅴ大伊香蕉精品视频| 国产日韩欧美中文在线播放| 日韩av在线最新| 久久久久一本一区二区青青蜜月| 91国内精品久久| 高清欧美性猛交xxxx黑人猛交| 久久资源免费视频| 国产不卡av在线免费观看| 日韩av电影手机在线| 亚洲激情在线观看| 久久夜精品香蕉| 亚洲美女av在线| 久久久99久久精品女同性| 亚洲欧美激情四射在线日| 亚洲视频精品在线| 97久久精品视频| 亚洲一区二区久久久久久久| 亚洲第一av网| 国产激情久久久久| 久久久久久噜噜噜久久久精品| 日韩av电影院| 久久久久久久成人| 亚洲精品国产精品自产a区红杏吧| 亚洲国产精品久久精品怡红院| 日韩大陆欧美高清视频区| 国产精品色悠悠| 亚洲精品国偷自产在线99热| 精品日韩视频在线观看| 日韩亚洲国产中文字幕| 亚洲欧洲激情在线| 亚洲人在线视频| 日韩亚洲欧美中文高清在线| 精品偷拍各种wc美女嘘嘘| 91天堂在线观看| 丝袜美腿精品国产二区| 亚洲欧洲视频在线| 日韩欧美综合在线视频| 久久久噜久噜久久综合| 中文字幕日韩视频| 91精品在线看| 国产成人中文字幕| 久久久亚洲成人| 国产福利视频一区二区| 在线一区二区日韩| 91国在线精品国内播放| 亚洲美女av电影| 国产精品第二页| 成人欧美一区二区三区黑人孕妇| 成人乱人伦精品视频在线观看| 欧美猛少妇色xxxxx| 久久精品色欧美aⅴ一区二区| 久久久久久国产免费| 成人免费观看49www在线观看| 欧美风情在线观看| 日本在线观看天堂男亚洲| 亚洲热线99精品视频| 国产欧洲精品视频| 色婷婷久久av| 日韩av免费在线播放| 国产久一一精品| 久久精品国产v日韩v亚洲| 欧美激情综合色综合啪啪五月| 欧美亚洲国产日韩2020| 国产成人精彩在线视频九色| 精品调教chinesegay| 91久久中文字幕| 亚洲免费影视第一页| 久久精品国产2020观看福利| 粉嫩老牛aⅴ一区二区三区| 色www亚洲国产张柏芝| 日本成熟性欧美| 亚洲精品丝袜日韩| 97免费中文视频在线观看| 国产精品美女免费视频| 91亚洲午夜在线| www.xxxx欧美| 日韩视频免费中文字幕| 成人国产在线激情| 欧美在线亚洲在线| www.日韩视频| 在线精品播放av| 国内精品伊人久久| 91精品久久久久久久久中文字幕| 亚洲人a成www在线影院| 欧美高清在线播放| 97精品伊人久久久大香线蕉| 久久精品99久久香蕉国产色戒| 色婷婷**av毛片一区| 最近中文字幕2019免费| 久久九九精品99国产精品| 亚洲男人第一网站| 97视频在线看| 精品久久久久人成| 中文字幕亚洲综合久久筱田步美| 91精品国产色综合久久不卡98口| 精品亚洲国产视频| 亚洲欧美中文日韩在线| 人妖精品videosex性欧美| 日韩女优人人人人射在线视频| 成人黄色中文字幕| 亚洲毛片在线免费观看| 91禁外国网站| 欧美日韩精品在线观看| 亚洲美女性视频| 日韩在线观看免费全集电视剧网站| 国产精品狠色婷| 日本免费在线精品| 97在线免费观看视频| 国产成人免费91av在线| 91中文在线视频| 91九色单男在线观看| 国产精品男人爽免费视频1| 欧美中在线观看| 国内久久久精品| 美女福利精品视频| 97国产一区二区精品久久呦| 国内自拍欧美激情| 欧美丰满少妇xxxxx做受| 亚洲高清一区二| 亚洲国产天堂久久综合网| 51午夜精品视频| 97成人精品视频在线观看| 精品国产精品自拍| 亚洲一区二区中文| 蜜臀久久99精品久久久久久宅男| 国产精品亚洲激情| 97人人模人人爽人人喊中文字| 欧美日韩亚洲系列| 亚洲精品suv精品一区二区| 日韩欧美精品网址| 日韩美女在线看| 久久精品国产电影| 伊是香蕉大人久久| 亚洲欧美日本精品| 亚洲国产精品va| 国产精品福利网|