Git
資訊系必備的版本管理工具
- 基本介紹
- git 原理
- 基本操作
- 實戰體驗
- 進階操作
大綱
data:image/s3,"s3://crabby-images/3da33/3da3375e2fa6852774b58d09a3ccb2d66716f529" alt=""
Git 前言
關於這是什麼
為什麼需要版本控制?
data:image/s3,"s3://crabby-images/3decf/3decf28a7f55653fd3090eb033a16b1dc44e6c7b" alt=""
- 不小心改到檔案然後按下儲存
- 有雷隊友把 code 整個刪掉了
- 大專案出 bug 了,想找出是什麼時候開始出了問題
為什麼需要版本控制?
Git 可以做什麼?
data:image/s3,"s3://crabby-images/e92bf/e92bf53d1fb5ff4ec903e2bb704a81f80c47f5f3" alt=""
Cloud
data:image/s3,"s3://crabby-images/454ed/454ed05ab865e192251fd569deb16ea2dbc075d1" alt=""
Local
data:image/s3,"s3://crabby-images/e92bf/e92bf53d1fb5ff4ec903e2bb704a81f80c47f5f3" alt=""
Cloud
data:image/s3,"s3://crabby-images/454ed/454ed05ab865e192251fd569deb16ea2dbc075d1" alt=""
data:image/s3,"s3://crabby-images/e6b4a/e6b4a7cd2a170f3f7eb49c548454ba4f99b30987" alt=""
Clone
Local
data:image/s3,"s3://crabby-images/751f8/751f8673989edc04e00b0574dd4f0166539616ef" alt=""
Repository
data:image/s3,"s3://crabby-images/e92bf/e92bf53d1fb5ff4ec903e2bb704a81f80c47f5f3" alt=""
Cloud
data:image/s3,"s3://crabby-images/454ed/454ed05ab865e192251fd569deb16ea2dbc075d1" alt=""
data:image/s3,"s3://crabby-images/e6b4a/e6b4a7cd2a170f3f7eb49c548454ba4f99b30987" alt=""
Pull
data:image/s3,"s3://crabby-images/e6b4a/e6b4a7cd2a170f3f7eb49c548454ba4f99b30987" alt=""
Push
Local
data:image/s3,"s3://crabby-images/e92bf/e92bf53d1fb5ff4ec903e2bb704a81f80c47f5f3" alt=""
Cloud
data:image/s3,"s3://crabby-images/454ed/454ed05ab865e192251fd569deb16ea2dbc075d1" alt=""
data:image/s3,"s3://crabby-images/e6b4a/e6b4a7cd2a170f3f7eb49c548454ba4f99b30987" alt=""
data:image/s3,"s3://crabby-images/e6b4a/e6b4a7cd2a170f3f7eb49c548454ba4f99b30987" alt=""
Local 1
data:image/s3,"s3://crabby-images/454ed/454ed05ab865e192251fd569deb16ea2dbc075d1" alt=""
Local 2
Merge
data:image/s3,"s3://crabby-images/e6b4a/e6b4a7cd2a170f3f7eb49c548454ba4f99b30987" alt=""
關於「Git」這個名字
- 可以發音唸出的隨機三個字母組合,而且並未被實際用在任何 UNIX 指令上
- Global Information Tracker,如果想硬湊的話
- 愚蠢的、鄙視和卑鄙的,來自字典的解釋
關於「Git」和「Github」
- Github 是一個「線上軟體原始碼代管服務平台」,它會用 Git 來進行版本控制
- 還有其他和 Github 類似的平台,例如 gitlab, bitbucket 等等
data:image/s3,"s3://crabby-images/146a1/146a15333c17d4e9e7e281a4b04efd43ef53a357" alt=""
基本概念
區域劃分
data:image/s3,"s3://crabby-images/6303b/6303be24da3bf1f45c7f146d9071eb138dd77932" alt=""
檔案狀態
data:image/s3,"s3://crabby-images/da00d/da00df3d15969a9a71a74d9eb8e5bf1fe386a408" alt=""
基本流程
- 修改 / 新增檔案
- 移到 staging area
- 存到 repository
誰說資訊系不用GUI?
終端機
data:image/s3,"s3://crabby-images/20b44/20b44e07ed6258502be7b7258fe0c6a7675e79a0" alt=""
終端機
data:image/s3,"s3://crabby-images/84a1d/84a1d7c94599eacd53179baa8c39227240bc2c6a" alt=""
Sublime Merge
- 簡潔易懂
- 指令操作
- 功能齊全
- 顯示分支樹
- 免費
下載安裝
data:image/s3,"s3://crabby-images/7902d/7902d99adcf4b97f3952509167ffa60d11ec2220" alt=""
官網:https://www.sublimemerge.com
介面
data:image/s3,"s3://crabby-images/b34ba/b34ba65c809c5b170147422b57d5688e6132bee9" alt=""
操作
data:image/s3,"s3://crabby-images/7f9e3/7f9e395ad88955bcd12d24e9b2e719d3b9e9f541" alt=""
指令
data:image/s3,"s3://crabby-images/1e606/1e6067dfc1346d1c1e7439f38fbc982ebd9114fb" alt=""
Ctrl + p
Git 基礎
關於自己做事
基礎指令
初始設定
創建倉庫 / 環境設定
創建倉庫
- repository(repo)
- 以 資料夾 為單位
- 讓 git 知道這些內容要被管理
- 初始化 git
儲存位置
data:image/s3,"s3://crabby-images/2a7b2/2a7b26bdf23bc20aef348bb05de4428930b25d9d" alt=""
Sublime Merge
data:image/s3,"s3://crabby-images/3b0ce/3b0ce46005d1a6d39a6da56da8aa4b7a3d9900cc" alt=""
Sublime Merge
選單 > File > New Repository
data:image/s3,"s3://crabby-images/0fbcc/0fbcce5763ce61e9377c9c04a4ac099a88050797" alt=""
環境設定
- 記錄作者資訊
- 調整記錄資訊
- file mode
- 調整環境設置
- 編輯器
e.g. 作者資訊
data:image/s3,"s3://crabby-images/94e8b/94e8b42c1b3690d663f649d18bfa77eb791e144e" alt=""
data:image/s3,"s3://crabby-images/f3c6b/f3c6baae8e0632b7a09d49b9b5789525b413e9a8" alt=""
儲存位置
- Global
~/.gitconfig
-
影響所有 repo
- Local
-
<repo>/.git/config
-
目前所在 repo
-
儲存位置 - Global
data:image/s3,"s3://crabby-images/1844f/1844f8d06c780d9378ca23f6aadfe4794309bb0f" alt=""
儲存位置 - Local
data:image/s3,"s3://crabby-images/91c66/91c66148f8adb03502f397c1939f1e596222e4cd" alt=""
Sublime Merge
data:image/s3,"s3://crabby-images/ef5a2/ef5a20eaf58e8727188ee2cee511060c82e49cc8" alt=""
Sublime Merge
data:image/s3,"s3://crabby-images/0a7c8/0a7c8817d57bb0542a91343fedd3c1fada4632d5" alt=""
創建 commit
放上卡車 / 檢查卡車 / 送進倉庫
放上卡車
data:image/s3,"s3://crabby-images/6303b/6303be24da3bf1f45c7f146d9071eb138dd77932" alt=""
git add
用途?
- 掌控每個 commit 大小
- 避免把某些資訊丟進倉庫
- 為工作設定斷點
Sublime Merge
data:image/s3,"s3://crabby-images/c5d03/c5d032e736e7fda7715b22260ece23f075c61feb" alt=""
檢查卡車
data:image/s3,"s3://crabby-images/6303b/6303be24da3bf1f45c7f146d9071eb138dd77932" alt=""
內容資訊
data:image/s3,"s3://crabby-images/da00d/da00df3d15969a9a71a74d9eb8e5bf1fe386a408" alt=""
Sublime Merge
data:image/s3,"s3://crabby-images/28306/2830641c5969fdfbdb358c59dd4b6ee6941aec51" alt=""
modified
untracked
staged
紀錄「改變」
modified
staged
data:image/s3,"s3://crabby-images/e0fd0/e0fd093cb3a878da585e24abf129e08f15e7f550" alt=""
送進倉庫
data:image/s3,"s3://crabby-images/6303b/6303be24da3bf1f45c7f146d9071eb138dd77932" alt=""
git commit
Sublime Merge
data:image/s3,"s3://crabby-images/a29f4/a29f480b5d7282cd1bfe593507fe51bdfeb9ad68" alt=""
commit message
成功!
data:image/s3,"s3://crabby-images/de772/de772758b900fb40a0c23494a6be4f317af40574" alt=""
Demo & Lab
- 新增一個 repo
- 做好環境設定
- 新增檔案
a.txt
並放入卡車 - 把卡車送進倉庫
指令 ver.
git init / config / add / status / commit
Review
- 新增倉庫
git init
- 環境設定
git config
- 放上卡車
git add
- 檢查卡車
git status
- 送進倉庫
git commit
創建倉庫
$ git init
$ git init <repo_name>
環境設定
-
$ git config [--global] user.name "<your_name>"
-
$ git config [--global] user.email "<your_email>"
放上卡車
$ git add a.txt
$ git add *.c
$ git add .
$ git add --all
檢查卡車
data:image/s3,"s3://crabby-images/4df27/4df2762ad5a8c707cc6b796f7b714d0e859eabfc" alt=""
$ git status
modified
untracked
staged
送進倉庫
$ git commit
$ git commit -m "<message>"
組合技
$ git status
$ git commit
請養成 commit 前檢查 status 的好習慣
Demo
$
指令 time
復原操作
git restore / rm
unstaged
- 剛剛
git add
某些檔案 - 發現這個東西還不能丟到倉庫
- 從 staging area 移除
Sublime Merge
data:image/s3,"s3://crabby-images/43df3/43df35a009b86b329a125d0bdf743213c4a505fa" alt=""
unmodified
- 剛剛修改了一些檔案
- 測試時發現整陀爛光光
- 只能回復到上一個 commit 的狀態
Sublime Merge
data:image/s3,"s3://crabby-images/0d24c/0d24c5172363f800f849ca03f49458f5c4f93724" alt=""
Sublime Merge
data:image/s3,"s3://crabby-images/78ec2/78ec28bf5352f5b48fec1bff0c95095e66a53780" alt=""
Demo & Lab
Task: 創建並讓 b.txt
在 staged 跟 modified 間反覆橫跳移動
data:image/s3,"s3://crabby-images/69176/69176c296075c913ed15c883f29848dedc655a91" alt=""
指令 ver.
git restore / rm
unstage
$ git restore --staged a.txt
$ git restore --staged *.c
第一個 staged
- 新的 repo
- 新增
a.txt
並放入 staging area - 從 staging area 移除
指令
data:image/s3,"s3://crabby-images/a61f1/a61f1aed103a69367bd5aab507049d9a8e762804" alt=""
哭阿
data:image/s3,"s3://crabby-images/a61f1/a61f1aed103a69367bd5aab507049d9a8e762804" alt=""
git restore
-
git restore
是以 commit 為基底進行復原 - 沒有半個 commit 當然會出事情
怎麼辦
- 下載 Sublime Merge
- 重開一個 repo
- 不管了直接 commit
- ...?
指令
data:image/s3,"s3://crabby-images/0e597/0e597581cfe5ccdfd5e2bc76d4383953a02c4a9f" alt=""
指令
$ git rm --cached a.txt
$ git rm --cached *.c
git rm
- 把檔案從 tracked 直接刪除
-
--cached:
改成變為 untracked
unmodified
$ git restore a.txt
複習
$ git restore [--staged] <file>
$ git rm [--cached] <file>
太多好難記?
data:image/s3,"s3://crabby-images/ca681/ca6819929217a31423b5f52fca8c88012fd70a24" alt=""
Demo
$
指令 time
其他指令
操作歷史
查看歷史 / 刪除歷史 / 暫存桌面 / 回到過去
查看歷史
- commit 順序
- 作者與內容
- 搜尋功能
- 作者
- 時間
- 檔案
Sublime Merge
data:image/s3,"s3://crabby-images/d2da5/d2da54692c045f2ebb82f63c0410f747e771af25" alt=""
Sublime Merge
data:image/s3,"s3://crabby-images/b9975/b99757dbd87c97af65b345b389035816ce83a434" alt=""
進階搜尋
data:image/s3,"s3://crabby-images/bae96/bae96f9ca6ca4c9996afe52bf6953522eafe03c2" alt=""
刪除歷史
- 不小心手殘把錯的東西
git add
- 不小心手殘把錯的東西
git commit
- 完蛋zzz
刪除歷史
- 創造一個新的 commit 修改
-
git log
長很醜 - 強迫症發作
- 做事沒效率
- 專案沒進度
人生沒希望
Sublime Merge
data:image/s3,"s3://crabby-images/0986c/0986c71a44d203adc4489aed499b68a3cace7d3c" alt=""
刪除 commit
-
soft
:改成 staged -
mixed
:改成 modified -
hard
:直接刪除
mixed
結果
data:image/s3,"s3://crabby-images/d77d3/d77d331a93269785529fedb1cf77c3c005fa7c14" alt=""
WARNING !
git reset --hard
是一個十分危險的指令,一旦使用所有進度都會直接消失。請勿搭配酒精使用,並確認無誤再下達。
回到過去
- 做了一堆 commit
- 突然想回到過去看看以前的狀態
Sublime Merge
data:image/s3,"s3://crabby-images/3c7c5/3c7c584500165db0eefe81798b21187dde5d3b61" alt=""
data:image/s3,"s3://crabby-images/96bfa/96bfa733b7800f37dbe3a1a2e9620a12e01b9600" alt=""
結果
data:image/s3,"s3://crabby-images/986eb/986eb427c91f17a7e21092517d5e227b781e123f" alt=""
data:image/s3,"s3://crabby-images/0d75e/0d75ef5a665b9b815b7576883897476d2ef14509" alt=""
桌面不乾淨
data:image/s3,"s3://crabby-images/b5185/b51856c294b886097e8660f1fa21d044eb316c3f" alt=""
暫存桌面
- 現在的檔案不足以 commit
- 但又需要清空以放下其他東西
- 暫時存檔之後取出
Sublime Merge
data:image/s3,"s3://crabby-images/0f1ad/0f1ad2dd128fc4b18634fcb3c5366210b7fae90a" alt=""
只會儲存 staged
- 先
stage all
- 再
stash
結果
data:image/s3,"s3://crabby-images/03fe3/03fe3f808bf8684009a59002da533027b80249a4" alt=""
data:image/s3,"s3://crabby-images/b0cde/b0cde6e7f4600194a0ed9cd3956509a345f613a1" alt=""
取回暫存 - pop
data:image/s3,"s3://crabby-images/70f70/70f705f468a1fd9870d03f20318ae668e4c460d5" alt=""
回歸正題
- 原本想要回到過去被卡住
-
stash
清空桌面 -
pop
之後取回
回到過去
data:image/s3,"s3://crabby-images/e3a28/e3a287b0f5b1e596965599a7d2fc5cfa8de5edec" alt=""
結果
data:image/s3,"s3://crabby-images/dead1/dead13acf9a9b10d445f4209504d82a0cae4a244" alt=""
data:image/s3,"s3://crabby-images/61855/61855bdcc097f6c33475132df1414827dcab6c43" alt=""
回到現在
data:image/s3,"s3://crabby-images/d9f3b/d9f3b9231d4ee70d38cdc026a68d67344ace61fa" alt=""
修改歷史?
- 先回到過去
- 在歷史紀錄上做不同的修改
- 甚至做出 commit...
結果
data:image/s3,"s3://crabby-images/4a5a5/4a5a52c3d1ff8fe9b321faf1bd1a9308a344321f" alt=""
Detached HEAD
- 目前 HEAD 所在的位置不屬於任何 branch
- 離開後不容易找到這些 commits
- 謹慎使用
先不要碰
Demo
把桌面清空 回到第一個 commit 再回來並回復桌面
Review
- 查看歷史
git log
- 刪除歷史
git reset
- 暫存桌面
git stash
- 回到過去
git checkout
查看歷史
$ git log
data:image/s3,"s3://crabby-images/287c9/287c9ea083c2a07b90e937c32a303d845840721a" alt=""
刪除歷史
$ git reset <commit_ID>
$ git reset --soft <commit_ID>
- $ git reset --hard <commit_ID>
回到過去 - 未清空
data:image/s3,"s3://crabby-images/e7d9c/e7d9c0caf91307b610f0810ff894587cdcdbacd5" alt=""
暫存桌面
$ git add --all
$ git stash
只會暫存 staging area
取回暫存
$ git stash pop
data:image/s3,"s3://crabby-images/d4996/d499650e98d6a19e808ce861a29b884cdbe4027a" alt=""
回到過去
$ git checkout <commit_ID>
data:image/s3,"s3://crabby-images/1b5b4/1b5b46e40ac23eb61161bd1229c4158d855ccc7e" alt=""
回到現在
$ git checkout master
data:image/s3,"s3://crabby-images/113b4/113b45d87f1515d6a588b3f958f6d5fdf0f9fffb" alt=""
git checkout
- git checkout 其實更常用於切換分枝(branch)
- 其中 master 就是分支名稱
- 有時視情況會是 main / 其他名稱
- 下半堂會更詳細介紹
Demo
$
指令 time
隱藏資訊
.gitignore
隱藏資訊
- 機密檔:
secret_key.txt
- 執行檔:
a.exe a.out
- 系統檔:
.vscode/ .DS_Store
- 暫存檔:
a.txt~ b.txt.swp
小心再小心
- 每次都不要 add 到那些檔案
- 不能使用
git add --all
-
git status
很醜 - 強迫症發作
- ...
.gitignore
- 名稱就叫 "
.gitignore"
- 告訴 git 哪些檔案 / 資料夾底下不要紀錄
- 不會被
git status / add
看到
.gitignore
secret_key.txt
*.exe
*.out
.vscode/
.DS_Store
*~
*.swp
Sublime Merge
data:image/s3,"s3://crabby-images/8290e/8290e19d5f78ba2031254fa23f12dd92f55cf62b" alt=""
data:image/s3,"s3://crabby-images/b812a/b812ad147af46c59945ff7280ec165211c7398b9" alt=""
指令
$ vim .gitignore
data:image/s3,"s3://crabby-images/3a438/3a4383cfd74ea438ecbb60dc113689d0a007804a" alt=""
結果
data:image/s3,"s3://crabby-images/774aa/774aad4902e12cbd7744c40ba8baa864eede6468" alt=""
影響範圍
-
.gitignore
可以每個資料夾底下都放一個 - 影響範圍會因為位置而有所不同
- 若撰寫前該檔案已 tracked 則不受影響
- 可透過
git rm --cached <file>
變回 untracked
Demo & Lab
Task: 撰寫 .gitignore
使 git 無法追蹤 secret_key.txt
Q & A
$ git checkout break_time !
Git 進階
關於分工合作
分工 = branch
合作 = merge
A
B
版本
+修改
=版本
b
A
B
E
C
D
F
G
branch
merge
g \(\approx\) e + f
b
e
c
f
d
g
branch:分化出不同的分支完成不同的部分
merge:將不同分支的內容整合在一起
branch
some rules:
- 一個 branch 指向一個 commit
- 同個 commit 可以有很多分支,但 Git 同時只會關注(checkout)一個分支
- 每次 commit ,當前的 branch 也會跟著往後移動
A
git checkout main
B
main
A
git checkout main
git commit -m "C"
B
C
main
A
git checkout main
git commit -m "C"
git commit -m "D"
B
C
main
D
A
git checkout main
git commit -m "C"
git commit -m "D"
git checkout <commit ID of B>
B
C
HEAD
main
D
A
git checkout main
git commit -m "C"
git commit -m "D"
git checkout <commit ID of B>
git checkout -b dev
B
C
dev
main
D
A
git checkout main
git commit -m "C"
git commit -m "D"
git checkout <commit ID of B>
git checkout -b dev
git commit -m "E"
B
C
dev
main
D
E
A
git checkout main
git commit -m "C"
git commit -m "D"
git checkout <commit ID of B>
git checkout -b dev
git commit -m "E"
git checkout main
B
C
dev
main
D
E
A
git checkout main
git commit -m "C"
git commit -m "D"
git checkout <commit ID of B>
git checkout -b dev
git commit -m "E"
git checkout main
git commit -m "F"
B
C
dev
main
D
E
F
A
git checkout main
git commit -m "C"
git commit -m "D"
git checkout <commit ID of B>
git checkout -b dev
git commit -m "E"
git checkout main
git commit -m "F"
git checkout dev
B
C
dev
main
D
E
F
A
git checkout main
git commit -m "C"
git commit -m "D"
git checkout <commit ID of B>
git checkout -b dev
git commit -m "E"
git checkout main
git commit -m "F"
git checkout dev
git commit -m "G"
B
C
dev
main
D
E
F
G
merge
some rules:
- A 併進 B ≠ B 併進 A
- A 併進 B:將 branch A 的資訊帶回 B
- 「合併」就可能發生「衝突」(兩邊的意見不合)
main
main
dev
main
dev
main
dev
merge!
int stack[100], top = 0;
void push(int x){
}
int pop(int x){
}
int stack[100], top = 0;
void push(int x){
stack[top++] = x;
}
int pop(){
}
int stack[100], top = 0;
void push(int x){
}
int pop(){
return stack[--top];
}
int stack[100], top = 0;
void push(int x){
stack[top++] = x;
}
int pop(){
return stack[--top];
}
data:image/s3,"s3://crabby-images/ffb0a/ffb0a512fffa515fad5c2af5e57719d44d45393c" alt=""
A
git checkout main
B
C
dev
main
D
E
F
D
F
A
git checkout main
git merge dev
B
C
dev
main
D
E
F
G
G
Merge this?
data:image/s3,"s3://crabby-images/c8209/c8209316bbc91f8bd965df6294d790afae1776c4" alt=""
main
dev
main
dev
?
conflict!
conflict
-
兩邊對相同的地方做不同的修改
-
Git 無法自動判斷哪邊該留下
-
需要手動合併
int gcd(int a, int b){
<<<<<<< HEAD
while(b != 0){
int tmp = b;
b = a % b;
a = tmp;
}
return a;
=======
return a % b == 0
? b
: gcd(b, a % b);
>>>>>>> rec
}
int gcd(int a, int b){
}
int gcd(int a, int b){
while(b != 0){
int tmp = b;
b = a % b;
a = tmp;
}
return a;
}
int gcd(int a, int b){
return a % b == 0
? b
: gcd(b, a % b);
}
int gcd(int a, int b){
return a % b == 0
? b
: gcd(b, a % b);
}
<<<<<<< HEAD
合併前的檔案內容
=======
合併進來的分支的檔案內容
>>>>>>> 合併進來的分支名稱
Git 進階
關於異地同步
Sync strategy
push & pull
data:image/s3,"s3://crabby-images/6a2fe/6a2fe19d807b72acdbedc1e4112d610cf38f873b" alt=""
data:image/s3,"s3://crabby-images/6a2fe/6a2fe19d807b72acdbedc1e4112d610cf38f873b" alt=""
data:image/s3,"s3://crabby-images/6a2fe/6a2fe19d807b72acdbedc1e4112d610cf38f873b" alt=""
pull
push
「嘿,現在的版本樹長怎樣」
「嘿,我做了某某某修改」
git pull
for synchronization
data:image/s3,"s3://crabby-images/6a2fe/6a2fe19d807b72acdbedc1e4112d610cf38f873b" alt=""
A
B
dev
main
C
D
E
F
A
B
dev
main
C
D
E
A
B
dev
main
C
D
E
F
data:image/s3,"s3://crabby-images/6a2fe/6a2fe19d807b72acdbedc1e4112d610cf38f873b" alt=""
git fetch
A
B
dev
main
C
D
E
F
A
B
dev
main
C
D
E
A
B
origin/dev
origin/main
C
D
E
F
data:image/s3,"s3://crabby-images/6a2fe/6a2fe19d807b72acdbedc1e4112d610cf38f873b" alt=""
git fetch
git merge
A
B
dev
main
C
D
E
F
A
B
dev
main
C
D
E
A
B
origin/dev
origin/main
C
D
E
F
= git pull
git pull
some rules:
- (fetch)會將所有遠端的 commit 都抓下來
- (merge)只會 merge 當前的 branch
- 有 merge 就可能有 conflict 發生
data:image/s3,"s3://crabby-images/6a2fe/6a2fe19d807b72acdbedc1e4112d610cf38f873b" alt=""
A
B
dev
main
C
D
E
F
A
B
dev
main
C
D
E
G
data:image/s3,"s3://crabby-images/6a2fe/6a2fe19d807b72acdbedc1e4112d610cf38f873b" alt=""
origin/dev
A
B
main
dev
C
D
E
F
main
C
dev
G
A
B
origin/main
C
D
E
F
data:image/s3,"s3://crabby-images/6a2fe/6a2fe19d807b72acdbedc1e4112d610cf38f873b" alt=""
A
B
main
C
D
E
F
main
C
G
A
B
origin/main
C
D
E
F
H
dev
dev
origin/dev
git push
for contribution
data:image/s3,"s3://crabby-images/6a2fe/6a2fe19d807b72acdbedc1e4112d610cf38f873b" alt=""
A
B
dev
main
C
D
E
A
B
dev
main
C
D
E
F
A
B
dev
main
C
D
E
F
data:image/s3,"s3://crabby-images/6a2fe/6a2fe19d807b72acdbedc1e4112d610cf38f873b" alt=""
A
B
main
C
D
E
A
B
dev
main
C
D
E
F
git push origin dev
A
B
dev
main
C
D
E
F
git push <remote> <branch>
some rules:
- 將本地的 branch 併進遠端的 branch
- 只會動到指定的 branch
- 必須是 fast-forward,否則會被 reject
- remote 永遠是對的
data:image/s3,"s3://crabby-images/6a2fe/6a2fe19d807b72acdbedc1e4112d610cf38f873b" alt=""
git push origin dev
A
B
main
C
D
E
F
dev
A
B
main
dev
C
D
E
G
To github.com:username/reponame.git
! [rejected] dev -> dev (fetch first)
error: failed to push some refs to 'github.com:username/reponame.git'
hint: Updates were rejected because the remote contains work that you do
hint: not have locally. This is usually caused by another repository pushing
hint: to the same ref. You may want to first integrate the remote changes
hint: (e.g., 'git pull ...') before pushing again.
hint: See the 'Note about fast-forwards' in 'git push --help' for details.
E
F
origin/dev
G
dev
H
E
F
origin/dev
G
dev
E
F
dev
pull
merge
push
E
G
dev
data:image/s3,"s3://crabby-images/6a2fe/6a2fe19d807b72acdbedc1e4112d610cf38f873b" alt=""
E
F
G
dev
H
> git push origin dev
To github.com:username/reponame.git
! [rejected] dev -> dev (fetch first)
error: failed to push some refs to 'github.com:username/reponame.git'
hint: Updates were rejected because the remote contains work that you do
hint: not have locally. This is usually caused by another repository pushing
hint: to the same ref. You may want to first integrate the remote changes
hint: (e.g., 'git pull ...') before pushing again.
hint: See the 'Note about fast-forwards' in 'git push --help' for details.
> git pull
...
Auto-merging <filename>
CONFLICT (content): Merge conflict in <filename>
Automatic merge failed; fix conflicts and then commit the result.
> git add .
> git commit -m "merged"
> git push
Demo & Lab
Task:合作開發 1A2B 猜數字遊戲
background
-
三個人一組
-
透過 Git / GitHub 協作開發猜數字遊戲
For each group...
A
B
C
A
B
C
A
B
C
3.
新增協作者
Repo 主頁
> Settings
> Collaborators
> Add people
A
B
C
local
ssh-keygen
cat ~/.ssh/id_rsa.pub
data:image/s3,"s3://crabby-images/000b3/000b36e7946f32c6244e262794f402d9da038b60" alt=""
github
4.
把 repo clone 到本機上
github.com/username/repo-name
A
B
C
data:image/s3,"s3://crabby-images/e59ee/e59eea4c1e2332264e1af72561e4c7efd2c5045a" alt=""
git@github.com:username/repo-name.git
cd /path/to/any/directory
git clone <repo_url>
cd repo-name
5.
A 先打個基本設定
A
B
C
*.exe
.gitignore
#include<stdio.h>
#include<stdlib.h>
int main()
{
}
main.c
git add .
git commit -m "template"
git push origin main
5
main
6.
BC 把 A 的修改 pull 下來
A
B
C
git pull
5
main
7.
B 強迫症發作,改了 A 的 coding style
A
B
C
#include<stdio.h>
#include<stdlib.h>
int main(){
}
main.c
git add .
git commit -m "coding style!!"
git push origin main
5
main
7
main
8.
C 寫了一些遊戲基本流程
A
B
C
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
#define len 4
void generate(char *s){
// TODO
}
void input(char *s){
// TODO
}
void response(char *s, char *a){
// TODO
}
int main()
{
char ans[10000], str[10000];
generate(ans);
while(true){
input(str);
response(str, ans);
}
printf("You won!\n");
}
main.c
git add .
git commit -m "game flow"
git push origin main
5
7
main
To github.com:username/repo-name.git
! [rejected] main -> main (fetch first)
error: failed to push some refs to 'github.com:username/repo-name.git'
hint: Updates were rejected because the remote contains work that you do
hint: not have locally. This is usually caused by another repository pushing
hint: to the same ref. You may want to first integrate the remote changes
hint: (e.g., 'git pull ...') before pushing again.
hint: See the 'Note about fast-forwards' in 'git push --help' for details.
void response(char *s, char *a){
// TODO
}
<<<<<<< HEAD
int main()
{
char ans[10000], str[10000];
generate(ans);
=======
int main(){
>>>>>>> <commit ID>
while(true){
input(str);
response(str, ans);
git pull
void response(char *s, char *a){
// TODO
}
int main(){
char ans[10000], str[10000];
generate(ans);
while(true){
input(str);
response(str, ans);
git add .
git commit -m "merge!"
git push origin main
data:image/s3,"s3://crabby-images/34f5e/34f5e6a3e04e8e7a8891b3b920bfd21fd0beb5a4" alt=""
C
5
7
origin/main
8
main
8
5
7
origin/main
8
main
5
7
main
5
7
8
8
main
pull
merge
push
5
8
main
9.
分工:
A 維護 main
B 處理 input
C 處理 output
A
B
C
5
7
8
8
main
input
output
git pull
git checkout -b "input"
B
C
git pull
git checkout -b "output"
10.
A 簡單加上了分隔線
A
B
C
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
#define len 4
void generate(char *s){
// TODO
}
void input(char *s){
// TODO
}
void response(char *s, char *a){
// TODO
}
int main(){
char ans[10000], str[10000];
generate(ans);
while(true){
printf("==========\n");
input(str);
response(str, ans);
}
printf("You won!\n");
}
main.c
git add .
git commit -m "add separator"
git push origin main
5
7
8
8
main
input
output
10
main
11.
B 完成了數字生成和使用者輸入
A
B
C
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
#include<time.h>
#include<string.h>
#define len 4
bool valid(char *s){
if(strlen(s) != len) return false;
for(int i = 0; i < len; i++){
for(int j = 0; j < len; j++){
if(i != j && s[i] == s[j]){
return false;
}
}
}
return true;
}
void generate(char *s){
printf("Set the answer (0 for auto generation): ");
scanf("%s", s);
if(s[0] == '0' && strlen(s) == 1){
srand(time(0));
while(true){
for(int i = 0; i < len; i++){
s[i] = '0' + rand() % 10;
}
s[len] = '\0';
if(valid(s)){
break;
}
}
}else{
if(!valid(s)){
printf("Invalid input\n");
generate(s);
}else{
printf("\033[1A\x1b[2K");
}
}
}
void input(char *s){
printf("Guess a 4-digit number: ");
scanf("%s", s);
}
void response(char *s, char *a){
// TODO
}
int main(){
char ans[10000], str[10000];
generate(ans);
while(true){
input(str);
if(!valid(str)){
printf("Invalid input\n");
}else{
response(str, ans);
}
}
printf("You won!\n");
}
main.c
git add .
git commit -m "finish input"
git push origin input
5
7
8
8
input
output
10
main
11
input
12.
C 完成回覆函數和勝利判斷
A
B
C
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
#include<string.h>
#define len 4
void generate(char *s){
// TODO
}
void input(char *s){
// TODO
}
bool win(char *s, char *a){
return strcmp(s, a) == 0;
}
void response(char *s, char *a){
int cnt_a = 0, cnt_b = 0;
for(int i = 0; i < len; i++){
if(s[i] == a[i]){
cnt_a++;
}else{
for(int j = 0; j < len; j++){
if(s[i] == a[j]){
cnt_b++;
break;
}
}
}
}
if(cnt_a) printf("%dA", cnt_a);
if(cnt_b) printf("%dB", cnt_b);
printf("\n");
}
int main(){
char ans[10000], str[10000];
generate(ans);
while(true){
input(str);
if(win(str, ans)){
break;
}else{
response(str, ans);
}
}
printf("You won!\n");
}
main.c
git add .
git commit -m "finish output"
git push origin output
5
7
8
8
10
main
11
input
12
output
output
13.
身為 main branch 的維護者, A 要負責將 feature branches 合併進 main
A
B
C
git pull
git merge origin/input
5
7
8
8
10
main
11
input
12
output
13
main
14.
身為 main branch 的維護者, A 要負責將 feature branches 合併進 main
A
B
C
git merge origin/output
5
7
8
8
10
11
input
12
output
13
main
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
<<<<<<< HEAD
#include<time.h>
=======
>>>>>>> output
#include<string.h>
#define len 4
bool valid(char *s){
if(strlen(s) != len) return false;
for(int i = 0; i < len; i++){
for(int j = 0; j < len; j++){
if(i != j && s[i] == s[j]){
return false;
}
}
}
return true;
}
void generate(char *s){
printf("Set the answer (0 for auto generation): ");
scanf("%s", s);
if(s[0] == '0' && strlen(s) == 1){
srand(time(0));
while(true){
for(int i = 0; i < len; i++){
s[i] = '0' + rand() % 10;
}
s[len] = '\0';
if(valid(s)){
break;
}
}
}else{
if(!valid(s)){
printf("Invalid input\n");
generate(s);
}else{
printf("\033[1A\x1b[2K");
}
}
}
void input(char *s){
printf("Guess a 4-digit number: ");
scanf("%s", s);
}
bool win(char *s, char *a){
return strcmp(s, a) == 0;
}
void response(char *s, char *a){
int cnt_a = 0, cnt_b = 0;
for(int i = 0; i < len; i++){
if(s[i] == a[i]){
cnt_a++;
}else{
for(int j = 0; j < len; j++){
if(s[i] == a[j]){
cnt_b++;
break;
}
}
}
}
if(cnt_a) printf("%dA", cnt_a);
if(cnt_b) printf("%dB", cnt_b);
printf("\n");
}
int main(){
char ans[10000], str[10000];
generate(ans);
while(true){
printf("==========\n");
input(str);
<<<<<<< HEAD
if(!valid(str)){
printf("Invalid input\n");
=======
if(win(str, ans)){
break;
>>>>>>> output
}else{
response(str, ans);
}
}
printf("You won!\n");
}
data:image/s3,"s3://crabby-images/4019e/4019ef0077a670cb3598cff02b3bfd41f92c1daa" alt=""
data:image/s3,"s3://crabby-images/f977a/f977a54ff5286b09873391559955c0e169e9962b" alt=""
14.
身為 main branch 的維護者, A 要負責將 feature branches 合併進 main
A
B
C
git add .
git commit -m "merge output"
5
7
8
8
10
11
input
12
output
13
main
14
main
5
7
8
8
10
11
input
12
output
13
14
main
git push origin main
gcc game.c -o game.exe
./game.exe
-
更系統化的分工合作
-
一個團隊的紀律
-
avoid conflicts!!
5
7
8
8
10
11
input
12
output
13
14
main
大家都想在 main branch 上做事
\(\Rightarrow\) 衝突多、merge 多、歷史記錄亂
每個人在自己的 branch 上做事
彼此不互相干擾
- 同一個 branch 盡量少人操作
- main 作為正式版本
- merge 進主要分支前要 code review
data:image/s3,"s3://crabby-images/7009d/7009da379ad111aeb97de237031fbbc9c3153823" alt=""
Feature Branching
data:image/s3,"s3://crabby-images/19f03/19f03eb4da8b2fceee2aa0abae68d2f7b4baa011" alt=""
後話
git rebase
A
B
C
A
B
C
A
B
C
B
C
A
B
C
git merge
git rebase
A
B
C
A
B
C
A
B
C
B
D
A
B
C
git merge
git rebase
git merge
git rebase
A
B
C
B
D
A
B
C
- commit D 幾乎沒有任何意義
- merge 會產生多餘的 commit
- merge 會造成非線性的歷史紀錄
- rebase 就是將 C 的修改重新作用在 B 上面
- rebase 兩邊的 commit 鏈越長,就容易產生越多的 conflict
- rebase 對版本樹結構影像較大
Git
By thomaswang2003
Git
- 567