如果把互聯網上的紛繁代碼比作一片海洋,那么git就是在這片海洋上航行的船只,正所謂“水可載舟,亦可覆舟”,git使用恰當可以遠征星辰,不然可能會墜入無窮無盡的代碼海洋無法自拔,書回正傳,我們的征途是星辰大海!
揚帆起航
git的下載安裝暫且不表,可參考網站https://git-scm.com/downloads,
git的安裝只是個入門條件,下面如何使用git命令控制代碼才是遠征的基礎,那么從何開始呢?讓我們一步步說起,
要想揚帆起航,首先得有艘帆船吧,我們要在本地建立一個專案,之后將本地專案初始化為git倉庫,可以使用
git init [--bare]
初始化本地專案,執行后會在當前執行目錄下生成.git檔案夾,這也就是我們的帆船了,[--bare]可選引數,可初始化裸倉庫,裸倉庫可以與原始碼專案分離,此時在當前目錄下不會生成.git檔案夾,而是直接生成.git檔案夾下的包括hooks、info、objects、refs共四個檔案夾和config、description、HEAD三個檔案,
在初始化裸倉庫后若想關聯原始碼,只需進入hooks目錄新建post-receive.sample檔案,并添加如下內容:
git --work-tree=<project-dir> --git-dir=<local-url> checkout -f
其中<project-dir>為本地專案檔案路徑,<local-url>為本地git倉庫路徑,
對于當前目錄下的.git檔案夾所代表的本地倉庫,不同的帆船有不同的配置,我們使用
git config –list
顯示當前git配置,如果內容過多可以使用上下鍵翻頁,查看結束按q退出即可,同時使用
git config –e [--global]
以vim形式編輯修改git組態檔,
或
git config [--global] user.name “name”
形式修改指定配置,其中[--global]可選,為全域配置,name為用戶名配置,
git儀表盤

