如果不用"pick"或者"edit",而是指定"squash",Git 会同时应用那个变更和它之前的变更并将提交说明归并。因此,如果想将这三个提交合并为单一提交, 可以将脚本修改成这样
保存并退出编辑器,Git 会应用全部三次变更然后将送回编辑器来归并三次提交说明。
然后保存,就会拥有包含前三次提交的全部变更的单一提交 。
注:最顶行pick 不能修改为squash
当你正在进行项目中某一部分的工作,里面的东西处于一个比较杂乱的状态,而你想转到其他分支上进行一些工作。问题是,你不想提交进行了一半的工作,否则以后你无法回到这个工作点。解决这个问题的办法就是 git stash 命令。
“‘储藏”“可以获取工作目录的中间状态——也就是修改过的被追踪的文件和暂存的变更——并将它保存到一个未完结变更的堆栈中,随时可以重新应用。
当追准备切换分支,但是还不想提交你正在进行中的工作;所以储藏这些变更。为了往堆栈推送一个新的储藏,只要运行 git stash :
这时,你可以方便地切换到其他分支工作;你的变更都保存在栈上。要查看现有的储藏,你可以使用 git stash list :
git stash apply :重新应用你刚刚实施的储藏
git stash apply stash@{number} :应用更早的储藏
只需回到需要合并的源分支,运行 git merge 命令指定要合并进来的分支即可;Git 作了合并,但没有提交,它会停下来等你解决冲突。要看看哪些文件在合并时发生冲突,可以用 git status 观察
任何包含未解决冲突的文件都会以未合并(unmerged)的状态列出。Git 会在有冲突的文件里加入标准的冲突解决标记,可以通过它们来手工定位并解决这些冲突。可以看到此文件包含类似下面这样的部分:
可以看到 ======= 隔开的上半部分,是 HEAD (即 master 分支,在运行 merge 命令时所切换到的分支)中的内容,下半部分是在 iss53 分支中的内容。解决冲突的办法无非是二者选其一或者亲自整合到一起。比如可以通过把这段内容替换为下面这样来解决:
在解决了所有文件里的所有冲突后,运行 git add 将把它们标记为已解决状态(译注:实际上就是来一次快照保存到暂存区域。)
pull时为了防止更改后pull Feiled的出现,先执行commit,stash or revert。
git中的文件对象,最终都是存储在git/objects目录中,以二进制形式存储的。git既松散存储,也进行差异化存储。每次修改一个文件,会在git/objects目录中,生成一个二进制对象,当再次修改时,又生成另一个二进制对象,这即是松散存储。但是当工作区的二进制对象数超过6700个时(数目可配置),git会进行一次打包行为,针对相同文件进行差异化存储来减少文件数。git大概总共有100多个命令,命令不是很多,但是每个命令,都可以带很多参数,功能很强大。git命令总共分为两类:高级命令和底层命令。
git add和git commit都属于高级命令,高级命令是通过调用多个底层命令来完成 *** 作,这一节希望大家能彻底搞明白git add和git commit
举个例子,假如我们新建一个工程,并新建了一个testgo文件,在这个文件里边随便写一些内容,下面我们用底层命令来生成一个完整的git历史记录:
第一步:针对testgo文件生成二进制存储对象
git hash-object命令就是针对某一个文件生成它的二进制存储对象,并返回它的sha1签名码,假设返回的是"0d5a2b611190403ef08124286fe64608456101cd"
第二步:更新索引文件
这一步是将我们上一步生成的对象以及它的路径添加到索引中。用到了我们上一节讲过的updae-index命令。
以上两步命令和git add testgo命令完全等价。
第三步:生成目录树对象
调用write-tree命令之后,git会针对当前索引区的目录结构,生成一个tree对象,并返回这个tree对象的sha1签名码,假设返回:d8329f
第四步:生成commit对象
echo后边单引号里边的内容就是我们的提交message了。第三步和第四步就是完成了一次git commit命令的 *** 作。
大家可以试着 *** 作一下以上四部命令,并观察一下git/objects目录和工作区git状态的变化。(1) git仓库(版本库):git仓库就是一个git文件夹。这个文件夹内包含了很多文件(见插图2),其中有一个很重要的文件夹objects,保存了暂存区的所有文件对象,包括blob对象、tree对象、commit对象等,这些对象都是一以文件的形式来保存的。还有HEAD文件,保存着最新的提交的指针。当然很多人到这里可能还是不理解objects中的文件对象和HEAD中保存的指针到底是什么意思,没关系,下面会详细讲解。
(2)工作区:在一个项目目录中,除了git文件的其他所有文件的集合就是工作区。
(3)暂存区:暂存区可以理解为文件从修改到最后提交到git版本库之间的一个缓存,为了防止一次提交了不必要的文件,有回退的余地,便有了暂存区。
(4)HEAD:HEAD在git文件夹中是一个文件,文件的内容是一个32位的16进制数,这只是一个指针,他指向最近一个提交点、这个提交点实质是一个commit对象,对象里包含里多个属性,包括最后一个提交点目录结构索引、上一次提交点id、提交人、提交时间等。
(1) 首先,跟着我的思路,我们来创建一个git仓库, 通过一个例子来理解git add命令
1、首先,我们通过git init来创建一个仓库,这时会生成一个git文件夹,这个文件夹就一个git仓库
2、创建一个文件Ajava
执行 git add Ajava
到git里面的objects文件中,发现,多了一个文件夹,打开,里面有一个文件,名字是一串符号,打开这个文件,里面是一串无法识别的字符。
git add做了啥?
其实,在执行git add命令的时候,git首先会通过hash函数算出文件名Ajava对应的sha-1值是1fed2a34f13dd2a34f13d,这个code是一个32位的16进制数,git会取这个code的前两位作为文件夹名在git/objects文件夹中创建一个文件夹1f,取code的后30位作为文件名,在git/objects/1f中创建一个文件ed2a34f13dd2a34f13d,git会将Ajava的文件内容通过一定的压缩保存在这个文件中
未完待续
本文到这里就快结束了,文中可能有些地方没有说明清楚,欢迎指定讨论,有问题欢迎骚扰,看到会第一时间回复!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)