git,一个分布式的版本管理工具。主要用处:版本管理、协作开发。
常见版本管理工具:
VSS ------ Visual Source Safe
CVS ------ Concurrent Versions System
SVN ------ CollabNet Subversion
GIT
GIT安装:下载安装文件:Git-2.41.0-64-bit.exe
data:image/s3,"s3://crabby-images/f264a/f264af15df455ea9e44516b92bf3af0c8f8c6f12" alt=""
运行:
Git是分布式版本控制系统。(另一种是集中式版本控制系统)
分布式版本控制系统没有"中央服务器",每台电脑上都有一个完整的版本库。
就是说,一开始,每台电脑上都有一个一样的版本库,然后每个人不需要联网,就可以做自己的版本控制,添加、修改、删除文件等操作,操作保存在自己本地的版本库中,经过一段时间后,再指定一个统一的地方,大家都把本地的版本库合并到这个统一的地方,实际上就是集中式版本系统的中央服务器,合并中有可能出现各种问题,如冲突等,解决完这些冲突,就形成一个大家都认可的版本,每个人再将这个版本的库下载到本地,再次基础上再次进行开发,循环往复,做到版本控制。
一、版本库创建
版本库又名仓库(repository),创建版本库,需要在一个合适的地方,创建一个空目录:
md e:\learngit
cd learngit
git init
Windows下:
data:image/s3,"s3://crabby-images/8ddd4/8ddd4ab2c0dc38cc528162a61278eb8d421b4ede" alt=""
centos6下:
data:image/s3,"s3://crabby-images/1c9f6/1c9f6879a2882ebea3873a0dc7a2029214c3c34b" alt=""
windows下可以使用git bash,环境就是模拟的linux:
data:image/s3,"s3://crabby-images/3b1bf/3b1bf5975d89fb844580defebc770e1e864b599e" alt=""
windows下可以使用图形界面:即Git GUI
data:image/s3,"s3://crabby-images/84ee2/84ee240dfd96debeb0091ae95f2130d3076d66a6" alt=""
创建仓库后,会创建.git子目录及相关文件、子目录:
data:image/s3,"s3://crabby-images/880d3/880d3733fe84c1d70220640b69c8ff0cec132c38" alt=""
二、使用
1、添加文件到仓库:git add filename
需要在learngit目录下创建文件,这个是工作目录,其他地方git是无法管理的
data:image/s3,"s3://crabby-images/1731f/1731fb13a65660a6362990961a1244d3bca823ce" alt=""
data:image/s3,"s3://crabby-images/0763c/0763c01b4c170d5d32385431832777eae4597736" alt=""
2、提交文件到仓库:git commit -m "说明信息"
data:image/s3,"s3://crabby-images/f8139/f8139b33be0ddc7c43b71cb91b98f1a50a87e316" alt=""
data:image/s3,"s3://crabby-images/772e9/772e950f3a769064b6154551611d693d2f3586c8" alt=""
第一次提交,出现提示信息,在linux下,默认使用你的登录名和hostname作为user.name和user.email,可以使用git config --global user.name "USERNAME" 和 git config --global user.email "EMAIL"来设置全局的用户名和邮箱地址,来标识提交文件的主人,不带--global,则只在本仓库有效。Windows中,这两项配置文件为C:\Users\Administrator\.gitconfig :
data:image/s3,"s3://crabby-images/786fe/786fe4575e8ec6467ce905b6147afd035b92171c" alt=""
windows下:
data:image/s3,"s3://crabby-images/61ede/61edeeb6ab31f1f49adb8a9621b678e00d43f818" alt=""
配置信息及其保存位置:
git config -l 是三个信息的汇总,分别是:
1).仓库级别 local。 2).用户级别global。 3).系统级别system。
优先级仓库级别最高,用户级别次之,系统级别最低。
1)、仓库级别:git config --local -l
data:image/s3,"s3://crabby-images/4fa81/4fa8151f6a3716862e59d2cfb7e66573ccca8701" alt=""
配置文件在仓库的.git目录下的config文件:
data:image/s3,"s3://crabby-images/b7ef0/b7ef03a4b0128045428fca4824151512bbc555ac" alt=""
2)、用户级别global、 git config --global -l
data:image/s3,"s3://crabby-images/8a1dc/8a1dca4bd43bc7e93de3a263419ea2c7c1a21f11" alt=""
配置文件在C:\Users\Administrator\.gitconfig
3)、系统界别system. git config --system -l
data:image/s3,"s3://crabby-images/cf3f9/cf3f987bc27fbf21e98e0e4f67820bdbcf563076" alt=""
配置文件在git安装目录下的etc中的gitconfig文件,如:D:\Program Files\Git\etc\gitconfig
data:image/s3,"s3://crabby-images/d717b/d717b72dec558e2d851c168e76d3cba5753b18ae" alt=""
在centos6.7中
1)、仓库级别:git config --local -l 配置文件在仓库目录下:.git/config
2)、用户级别global、 git config --global -l 配置文件在用户家目录下的.gitconfig
3)、系统界别system. git config --system -l 配置文件在/etc/gitconfig
3、git status :掌握仓库的当前状态:
当修改了某个文件,而又没有add,更没有commit时,仓库中的文件就与工作区的文件不同了,此时使用gitstatus查看:
data:image/s3,"s3://crabby-images/4f88d/4f88d984ca91284b3616ed9ab117fdff34624978" alt=""
git add后
data:image/s3,"s3://crabby-images/975cf/975cfaf33d31b9683b90b73bd5b5b8a722874e73" alt=""
提交后,再次修改:使用git diff a1.txt查看不同:
data:image/s3,"s3://crabby-images/632a2/632a214f3cd0a337fd11e7c1f10c2786790319ba" alt=""
4、git log 查看提交日志
data:image/s3,"s3://crabby-images/ded1d/ded1d4bdb4dbe0f189bed333c7478560c0685eda" alt=""
黄色的字符串,是commit id,即版本号。简洁版的显示
data:image/s3,"s3://crabby-images/74491/744919ad5a9e40e504f21ee97bfc06f6e1e8e399" alt=""
5、git reset 版本回退(或叫版本回滚),恢复到以前的某个版本
Git必须知道当前版本是哪个版本,Git中,用HEAD表示当前版本,上一个版本使用HEAD^表示,上上一个版本用HEAD^^,版本多了,使用HEAD~#,如HEAD~10,回退到往前第10个版本。
data:image/s3,"s3://crabby-images/45a2a/45a2a47ec60cc26d2c4c43d81bf6ae9098ef17e8" alt=""
data:image/s3,"s3://crabby-images/f2721/f2721657f54b8abe7510bc0a91a0604c231b4143" alt=""
如果要再回到第四次的版本,需要先找到第四次的commitid,方法是使用git reflog
git log无法显示回退前的版本信息。
data:image/s3,"s3://crabby-images/021c1/021c176f01255128d8ae502b5a2e4441b3bab0d0" alt=""
这里看到readme.txt-4的commitid,使用如下命令:
git reset --hard commitid
data:image/s3,"s3://crabby-images/45fec/45fec8c38f77f57146f7b412e68dc8321885c489" alt=""
data:image/s3,"s3://crabby-images/fed24/fed24cc979b7118798ee4c8a6abcaa63ec3c0c95" alt=""
data:image/s3,"s3://crabby-images/d22ee/d22ee607315ecc257083d1b55fb63629be4f7c91" alt=""
6、工作区和暂存区
工作区(working Directory):就是电脑中能看到的目录,比如这里的e:\learngit
版本库(Repository):工作区中有一个隐藏目录".git",这个不算工作区,而是Git版本库。
版本库中存了很多东西,其中最重要的就是称为stage(或叫index)的暂存区,Git自动创建的第一个分支master,以及指向master的一个指针叫做HEAD。
data:image/s3,"s3://crabby-images/85561/855610c0bd0db0b497a08a22c1930ffe1d796822" alt=""
data:image/s3,"s3://crabby-images/826c0/826c0e17abb0558b95486c325123466517633e3e" alt=""
把文件往Git版本库⾥添加的时候,是分两步执行的:
第一步是用"git add"把文件添加进去,实际上就是把文件修改添加到暂存区;
第二步是用"git commit"提交更改,实际上就是把暂存区的所有内容提交到当前分支。
因为我们创建Git版本库时,Git自动为我们创建了唯一一个master分支,所以,现在,
commit就是往master分支上提交更改。
需要提交的文件修改通通放到暂存区,然后,一次性提交暂存区的所有修改。
对于第一次修改后,执行了git add,然后又进行了第二次修改,然后直接进行git commit,则在工作区的第二次修改因为没有放入暂存区,所以,git commit只把暂存区的第一次修改提交了,第二次修改不会被提交。
使用:git diff HEAD -- filename 来比较工作区和版本库里面最新版本的区别:
data:image/s3,"s3://crabby-images/5858e/5858e0d1644f07c9cc8878dfaa59585f142b1ca5" alt=""
所以,每次工作区的修改最好都add一下,最后一起commit。
7、撤销修改
对于存在不一致的文件,如下:
data:image/s3,"s3://crabby-images/29fd2/29fd231d9d1dcc36e357f7efe1cb6a9a2c9e9f8b" alt=""
git提示,可以使用git restore <file>来丢弃工作区的修改
data:image/s3,"s3://crabby-images/439db/439db1f79f605a2ed0d50e1e93e438e01f5ffe7d" alt=""
使用git checkout -- <filename>也可以实现。
git restore --staged <file>丢弃暂存区的修改,即git add的修改丢弃,但只是丢弃暂存区的修改,所以执行后,status会回到Changes not staged for commit状态。
data:image/s3,"s3://crabby-images/af72c/af72c22eaf3d984d48ed82e537226318a73b5237" alt=""
关于git checkout命令:
Git社区发布了Git的新版本2.23。在该版本中,有一个特性非常引人瞩目,就是新版本的Git引入了两个新命令 git switch 和 git restore,用以替代现在的 git checkout。换言之,git checkout 将逐渐退出历史舞台。
Git社区决定这样做,是因为目前 git checkout 命令承载了太多的功能,这让新手们感到困惑。git checkout 的核心功能包括两个方面,一个是分支的管理,一个是文件的恢复。这两个核心功能,未来将由 git switch 和 git restore 分别负责。
相比之下,新命令旨在将职责明确分为两个较窄的类别:更改分支的操作和更改文件的操作
8、删除文件
已经提交的文件,在工作区删除了
data:image/s3,"s3://crabby-images/b5ecc/b5ecc26112661c062086fb7486e3496d2de55d13" alt=""
需要使用git rm <file>执行删除的添加,或者还是使用git add <file>,将删除的信息添加到暂存区,然后git commit,就将仓库中的文件删除了。
data:image/s3,"s3://crabby-images/f9f0a/f9f0ab60f7197277861cebc6024127631fb0dfd9" alt=""
恢复:
data:image/s3,"s3://crabby-images/42f2f/42f2f54c4a425d90da772f18742ec1024a2970e4" alt=""
9、远程仓库:github
本地Git仓库和GitHub仓库之间的传输是通过SSH加密的,所以,需要一点设置:
第1步:创建SSH Key。在用户主目录下,看看有没有.ssh目录,如果有,再看看这个目录下有没有id_rsa和id_rsa.pub这两个文件,如果已经有了,可直接跳到下一步。如果没有,打开Shell(Windows下打开Git Bash),创建SSH Key:
$ ssh-keygen -t rsa -C "youremail@example.com"
需要把邮件地址换成自己的邮件地址,然后一路回车,使用默认值即可,由于这个Key也不是用于军事目的,所以也不需设置密码。在用户主目录中找到.ssh目录,有id_rsa和id_rsa.pub两个文件,这两个就是SSH Key的秘钥对,id_rsa是私钥,不能泄露出去,id_rsa.pub是公钥,可以放心地告诉任何人。
第2步,设置github上的SSH KEY
data:image/s3,"s3://crabby-images/3013a/3013aaec68d7974ce59ede11d9ac56e3510bea7b" alt=""
data:image/s3,"s3://crabby-images/bd422/bd422924d4ff4e8866537a0addc49cbde111d3b2" alt=""
这样,就可以在本地连接github了。
先在github上创建一个空仓库,learngit:
然后在本地learngit工作区中执行:
git remote add origin git@github.com:kaoa000/learngit.git
这样,就将本地仓库与github上的learngit仓库进行了关联。添加后,远程库的名字就是origin,这是Git默认的叫法,也可以改成别的。
git push -u origin master 将本地仓库推送到远程仓库。
(github访问确实是慢,而且时断时续)
data:image/s3,"s3://crabby-images/7c00c/7c00c696969eb0600cbb3e03b0cab7eb343de4c0" alt=""
推送前
data:image/s3,"s3://crabby-images/ffc48/ffc489fa2c58f93dd23dbe3b6d4a1c36fca734e0" alt=""
推送后:
data:image/s3,"s3://crabby-images/dba4d/dba4d91c5d63306f9ad50cb8a1b6e5b49959e054" alt=""
把本地库的内容推送到远程,用git push命令,实际上是把当前分支master推送到远程。
由于远程库是空的,第一次推送master分支时,加上了-u参数,Git不但会把本地的 master分支内容推送的远程新的master分支,还会把本地的master分支和远程的master 分支关联起来,在以后的推送或者拉取时就可以简化命令。
现在只要本地作了提交,就可以通过命令: $ git push origin master
把本地master分支的最新修改推送至GitHub
从远程库克隆:
git clone git@github.com:kaoa000/learngit.git
data:image/s3,"s3://crabby-images/251a0/251a0f115daddc302634254db6fc31a281db2eae" alt=""
gitee的使用与github相似:
git remote add origin https://gitee.com/persistself/giteetest.git
git clone https://gitee.com/persistself/giteetest.git
不同的是,需要输入gitee的账号和密码。
10、分支管理:
创建与合并分支:
每次提交,Git都把它们串成一条时间线,这条时间线就是一个分支。目前为止只有一条时间线,在Git中,这个分支叫主分支,即 master分支。
HEAD严格来说不是指向提交,而是指向master,master才是指向提交的,所以,HEAD指向的就是当前分支。
一开始的时候,master分支是一条线,Git用master指向最新的提交,再用HEAD指向master,就能确定当前分支,以及当前分支的提交点:
data:image/s3,"s3://crabby-images/c9598/c9598a219055ee8610db5dd9ec78eee6c3ccd69f" alt=""
创建新的分支,如dev,Git新建了一个指针叫dev,指向master相同的提交,再把HEAD指向dev,就表示当前分支在dev上:
data:image/s3,"s3://crabby-images/8e150/8e150ca8134e06a08fa69816b04f8013ff6d321a" alt=""
从现在开始,对工作区的修改和提交就是针对dev分支了,如新提交一次后,dev 指针往前移动一步,而master指针不变:
data:image/s3,"s3://crabby-images/dade4/dade43f7e4943ae78bcd9acc603673abe2c3bbbe" alt=""
在dev上的工作完成了,就可以把dev合并到master上。就是直接把master指向dev的当前提交,就完成了合并:
data:image/s3,"s3://crabby-images/39f82/39f82fba2fe44f0fcb13e5da7ef492754805a9a0" alt=""
可以删除dev分支。删除dev分支就是把dev指针给删掉,删掉后,就剩下了一条master分支:
data:image/s3,"s3://crabby-images/0ecc3/0ecc38cec8420833523fbb733f142ac2ec2f97c7" alt=""
创建dev分支,然后切换到dev分支:
git checkout -b dev
git checkout命令加上-b参数表示创建并切换,相当于以下两条命令:
git branch dev
git checkout dev
使用git branch查看当前分支:
data:image/s3,"s3://crabby-images/c3cf8/c3cf8eb5c25678dd621f38eadf337c5201bc8aa5" alt=""
git branch列出所有分支,当前分支前标有一个星号:*
在dev分支上提交:
data:image/s3,"s3://crabby-images/23644/23644ddea4107b0eb846514de7a7a3e35407e1d7" alt=""
data:image/s3,"s3://crabby-images/167e5/167e55d3877c70bb27554ecb939e73d4f598dae6" alt=""
切换回master:
git checkout master
data:image/s3,"s3://crabby-images/889af/889af7893237ac800a7ee351fb829a24fc9ffdf9" alt=""
可以看到,readme.txt中dev分支上提交的修改没有了。因为那个提交是在dev分支上。
data:image/s3,"s3://crabby-images/a441e/a441e45c8417ebf287e563fefb891e0d51f606a5" alt=""
现在,把dev分支的工作成果合并到master分支上:
git merge dev :此命令用于合并指定分支到当前分支。
data:image/s3,"s3://crabby-images/e6f6f/e6f6fda1c0342d931828ab82ef6a2de61478283d" alt=""
删除dev分支,合并完成后,就可以删除dev分支:
git branch -d dev
data:image/s3,"s3://crabby-images/b9739/b973906fa112edfaa1fb1e7f1e099a0ae2f6f007" alt=""
Git鼓励大量使用分支:
查看分支:git branch
创建分支:git branch name
切换分支:git checkout name 或 git switch name
创建+切换分支:git checkout -b name 或 git switch -c name
合并某分支到当前分支:git merge name
删除分支:git branch -d name
新版本增加git switch切换分支。
11、解决冲突
git switch -c feature1
修改readme.txt最后一行,在feature1上提交,切换到master分支:
git switch master ,修改readme.txt,然后提交master分支上
data:image/s3,"s3://crabby-images/9df49/9df49d2ba8c7062de51d810b37a89293b3c15b78" alt=""
此时:
data:image/s3,"s3://crabby-images/52648/52648fdd840846f0fa08b5181797275a47740193" alt=""
这种情况下,Git无法执行"快速合并",即Fast-forward,这种合并会有冲突:
data:image/s3,"s3://crabby-images/c2fd7/c2fd73fbb031c8d6d360942e8ce9585ff6887b7f" alt=""
data:image/s3,"s3://crabby-images/87e9e/87e9e736155cfb55ddc59f679f60ee02effef25c" alt=""
Git用<<<<<<<,=======,>>>>>>标记出不同分支的内容
修改readme.txt,保存后再次添加提交:
data:image/s3,"s3://crabby-images/278e8/278e8991681e08167dea57b6901aa972fbf45020" alt=""
现在,分支变成如下图:
data:image/s3,"s3://crabby-images/6c968/6c968423bda7b08d4a2a32a0cbd8c3bb07ba00fb" alt=""
查看合并情况:git log --graph --pretty=oneline --abbrev-commit
data:image/s3,"s3://crabby-images/e1761/e1761ab5b4923e918b007104dd292f70f2bc1e7c" alt=""
删除feature1分支:git branch -d feature1
强行删除分支:git branch -D feature1 :在feature1分支上已经添加并提交,但是还没有合并,可以使用-D强制删除分支。
12、分支管理策略
合并时,使用Fast forward模式,删除分支后,会丢掉分支信息,可以强制禁用Fast forward模式。Git就会在merge时生成一个新的commit,这样从分支历史上就可以看出分支信息。
data:image/s3,"s3://crabby-images/5c35f/5c35ffba26cfdd4d23324ea6155387d85f341411" alt=""
data:image/s3,"s3://crabby-images/7b5a3/7b5a325bb48f3a19e000dd573eefeaf03b9fae45" alt=""
13、Bug分支:
git stash
git stash list
git stash apply git stash drop
git stash pop
14、多人协作:
从远程仓库克隆时,实际上Git自动把本地的master分支和远程的master分支对应起来了,并且,远程仓库的默认名称是origin。
查看远程库的信息,用:git remote 或git remote -v
data:image/s3,"s3://crabby-images/be64b/be64be8052231e4e6c4345cf08214514a08b9311" alt=""
抓取(fetch)和推送(push)的origin的地址。如果没有推送权限,就看不到push的地址。
推送分支,就是把该分支上的所有本地提交推送到远程库。推送时,要指定本地分支,这样Git就会把该分支推送到远程库对应的远程分支上:
git push origin master
推送其他分支:git push origin dev
data:image/s3,"s3://crabby-images/acc32/acc327a382bb734dfb8da9ca27b41fa60ab8fbf1" alt=""
data:image/s3,"s3://crabby-images/58dcb/58dcb74c090ff2ff8ccab17c8ca99beb22695dcc" alt=""
使用push推送的分支,如果远程库中不存在,会创建
抓取分支:其他人克隆远程库,以进行协作开发时,从远程库clone时,默认情况下,只能看到本地的master分支,可以用git branch查看。
要在mydev分支上开发,就必须创建远程origin的mydev分支到本地,使用如下命令创建本地mydev分支:
git checkout -b mydev origin/mydev
data:image/s3,"s3://crabby-images/32d58/32d58a2ff471cc659e9c820a31ee56db33c874c4" alt=""
冲突及解决:
如果其他人向origin/mydev分支推送了提交,而碰巧自己也对同样的文件做了修改,并试图推送,就会推送失败。其他人先推送
data:image/s3,"s3://crabby-images/589f4/589f4a0fa4aa9e6aeee439a7c41a062d77712258" alt=""
自己修改后又要推送:
data:image/s3,"s3://crabby-images/a70aa/a70aafb8f2c4ce7a6ff0903c4d51d01813b9a504" alt=""
解决办法是按照Git的提示,使用git pull把最新的提交从
origin/mydev抓下来,然后在本地合并,解决冲突在推送:
data:image/s3,"s3://crabby-images/fde92/fde92818905c18c6908e60586e8cb435ae0997c6" alt=""
git pull 一开始也出错了,需要git branch --set-upstream-to=origin/mydev mydev关联上
data:image/s3,"s3://crabby-images/b77d5/b77d52ef1a2770fb3d7ace72aef671720d467ddb" alt=""
修改完后需要再次git add 和git commit,然后gitpush
其他人要接着开发时,一般先git pull一下,抓取最新的版本,再继续开发:
data:image/s3,"s3://crabby-images/cd175/cd175747bd974078a0645f1771fe6f791e3b7fc3" alt=""
因此,多人协作的工作模式通常是这样:
-
首先,可以试图用git push origin branch-name推送自己的修改;
-
如果推送失败,则因为远程分支比你的本地更新,需要先用git pull试图合并;
-
如果合并有冲突,则解决冲突,并在本地提交;
-
没有冲突或者解决掉冲突后,再用git push origin branch-name推送就能成功!
如果git pull提示"no tracking information",则说明本地分支和远程分支的链接关系没有创建,用命令git branch --set-upstream-to branch-name origin/branch-name。
多人协作小结
• 查看远程库信息,使用git remote -v;
• 本地新建的分支如果不推送到远程,对其他人就是不可见的;
• 从本地推送分支,使用git push origin branch-name,如果推送失败,先用git pull抓
取远程的新提交;
• 在本地创建和远程分支对应的分支,使用git checkout -b branch-name origin/branchname,本地和远程分支的名称最好一致;
• 建立本地分支和远程分支的关联,使用git branch --set-upstream branch-name
origin/branch-name;
• 从远程抓取分支,使用git pull,如果有冲突,要先处理冲突。
15、忽略文件
在git的工作区的根目录下创建一个特殊的.gitignore文件,然后把要忽略的文件名填进去,Git就会自动忽略这些文件。