圖 1 Git指令關系圖
建立git倉庫作為帆船后,就可以在當前目錄下任意使用git指令遨游了,我們需要先認識下git控制的命令儀表盤,
如圖1所示,git整艘船主要分為四大區域,包括Remote遠程倉庫,Repository本地倉庫,Index/Stage暫存區,Workspace作業區,不同區域之間可以使用相關指令進行代碼操作,在上節使用git init創建的檔案夾下,會生成名為.git的隱藏檔案夾,四大區域的配置都儲存在該檔案夾下,其中在遠程倉庫和本地倉庫中,儲存有不同的branch分支,作業區和暫存區的檔案只能針對某一分支進行修改提交操作,當然也可以使用分支操作指令對不同的分支進行增刪合并等操作,
針對一些倉庫的操作指令,使用前請務必保持頭腦清醒,不然你的一個蝴蝶煽動翅膀似的操作,可能會引發倉庫里的一場代碼風暴,造成不可挽回的結果,
根據不同命令操作區域,可以將git命令大致分為全域顯示資訊、作業區與遠程倉庫互動、作業區與本地倉庫互動、作業區與暫存區互動、暫存區與本地倉庫互動、本地倉庫與遠程倉庫互動、以及倉庫內部分支操作,下面對這些命令進行了粗略的統計介紹,詳細使用方式可參考git官方檔案介紹,如有不足還請補充,
顯示資訊
git help [command]
獲取命令的幫助資訊,
git status
顯示所有變更檔案
git log [--stat] [--graph]
顯示當前分支的版本資訊,[--stat]引數指定顯示commit發生變更的檔案,[--graph]引數以資料圖形式查看合并分支記錄,
git blame <file>
顯示檔案的每一行最后修改的版本和作者
git show [commit][:filename]
顯示某次提交的變更內容,其中[commit]為某次提交版本,也可在:后邊加引數[filename]指定查看某個檔案內容,
git diff [HEAD] [first-branch] [second-branch]
顯示檔案差異,無參時比較快取區和上一次commit的差異;[HEAD]為作業區與當前分支最新commit的差異;或者兩個分支之間的差異,
git reflog
顯示已執行過的所有git動作日志,
作業區與遠程倉庫
git pull <remote> <branch>
拉取遠程倉庫的變化,并與本地分支合并,<remote>為遠程倉庫名,<branch>為遠程倉庫中的某一分支名,
作業區與本地倉庫
git checkout [–b] <branch> [tag]
將暫存區切換到分支名,其中[-b]引數可選,當分支不存在時則創建,<branch >必須,為本地倉庫分支,[tag]可選,指定切換到倉庫分支中的某條標簽,不標注則默認為切換分支的最近一次提交,
作業區與暫存區
git add <dir>
添加指定目錄到暫存區,<dir>為添加路徑,允許多個,包括子目錄都會添加到暫存區中等待提交,
git rm [--cached] <file>
洗掉暫存區中的檔案,<file>為暫存區中要洗掉的檔案全路徑,[--cache]可選引數,只會停止繼續追蹤指定檔案,但該檔案目前仍然保留在暫存區,
git mv <file-old> <file-new>
修改暫存區中的檔案名,<file-old>為原檔案全路徑,<file-new>為修改后的檔案全路徑,
git tag
查看暫存區中的所有標簽資訊,
git tag –a <tag> [commit]
新建一個標簽,[tag]為標簽名;[commit]為指定的一次從暫存區到本地倉庫的提交中,默認為最新一次提交,
git tag –d [tag]
洗掉本地標簽,
暫存區與本地倉庫
git commit [--amend] [–m <message>] [file] [-a] [-v]
從暫存區提交到本地倉庫,其中[--amend]重做上次從本地專案到暫存區的commit,當代碼與上次提交相比無變化時使用,只修改上次commit的<message>內容;[-m]引數為提交資訊,<message>必寫且詳寫,以區別提交代碼的修改內容;[file]為指定暫存區中的檔案;[-a]可直接提交專案中的變化到本地倉庫,在沒有新增檔案時不需每次先git add提交到暫存區再提交到本地倉庫;[-v]可以在提交時顯示所有變化檔案的diff資訊,
git cherry-pick [commit]
選擇一個commit版本合并到當前分支,[commit]為暫存區中的commit版本,
本地倉庫與遠程倉庫
git remote [-v]
查看關聯的遠程倉庫資訊,[-v]可以查看詳細資訊,
git remote add <remote-name> <remote-url>
本地路徑關聯遠程倉庫,
<remote-name>必要引數,為遠程倉庫的名字,默認是origin,
<remote-url>必要引數,為遠程倉庫的地址,git服務器通常都是以.git結尾,
git remote remove <remote-name>
洗掉關聯的遠程倉庫,其中<remote-name>為遠程倉庫的名字,
git push [remote] [branch] [--force]
推送本地指定分支到遠程倉庫,[remote]為遠程倉庫名,[branch]為本地分支名,[--force]為強制推送本地到遠程,如有沖突則覆寫,
git fetch <remote>
將遠程倉庫拉到本地倉庫,<remote>為遠程倉庫名,
git clone <url> [name]
創建一個本地倉庫,<url>必須,可以是遠程git服務器上的倉庫,也可以是本地倉庫,[name]選填是創建的新倉庫名,默認與原倉庫名一致,
分支指令
git branch [-r] [-a]
查看分支資訊,無參只會查看本地倉庫所有分支,[-r]是遠程倉庫所有分支,[-a]則是包括本地和遠程倉庫所有的所有分支,
git branch [branch-name] [commit]
在本地倉庫新建分支,但暫存區仍然指向當前分支,其中[branch-name]為新建的本地倉庫分支;[commit]可將分支指向指定commit版本,
git branch –track [local-branch-name] [remote-branch]
新建一個分支,并連接指定的遠程分支,其中[local-branch-name]為本地倉庫新建分支,[remote-branch]為遠程倉庫分支,
git branch –set-upstream [local-branch] [remote-branch]
連接本地倉庫分支與遠程倉庫分支,其中[local-branch]為本地倉庫已有分支,[remote-branch]為遠程倉庫分支,
git branch –d [branch]
洗掉本地倉庫分支,[branch]為本地倉庫中的已有分支,
git branch -m [branch-old] [branch-new]
修改本地倉庫分支,其中[branch-old]為原分支,[branch-new]為改名后的分支,
git branch –dr [remote-branch]
洗掉遠程倉庫分支,[remote-branch]為遠程倉庫中的分支,不推薦使用,如果遠程倉庫未更新,可能會執行失敗,推薦使用git push origin-delete [remote-branch],
git merge <local-branch>
合并指定分支到當前分支,<local-branch>為本地倉庫中的已有分支,
git rebase <remote-branch>
將當前分支的提交復制到指定的遠程分支上,<remote-branch>為指定遠程倉庫中的已有分支,
git reset [--mixed|--soft|--hard] [commit]
重置倉庫索引,重置一旦清空后的內容不會在倉庫歷史版本中留下歷史記錄,[--mixed]為默認引數,重置后只在作業區保留原節點修改檔案,清空暫存區和本地倉庫并均恢復到指定重置節點;[--soft]為軟重置,重置后在作業區和暫存區均保留原節點修改檔案,清空本地倉庫并恢復到指定重置節點;[--hard]為硬重置,重置后均不會保留原節點修改檔案,[commit]為要重置的節點號,
git revert [commit]
還原檔案到之前修改提交節點時,會在倉庫歷史版本中留下歷史記錄,[commit]為要還原的節點號,
常規操作
git這艘大船雖然功能繁雜,但是用起來是有章可循的,入門之后就駕駛下這艘大船來試試吧,
一般git有三種作業流程,包括Git flow,專案存在兩個長期分支(主分支master和開發分支develop),適用于基于版本發布的普通專案;Github flow,只有一個長期分支(主分支master),適用于持續發布的小型專案;Gitlab flow,專案存在多個長期分支,其中主分支master是其他所有分支的上游,只有上游分支采納的代碼才能應用到其下游分支,適用于長期維護的大型專案,
圖2展示了一次專案git流程演變程序,在master主分支上有Tag1-Tag4四次代碼更新,其中基于Tag2對應的版本1號創建了新的branch1分支,新分支創建后自動生成了版本2號并打上了Tag2-1標簽,之后master主分支和branch1分支都同時進行了代碼演變,在branch1分支提交的版本4號及標簽Tag2-3之后,branch1分支合并到了master主分支,合并前master主分支位于版本5號,合并后可能重新生成版本6號,并打上新的tag4標簽,之后master主分支修改提交為版本7號,而branch1分支則停留在tag2-3標簽的位置處,

