Git学习笔记
安装配置
- 安装git
1 | brew install git |
- 配置git
1 | git config --global user.name "Your Name" |
因为Git是分布式版本控制系统,所以,每个机器都必须自报家门:你的名字和Email地址。你也许会担心,如果有人故意冒充别人怎么办?这个不必担心,首先我们相信大家都是善良无知的群众,其次,真的有冒充的也是有办法可查的。
注意
git config
命令的--global
参数,用了这个参数,表示你这台机器上所有的Git仓库都会使用这个配置,当然也可以对某个仓库指定不同的用户名和Email地址。
创建版本库
- 什么是版本库呢?版本库又名仓库(Repository),你可以简单理解成一个目录,这个目录里面的所有文件都可以被Git管理起来,每个文件的修改、删除,Git都能跟踪,以便任何时刻都可以追踪历史,或者在将来某个时刻可以“还原”。
1 | cd /Users/wly |
然后就会显示
1 | /Users/wly/learngit |
可以看到这个pwd
显示当前目录
git init
可以把这个目录变成Git可以管理的仓库
1 | 提示: 使用 'master' 作为初始分支的名称。这个默认分支名称可能会更改。要在新仓库中 |
空的仓库(empty Git repository)
多了一个
.git
的目录- 这个目录是Git来跟踪管理版本库的
- ==不要轻易改动w==
如果你没有看到
.git
目录,那是因为这个目录默认是隐藏的,用ls -ah
命令就可以看见这样就看到了
把文件添加到版本库
注意喵:
- 首先这里再明确一下,所有的版本控制系统,其实只能跟踪文本文件的改动,比如TXT文件,网页,所有的程序代码等等,Git也不例外。版本控制系统可以告诉你每次的改动,比如在第5行加了一个单词“Linux”,在第8行删了一个单词“Windows”。而图片、视频这些二进制文件,虽然也能由版本控制系统管理,但没法跟踪文件的变化,只能把二进制文件每次改动串起来,也就是只知道图片从100KB改成了120KB,但到底改了啥,版本控制系统不知道,也没法知道。
- 不幸的是,Microsoft的Word格式是二进制格式,因此,版本控制系统是没法跟踪Word文件的改动的,前面我们举的例子只是为了演示,如果要真正使用版本控制系统,就要以纯文本方式编写文件。
- 因为文本是有编码的,比如中文有常用的GBK编码,日文有Shift_JIS编码,如果没有历史遗留问题,强烈建议使用标准的UTF-8编码,所有语言使用同一种编码,既没有冲突,又被所有平台所支持。
- 使用Windows的童鞋要特别注意:千万不要使用Windows自带的记事本编辑任何文本文件。原因是Microsoft开发记事本的团队使用了一个非常弱智的行为来保存UTF-8编码的文件,他们自作聪明地在每个文件开头添加了0xefbbbf(十六进制)的字符,你会遇到很多不可思议的问题,比如,网页第一行可能会显示一个“?”,明明正确的程序一编译就报语法错误,等等,都是由记事本的弱智行为带来的。建议你下载Visual Studio Code代替记事本,不但功能强大,而且免费!
- 编写一个txt
vi readme.txt
esc退出编辑:wq!
保存退出readme.txt
如下:
1 | Git is a version control system. |
- 一定要放在learngit的目录下面,子目录也行。
- 接下来把这个文件放到git仓库当中去:
添加add
1 | git add readme.txt |
提交commit
- -m是本次提交的说明,就相当于写了注释,这次提交是为了干什么
1 | git commit -m "wrote a readme file" |
1 | [master(根提交) 6e8be91] wrote a readme file |
git commit
命令执行成功后会告诉你,1 file changed
:1个文件被改动(我们新添加的readme.txt文件);2 insertions
:插入了两行内容(readme.txt有两行内容)。
另外为什么Git添加文件需要add
,commit
一共两步呢?因为commit
可以一次提交很多文件,所以你可以多次add
不同的文件,比如:
1 | $ git add file1.txt |
时光机穿梭 git stutus 随时查看工作区状态
修改readme.txt后
1 | git status |
git status
命令可以让我们时刻掌握仓库当前的状态,上面的命令输出告诉我们,readme.txt
被修改过了,但还没有准备提交的修改。- 看看具体修改了什么内容
git diff xx
1 | git diff readme.txt |
- 然后
git add
git status
git commit -m "xx"
git status
可以添加 确认状态 再上传commit
1 | git commit -m "add distributed" |
1 | git status |
版本回退
再次修改readme.txt
1 | git add readme.txt |
可以理解下git的commit作为虚拟机的快照
游戏存档之类的git log
查看最近提交的日志
1 | git log |
- 如果觉得输出的内容太多,可以这么修改
--pretty=oneline
1 | git log --pretty=oneline |
- 和SVN不一样,Git的
commit id
不是1,2,3……递增的数字,而是一个SHA1计算出来的一个非常大的数字,用十六进制表示,而且你看到的commit id
和我的肯定不一样,以你自己的为准。为什么commit id
需要用这么一大串数字表示呢?因为Git是分布式的版本控制系统,后面我们还要研究多人在同一个版本库里工作,如果大家都用1,2,3……作为版本号,那肯定就冲突了。
把readme.txt版本回退
每提交一个新版本,实际上Git就会把它们自动串成一条时间线。
首先,Git必须知道当前版本是哪个版本,在Git中,用HEAD
表示当前版本,也就是最新的提交,上一个版本就是HEAD^
,上上一个版本就是HEAD^^
,当然往上100个版本写100个^
比较容易数不过来,所以写成HEAD~100
。
- 回退
git reset
1 | git reset --hard HEAD^ |
- 参数
--hard
参数有啥意义?--hard
会回退到上个版本的已提交状态- 而
--soft
会回退到上个版本的未提交状态 --mixed
会回退到上个版本已添加但未提交的状态- 现在,先放心使用
--hard
。
看readme.txt的内容,回退了。
其实还可以继续回退到之前的版本
但是查看版本库
1 | git log |
之前的记录已经找不到了捏
办法其实还是有的,只要上面的命令行窗口还没有被关掉,你就可以顺着往上找啊找啊,找到那个append GPL
的commit id
是f1db449a9da173805d7170f023cf5f4f86e895a8
,于是就可以指定回到未来的某个版本:
1 | git reset --hard f1db4 |
再看readme.txt的文本
Git在内部有个指向当前版本的HEAD
指针
git版本回退速度非常快
当你回退版本的时候,Git仅仅是把HEAD从指向append GPL
然后顺便把工作区的文件更新了。所以你让HEAD
指向哪个版本号,你就把当前版本定位在哪。
- 如果找不到新版本的commit id怎么办
git reflog
用来记录你的每一次命令
1 | git reflog |
工作区和暂存区
Git和其他版本控制系统如SVN的一个不同之处就是有暂存区的概念。
工作区(Working Directory)
在电脑里可以看到的目录,learngit文件夹就是一个工作区。
版本库(Repository)
工作区有一个隐藏目录.git
,这个不算工作区,而是Git的版本库。
Git的版本库里存了很多东西,其中最重要的就是称为stage(或者叫index)的暂存区,还有Git为我们自动创建的第一个分支master
,以及指向master
的一个指针叫HEAD
。
文件添加到版本库
git add
把文件添加进去,添加到暂存区stagegit commit
提交更改,实际上就是把暂存区的所有内容提交到当前分支master
因为我们创建Git版本库时,Git自动为我们创建了唯一一个master
分支,所以,现在,git commit
就是往master
分支上提交更改。
==需要提交的文件修改通通放到暂存区,然后,一次性提交暂存区的所有修改。==
进行一些修改
readme.txt
加一个LICENSE文本文件
1 | git status |
使用git add
把readme.txt和LICENSE都添加到stage
使用git commit
把暂存区的东西都提交到分支上
1 | git status |
一旦提交后,如果你又没有对工作区做任何修改,那么工作区就是“干净”的
变成了这样喵
管理修改
Git比其他版本控制系统设计得优秀,因为Git跟踪并管理的是修改,而非文件。
修改readme.txt
1 | cat readme.txt |
add
1 | git add readme.txt |
再次修改readme.txt commit
1 | git commit -m "git tracks changes" |
查看状态
1 | git status |
==你看,我们前面讲了,Git管理的是修改,当你用git add
命令后,在工作区的第一次修改被放入暂存区,准备提交,但是,在工作区的第二次修改并没有放入暂存区,所以,git commit
只负责把暂存区的修改提交了,也就是第一次的修改被提交了,第二次的修改不会被提交。==
用git diff HEAD -- readme.txt
命令可以查看工作区和版本库里面最新版本的区别
1 | git diff HEAD -- readme.txt |
第二次修改没提交
撤销修改
git checkout -- file
可以撤销工作区的修改,file是文件名
- 一种是
readme.txt
自修改后还没有被放到暂存区,现在,撤销修改就回到和版本库一模一样的状态; - 一种是
readme.txt
已经添加到暂存区后,又作了修改,现在,撤销修改就回到添加到暂存区后的状态。 - 让这个文件回到最近一次
git commit
或git add
时的状态。
注意: git checkout -- file
命令中的--
很重要,没有--
,就变成了“切换到另一个分支”的命令,我们在后面的分支管理中会再次遇到git checkout
命令。
命令git reset HEAD <file>
可以把暂存区的修改撤销掉(unstage),重新放回工作区git reset
命令既可以回退版本,也可以把暂存区的修改回退到工作区。当我们用HEAD
时,表示最新的版本。
删除文件
1 | rm test.txt |
1 | git status |
选择1 从版本库当中删除文件
git rm
删掉git commit
提交
选择2 删错了 但是版本库里还有 可以恢复
1 | git checkout -- test.txt |
注意
先手动删除文件,然后使用git rm <file>
和git add<file>
效果是一样的。
远程仓库
Git是分布式版本控制系统,同一个Git仓库,可以分布到不同的机器上。怎么分布呢?最早,肯定只有一台机器有一个原始版本库,此后,别的机器可以“克隆”这个原始版本库,而且每台机器的版本库其实都是一样的,并没有主次之分。
github
由于你的本地Git仓库和GitHub仓库之间的传输是通过SSH加密的,所以,需要一点设置:
- 第1步:创建SSH Key。在用户主目录下,看看有没有.ssh目录,如果有,再看看这个目录下有没有
id_rsa
和id_rsa.pub
这两个文件,如果已经有了,可直接跳到下一步。如果没有,打开Shell(Windows下打开Git Bash),创建SSH Key:
1 | ssh-keygen -t rsa -C "youremail@example.com" |
在用户主目录里找到.ssh
目录,里面有id_rsa
和id_rsa.pub
两个文件,这两个就是SSH Key的秘钥对,id_rsa
是私钥,不能泄露出去,id_rsa.pub
是公钥,可以放心地告诉任何人。
- 第2步:登陆GitHub,打开“Account settings”,“SSH Keys”页面,然后,点“Add SSH Key”,填上任意Title,在Key文本框里粘贴
id_rsa.pub
文件的内容。
注意
为什么GitHub需要SSH Key呢?因为GitHub需要识别出你推送的提交确实是你推送的,而不是别人冒充的,而Git支持SSH协议,所以,GitHub只要知道了你的公钥,就可以确认只有你自己才能推送。
当然,GitHub允许你添加多个Key。假定你有若干电脑,你一会儿在公司提交,一会儿在家里提交,只要把每台电脑的Key都添加到GitHub,就可以在每台电脑上往GitHub推送了。
最后友情提示,在GitHub上免费托管的Git仓库,任何人都可以看到喔(但只有你自己才能改)。所以,不要把敏感信息放进去。
如果你不想让别人看到Git库,有两个办法,一个是交点保护费,让GitHub把公开的仓库变成私有的,这样别人就看不见了(不可读更不可写)。
另一个办法是自己动手,搭一个Git服务器,因为是你自己的Git服务器,所以别人也是看不见的。
添加远程库
本地git仓库 远程github也有git仓库 两个仓库进行同步
github创建一个库
在本地的github仓库里执行以下命令
1 | git remote add origin git@github.com:auberginewly/learngit.git |
让本地库关联远程库
添加后,远程库的名字就是origin
,这是Git默认的叫法,也可以改成别的,但是origin
这个名字一看就知道是远程库。
把本地库所有内容推送到远程库上
1 | git push -u origin master |
把本地库的内容推送到远程,用git push
命令,实际上是把当前分支master
推送到远程。由于远程库是空的,我们第一次推送master
分支时,加上了-u
参数,Git不但会把本地的master
分支内容推送的远程新的master
分支,还会把本地的master
分支和远程的master
分支关联起来,在以后的推送或者拉取时就可以简化命令。
after
从现在起,只要本地作了提交,就可以通过命令:
1 | git push origin master |
把本地master
分支的最新修改推送至GitHub
删除远程库(解除关联)
1 | git remote -v |
可以查看远程库信息
1 | git remote rm origin |
删除远程库 实际上是解除了本地库和远程库绑定的联系
但是要删除在github上的远程库的时候 还是需要在github上进行操作
从远程克隆库
1 | git clone xxx |
Git支持多种协议,默认的git://
使用ssh
,但也可以使用https
等其他协议。
使用https
除了速度慢以外,还有个最大的麻烦是每次推送都必须输入口令,但是在某些只开放http
端口的公司内部就无法使用ssh
协议而只能用https
。ssh
最快
eg
1 | git clone https://github.com/auberginewly/learngit.git |
分支管理
创建与合并分支
理解分支
每次提交,Git都把它们串成一条时间线,这条时间线就是一个分支。截止到目前,只有一条时间线,在Git里,这个分支叫主分支,即master
分支。HEAD
严格来说不是指向提交,而是指向master
,master
才是指向提交的,所以,HEAD
指向的就是当前分支。
一开始的时候,master
分支是一条线,Git用master
指向最新的提交,再用HEAD
指向master
,就能确定当前分支,以及当前分支的提交点:
每次提交,master
分支都会向前移动一步,这样,随着你不断提交,master
分支的线也越来越长。
当我们创建新的分支,例如dev
时,Git新建了一个指针叫dev
,指向master
相同的提交,再把HEAD
指向dev
,就表示当前分支在dev
上:
你看,Git创建一个分支很快,因为除了增加一个dev
指针,改改HEAD
的指向,工作区的文件都没有任何变化!
不过,从现在开始,对工作区的修改和提交就是针对dev
分支了,比如新提交一次后,dev
指针往前移动一步,而master
指针不变:
假如我们在dev
上的工作完成了,就可以把dev
合并到master
上。Git怎么合并呢?最简单的方法,就是直接把master
指向dev
的当前提交,就完成了合并:
所以Git合并分支也很快!就改改指针,工作区内容也不变!
合并完分支后,甚至可以删除dev
分支。删除dev
分支就是把dev
指针给删掉,删掉后,我们就剩下了一条master
分支:
命令
创建并切换到dev分支
1 | git checkout -b dev |
相当于这两个命令的合并
1 | git branch dev |
然后可以使用git branch
查看命令
对readme.txt做出一些修改
添加
1 | Creating a new branch is quick. |
添加并提交
1 | git add readme.txt |
到这里对dev的工作就结束了,然后我们切换回master分支
1 | git checkout master |
然后查看readme.txt
1 | cat readme.txt |
添加的修改没了
因为那个提交是在dev
分支上,而master
分支此刻的提交点并没有变
我们把dev
分支的工作成果合并到master
分支上
1 | git merge dev |
git merge
命令用于合并指定分支到当前分支。合并后,再查看readme.txt
的内容,就可以看到,和dev
分支的最新提交是完全一样的。
注意到上面的Fast-forward
信息,Git告诉我们,这次合并是“快进模式”,也就是直接把master
指向dev
的当前提交,所以合并速度非常快。
当然,也不是每次合并都能Fast-forward
,还有其他方式的合并。
合并完成后,就可以放心地删除dev
分支:
1 | git branch -d dev |
然后查看分支
1 | git branch |
只剩下master了
为什么要创建多个分支?
因为创建、合并和删除分支非常快,所以Git鼓励你使用分支完成某个任务,合并后再删掉分支,这和直接在master
分支上工作效果是一样的,但过程更安全。
补充:switch
我们注意到切换分支使用git checkout <branch>
,而前面讲过的撤销修改则是git checkout -- <file>
,同一个命令,有两种作用,确实有点令人迷惑。
实际上,切换分支这个动作,用switch
更科学。因此,最新版本的Git提供了新的git switch
命令来切换分支,创建 并 切换 到新的dev
分支,可以使用:
1 | git switch -c dev |
直接切换到 已有 的master
分支,可以使用:
1 | git switch master |
switch比checkout更好理解
解决冲突
当Git无法自动合并分支时,就必须首先解决冲突。解决冲突后,再提交,合并完成。
解决冲突就是把Git合并失败的文件手动编辑为我们希望的内容,再提交。
准备好新的feature1分支
1 | git switch -c feature1 |
先修改readme.txt最后一行为
1 | Creating a new branch is quick AND simple. |
添加并提交变化
1 | git add readme.txt |
切换到master分支
1 | git switch master |
Git还会自动提示我们当前master
分支比远程的master
分支要超前1个提交
在master
分支上把readme.txt
文件的最后一行改为:
1 | Creating a new branch is quick & simple. |
提交
1 | git add readme.txt |
现在master分支和feature1分支都有了各自的提交
likethis
这种情况下,Git无法执行“快速合并”,只能试图把各自的修改合并起来,但这种合并就可能会有冲突,比如:
1 | git merge feature1 |
Git告诉我们,readme.txt
文件存在冲突,必须手动解决冲突后再提交。git status
也可以告诉我们冲突的文件:
1 | git status |
我们可以直接查看readme.txt的内容:
1 | cat readme.txt |
Git用<<<<<<<
,=======
,>>>>>>>
标记出不同分支的内容,我们修改如下后保存:
1 | Git is a distributed version control system. |
再提交:
1 | git add readme.txt |
现在,master
分支和feature1
分支变成了下图所示:
用带参数的git log
也可以看到分支的合并情况:
1 | git log --graph --pretty=oneline --abbrev-commit |
最后,删除feature1
分支:
1 | git branch -d feature1 |
用git log --graph
命令可以看到分支合并图。
关于一个参数的解释
这条命令:
1 | git log --graph --pretty=oneline --abbrev-commit |
我们可以拆成三部分来解释,每个参数都有它的作用:
1. --graph
作用:用 ASCII 字符画出分支和合并的图示,比如:
1
2
3
4
5* commit A
|\
| * commit B
|/
* commit C这个图示能直观地看到分支的结构,比如哪儿合并了、哪儿分支了。
2. --pretty=oneline
作用:让每个 commit 显示成一行,只包含:
- 提交的哈希值(默认是完整的,但配合
--abbrev-commit
就是简写的) - 提交信息(commit message)
显示示例:
1
2a1b2c3d 修复登录页面bug
4d5e6f7 添加用户注册功能- 提交的哈希值(默认是完整的,但配合
3. --abbrev-commit
- 作用:缩短提交哈希值,只显示前几位(一般是前 7 位)
- 比如完整哈希是:使用这个参数后只显示:
1
a1b2c3d4e5f67890123456789abcdef12345678
1
a1b2c3d
总结一句话:
这条命令会以「一行一个提交、缩短哈希值、带有分支图」的形式,显示 Git 提交历史,简洁直观,非常适合快速查看分支结构和提交信息。
分支管理策略
通常,合并分支时,如果可能,Git会用Fast forward
模式,但这种模式下,删除分支后,会丢掉分支信如果要强制禁用Fast forward
模式,Git就会在merge时生成一个新的commit,这样,从分支历史上就可以看出分支信息。
创建并切换dev分支
1 | git switch -c dev |
修改readme.txt并将其提交到一个新的commit上
1 | git add readme.txt |
切换回master
1 | git switch master |
禁用fast forward合并dev分支
1 | git merge --no-ff -m "merge with no-ff" dev |
请注意--no-ff
参数,表示禁用Fast forward
,因为本次合并要创建一个新的commit,所以加上-m
参数,把commit描述写进去。
用git log
看看分支历史
1 | git log --graph --pretty=oneline --abbrev-commit |
合并分支时,加上--no-ff
参数就可以用普通模式合并,合并后的历史有分支,能看出来曾经做过合并,而fast forward
合并就看不出来曾经做过合并。
不使用Fast forward
模式,merge后就像这样
分支策略应用
在实际开发中,我们应该按照几个基本原则进行分支管理:
首先,master
分支应该是非常稳定的,也就是仅用来发布新版本,平时不能在上面干活;
那在哪干活呢?干活都在dev
分支上,也就是说,dev
分支是不稳定的,到某个时候,比如1.0版本发布时,再把dev
分支合并到master
上,在master
分支发布1.0版本;
你和你的小伙伴们每个人都在dev
分支上干活,每个人都有自己的分支,时不时地往dev
分支上合并就可以了。
所以,团队合作的分支看起来就像这样:
Bug分支
软件开发中会有bug的产生,git当中分支相当的强大,每个bug都可以通过一个新的临时分支来修复,修复后,合并分支,然后将临时分支删除。
如果当你接到一个修复一个代号101的bug的任务时,很自然地,你想创建一个分支issue-101
来修复它,但是,等等,当前正在dev
上进行的工作还没有提交,但是必须尽快修复bug,这个时候,Git还提供了一个stash
功能,可以把当前工作现场“储藏”起来,等以后恢复现场后继续工作:
1 | git stash |
这个时候查看状态就是干净的
1 | git status |
首先确定要在哪个分支上修复bug,假定需要在master
分支上修复,就从master
创建临时分支:
1 | git checkout master |
修改并提交
1 | git add readme.txt |
修改完,切换到master分支,并且删除issue-101分支:
1 | git switch master |
切换回dev干活
1 | git switch dev |
刚刚存的工作现场去哪了?用git stash list
命令看看:
1 | git stash list |
工作现场还在,Git把stash
内容存在某个地方了,但是需要恢复一下,有两个办法:
一是用git stash apply
恢复,但是恢复后,stash
内容并不删除,你需要用git stash drop
来删除;
另一种方式是用git stash pop
,恢复的同时把stash
内容也删了:
1 | git stash pop |
这个时候再查看,就没有任何stash暂存工作区的内容了
你可以多次stash
,恢复的时候,先用git stash list
查看,然后恢复指定的stash
,用命令:
1 | git stash apply stash@{0} |
在master
分支上修复了bug后,我们要想一想,dev
分支是早期从master
分支分出来的,所以,这个bug其实在当前dev
分支上也存在。那怎么在dev
分支上修复同样的bug?
复制3e0825f fix bug 101
这个提交所做的修改,并不是把整个master
分支merge过来。
1 | git cherry-pick 3e0825f |
这两个commit只是改动相同,但确实是两个不同的commit。用git cherry-pick
,我们就不需要在dev
分支上手动再把修bug的过程重复一遍。既然可以在master
分支上修复bug后,在dev
分支上可以“重放”这个修复过程,那么直接在dev
分支上修复bug,然后在master
分支上“重放”行不行?当然可以,不过你仍然需要git stash
命令保存现场,才能从dev
分支切换到master
分支。
Feature分支
软件开发中,总有无穷无尽的新的功能要不断添加进来。
添加一个新功能时,你肯定不希望因为一些实验性质的代码,把主分支搞乱了,所以,每添加一个新功能,最好新建一个feature分支,在上面开发,完成后,合并,最后,删除该feature分支。
1 | git switch -c xx |
多人协作
当你从远程仓库克隆时,实际上Git自动把本地的master
分支和远程的master
分支对应起来了,并且,远程仓库的默认名称是origin
。
要查看远程库的信息,用git remote
,或者git remote -v
1 | git remote |
上面显示了可以抓取和推送的origin
的地址。如果没有推送权限,就看不到push的地址。
推送分支
推送分支,就是把该分支上的所有本地提交推送到远程库。推送时,要指定本地分支,这样,Git就会把该分支推送到远程库对应的远程分支上:
1 | git push origin master |
1
1 | git push origin master |
2
1 | git push origin dev |
但是,并不是一定要把本地分支往远程推送,那么,哪些分支需要推送,哪些不需要呢?
master
分支是主分支,因此要时刻与远程同步;dev
分支是开发分支,团队所有成员都需要在上面工作,所以也需要与远程同步;- bug分支只用于在本地修复bug,就没必要推到远程了,除非老板要看看你每周到底修复了几个bug;
- feature分支是否推到远程,取决于你是否和你的小伙伴合作在上面开发。
总之,就是在Git中,分支完全可以在本地自己藏着玩,是否推送,视你的心情而定!
抓取分支
多人协作时,大家都会往master
和dev
分支上推送各自的修改。
模拟多人协作,在同一台电脑另外一个目录下clone
但是只能看到默认分支
我这里是main
要在dev
分支上开发,就必须创建远程origin
的dev
分支到本地
在dev
上继续修改,然后,时不时地把dev
分支push
到远程
1 | git add env.txt |
如果这个时候另外一个人也想推送,先用git pull
把最新的提交从origin/dev
抓下来,然后,在本地合并,解决冲突,再推送。
1 | git branch --set-upstream-to=origin/dev dev # 指定本地dev与远程联系 |
如果有合并冲突,手动解决之后在push
参考解决冲突那一小节
Rebase
多人在同一个分支上协作时,很容易出现冲突。即使没有冲突,后push的不得不先pull,在本地合并,然后才能push成功,然后分支就会乱成一坨狗屎。
Git有一种称为rebase
的操作,有人把它翻译成“变基”。
1 | git rebase |
可以让原来分叉的提交变成一条直线,Git把本地的提交“挪动”了位置,这就是rebase操作的特点:把分叉的提交历史“整理”成一条直线,看上去更直观,缺点是本地的分叉提交已经被修改过,远程分支的提交历史也是一条直线。
标签管理
创建标签
Git中打标签先切换到需要打标签的分支上
1 | git checkout xxx |
然后打标签
1 | git tag <name> |
使用git tag
可以查看所有的标签
默认的标签打在最新的commit上面
如果需要打给其他的commit的标签
找到历史提交的commit id,然后打上就可以了
1 | git tag <tag name> <commit id> |
一个commit可以对应多个tags
标签不是按时间顺序列出,而是按字母排序的,可以用git show <tagname>
查看标签信息。
还可以创建带有说明的标签,用-a
指定标签名,-m
指定说明文字,eg:
1 | git tag -a v0.1 -m "version 0.1 released" 1094adb |
用命令git show <tagname>
可以看到说明文字
注意
标签总是和某个commit挂钩。如果这个commit既出现在master分支,又出现在dev分支,那么在这两个分支上都可以看到这个标签。
操作标签
删除
1 | git tag -d <tag name> |
因为创建的标签都只存储在本地,不会自动推送到远程。所以,打错的标签可以在本地安全删除。
如果要推送某个标签到远程,使用命令git push origin <tagname>
或者,一次性推送全部尚未推送到远程的本地标签
1 | git push origin --tags |
如果标签已经推送到远程,要删除远程标签就麻烦一点,先从本地删除:
1 | git tag -d <tag name> |
然后从远程删除 也是push 格式如下
1 | git push origin :refs/tags/<tag name> |
关于使用github
- 在GitHub上,可以任意Fork开源仓库;
- 自己拥有Fork后的仓库的读写权限;
- 可以推送pull request给官方仓库来贡献代码。
关于使用gitee
// 待更新 可以同时连接多个远程仓库
自定义Git
修改git的颜色让其更醒目
1 | git config --global color.ui true |
给git配置名字和邮箱
1 | # Set global username |
--global
全局
忽略特殊文件
有些时候,你必须把某些文件放到Git工作目录中,但又不能提交它们,比如保存了数据库密码的配置文件,每次git status
都会显示Untracked files ...
,添加.gitignore
文件,然后把要忽略的文件名填进去,Git就会自动忽略这些文件。
注意喔
.gitignore
文件本身应该提交给Git管理,这样可以确保所有人在同一项目下都使用相同的.gitignore
文件。
不需要从头写.gitignore
文件,GitHub已经为我们准备了各种配置文件,只需要组合一下就可以使用了。所有配置文件可以直接在线浏览:GitHub/gitignore
忽略文件的原则是:
- 忽略操作系统自动生成的文件,比如缩略图等;
- 略编译生成的中间文件、可执行文件等,也就是如果一个文件是通过另一个文件自动生成的,那自动生成的文件就没必要放进版本库,比如Java编译产生的
.class
文件;- 自己的带有敏感信息的配置文件,比如存放口令的配置文件。
假设你在Windows下进行Python开发,Windows会自动在有图片的目录下生成隐藏的缩略图文件,如果有自定义目录,目录下就会有Desktop.ini
文件,因此你需要忽略Windows自动生成的垃圾文件:
1 | # Windows: |
然后,继续忽略Python编译产生的.pyc
、.pyo
、dist
等文件或目录:
1 | # Python: |
加上你自己定义的文件,最终得到一个完整的.gitignore
文件,内容如下:
1 | # Windows: |
然后把.gitignore
也提交到Git,检验.gitignore
的标准是git status
命令是不是说working directory clean
。
有时候,想添加一个文件到Git,但发现添加不了,原因是这个文件被.gitignore
忽略了:
1 | git add App.class |
如果你确实想添加该文件,可以用-f
强制添加到Git:
1 | git add -f App.class |
或者你发现,可能是.gitignore
写得有问题,需要找出来到底哪个规则写错了,可以用git check-ignore
命令检查(别忘了-v哦)
1 | git check-ignore -v App.class |
Git会告诉我们,.gitignore
的第3行规则忽略了该文件,于是我们就可以知道应该修订哪个规则。
还有些时候,当我们编写了规则排除了部分文件时:
1 | # 排除所有.开头的隐藏文件: |
但是我们发现.*
这个规则把.gitignore
也排除了,并且App.class
需要被添加到版本库,但是被*.class
规则排除了。
虽然可以用git add -f
强制添加进去,但有强迫症还是希望不要破坏.gitignore
规则,这个时候,可以添加两条例外规则:
1 | # 排除所有.开头的隐藏文件: |
把指定文件排除在.gitignore
规则外的写法就是!
+文件名,所以,只需把例外文件添加进去即可。
另外,可以通过GitIgnore Online Generator在线生成
.gitignore
文件并直接下载。
还有一点
.gitignore
文件放哪放Git仓库根目录下,但其实一个Git仓库也可以有多个.gitignore
文件,.gitignore
文件放在哪个目录下,就对哪个目录(包括子目录)起作用。
就像这样
.gitignore
文件本身要放到版本库里,并且可以对.gitignore
做版本管理!
配置别名
我们只需要敲一行命令,告诉Git,以后st
就表示status
:
1 | git config --global alias.st status |
还有其他简写命令:
1 | $ git config --global alias.co checkout |
以后提交就可以简写
--global
参数是全局参数,也就是这些命令在这台电脑的所有Git仓库下都有用。
实用的别名配置
命令git reset HEAD file
可以把暂存区的修改撤销掉(unstage),重新放回工作区。既然是一个unstage操作,就可以配置一个unstage
别名:
1 | git config --global alias.unstage 'reset HEAD' |
当你敲入命令:
1 | git unstage test.py |
实际上Git执行的是:
1 | git reset HEAD test.py |
再者
配置一个git last
,让其显示最后一次提交信息:
1 | git config --global alias.last 'log -1' |
这样,用git last
就能显示最近一次的提交:
1 | git last |
可以丧心病狂地把lg
配置成:
1 | git config --global alias.lg "log --color --graph --pretty=format:'%Cred%h%Creset -%C(yellow)%d%Creset %s %Cgreen(%cr) %C(bold blue)<%an>%Creset' --abbrev-commit" |
配置文件
配置Git的时候,加上--global
是针对当前用户起作用的,如果不加,那只针对当前的仓库起作用。
配置文件放哪了?每个仓库的Git配置文件都放在.git/config
文件中:
1 | cat .git/config |
而当前用户的Git配置文件放在用户主目录下的一个隐藏文件.gitconfig
中:
1 | cat ~/.gitconfig |
别名就在[alias]
后面,要删除别名,直接把对应的行删掉即可。
配置别名也可以直接修改这个文件,如果改错了,可以删掉文件重新通过命令配置,或者直接删掉配置文件错误的那一行。
搭建git服务器
参考:https://blog.csdn.net/qq_47553403/article/details/120596743
虽然但是还没用到()
GitGui——使用SourceTree
// 待更新 图形化操作