圖 2 Git流程示意圖
在這份專案流程中,分支創建之后,可能在不同的階段修改提交檔案,根據對檔案的讀取權限范圍,我們可以形象地將這些階段劃分為三種身份型別,船長、水手和游客,船長身份,作為專案管理者,主要負責遠程倉庫和本地倉庫之間的分支操作,協調分支沖突;水手身份,作為專案貢獻者,主要負責某一分支的迭代更新;游客身份,作為專案使用者,只是訪問使用倉庫及其分支內容,不能提交任何修改,同一人在專案的不同階段可以是其中任意一種身份,下面以這三種身份為維度簡單介紹下使用到的相關git指令步驟,并輔以示意圖方式直觀解釋git指令執行前后git專案變化,
專案使用者-游客
作為git專案的游客,當然只能將專案從遠程倉庫拉取到本地使用,期間除了切換倉庫分支外不會涉及其他遠程操作,
git pull origin master
拉取遠程倉庫origin的master分支到本地,
git checkout branch1 tag1
切換到分支branch1,

專案貢獻者-水手
作為專案的水手,除了可以使用游客的功能指令外,還會涉及到修改作業區檔案,并將作業區檔案提交到暫存區和倉庫等任務,通常水手只需要維護倉庫中的某一條分支并只對該分支負責,因此水手更注重作業區的代碼檔案修改作業,
git pull origin dev:branch1
拉取遠程倉庫origin的dev分支到本地,并與本地branch1分支合并,作業區中檔案即顯示branch1分支,可在作業區做檔案修改操作,
git add .
在作業區的檔案修改之后,可先添加當前目錄所有檔案到暫存區,之后可繼續修改作業區其他檔案,也可將暫存區檔案提交到本地倉庫,
git rm –cached file
針對作業區編譯生成的配置file檔案,一般不需提交到倉庫,可使用該命令將file檔案從暫存區洗掉并停止后續追蹤,另外一種添加忽略檔案的方式,在.git檔案夾的同級目錄下新建.gitignore檔案,在該檔案中根據規則增加要忽略的檔案路徑,之后將該檔案提交到本地倉庫中,
git commit –m “commit message 1”
在確保作業區的所有修改檔案均已提交到暫存區后,便將暫存區的修改提交到本地倉庫,同時附帶當次提交資訊,每次提交都會在本地倉庫生成一個新的提交commit版本號,由于提交的commit版本號是冗長的sha1碼,所以為了方便后期溯源,通常會在主要的commit號版本上再打一個鮮明的標簽以作標記,

git tag tag1 1
在提交的commit版本號為1的節點上打標簽,打上標簽后的commit號便可使用簡短的標簽名tag1訪問,以便后期對該節點溯源,

git push origin branch1:dev –tags
將本地branch1分支及相關標簽推送到遠程倉庫origin的dev分支,如果本地branch1分支已經與遠程dev分支建立追蹤關系,也可直接使用
git push origin branch1 --tags
指令,
git branch –set-upstream-to=origin/dev branch1
設定本地倉庫的branch1分支與遠程倉庫origin的dev分支的追蹤關系,通常從遠程分支pull到本地的分支都已經建立了追蹤關系,不需要手動修改,
git branch –vv
查看本地分支及追蹤的遠程分支資訊,
專案管理者-船長
作為專案的船長,自然擁有整個git這艘大船的專案所有權限,除了使用水手的操作指令外,另需完成分支增刪合并等任務,通常船長是專案倉庫的創建者,負責管理維護倉庫的各分支關系,對專案的整個倉庫負責,因此相較于水手,船長更注重倉庫的分支管理相關作業,
git clone https://github.com/xxx.git -b dev
克隆遠程倉庫的dev分支到本地,默認會將檔案更新到本地倉庫建立的同名dev分支,
git checkout –b branch1 origin/dev
在本地倉庫新建branch1分支,與遠程倉庫origin中的dev分支對應,并在本地切換到branch1分支,如果不指定遠程倉庫及分支資訊origin/dev,則默認從本地倉庫dev分支創建,至此可以切換為水手身份,從該分支更新代碼,并將修改檔案提交到該branch1分支,
git add .
git commit –m “modify file commit”
在完成對作業區檔案的修改之后,使用水手身份將作業區的修改提交到本地branch1分支,
git checkout dev
切換到本地倉庫的dev主分支,作為本地倉庫與遠程倉庫代碼合并的操作分支,
git fetch origin dev
拉取遠程倉庫origin中的dev分支到本地倉庫當前dev分支
git pull
將本地倉庫dev分支的檔案修改合并到作業區,
git merge branch1
將本地倉庫的branch1分支合并到當前dev分支,如果當前dev分支與branch1分支有沖突,需要根據沖突檔案提示分別修改,之后再重新合并該分支,
git checkout branch1 build/files
或者只將branch1分支的build/files目錄下所有檔案合并到本地倉庫的當前dev分支,同樣需要做沖突處理,
git push origin dev:dev –tags
向遠程倉庫origin中的dev分支并推送本地倉庫dev分支,推送時如果確認以本地分支覆寫遠程分支,則可使用
git push —force origin dev:dev –tags
強制推送,最后如果想洗掉遠程分支,有以下兩條指令
git push origin –delete dev
git push origin :dev
這兩種方式都可以洗掉指定的遠程倉庫origin中的dev分支,
應急預案
上面的圖2Git流程圖簡單涉及了一次版本演變程序中的一些git資訊,包括分支切換,打標簽等,看上去簡單易懂,而實際我們作業中駕船航行時卻并不總是風平浪靜,
通常出現的緊急情況需要修改分支版本,包括變基、還原、重置等操作,針對不同場景需要選擇不同的操作方式,
變基
git rebase master
變基會將當前分支的修改檔案復制到master分支,同時創建新的commit版本號并修改專案的歷史記錄,當master分支已經更新,且確認當前分支與master分支沒有沖突,那可以使用變基以便當前分支獲取master分支的更新,

重置
git reset 1
將HEAD重置到歷史提交版本1的狀態,還原倉庫和暫存區的檔案與提交版本1一致,作業區維持修改檔案狀態,

還原
git revert 1
重新創建一次提交版本節點,檔案狀態與歷史提交版本1一致,作業區、暫存區與倉庫均保持一致,提交版本2相對于提交版本1新增了file.txt檔案,執行該指令后,在提交版本3中將恢復到提交版本1的狀態,因此提交版本3相對于提交版本2則洗掉了file.txt檔案,

強制遠程覆寫本地
git fetch –all
拉取遠程所有倉庫到本地倉庫,作業區不會有任何合并更新,
git reset –hard origin/dev
把作業區HEAD指向最新的遠程倉庫origin中的dev版本,
待補充
除此之外在駕馭git這艘大船時肯定還會出現各種意外,屆時將酌情補充,
轉載請註明出處,本文鏈接:https://www.uj5u.com/qita/6645.html
標籤:其他
