首页 > 其他分享 >git 入门、reset的3种模式、回滚文件、还原文件、变基、merge

git 入门、reset的3种模式、回滚文件、还原文件、变基、merge

时间:2023-06-28 14:31:39浏览次数:53  
标签:reset 文件 回滚 git feature1 rebase 索引 commit 分支


git基础知识

盗用网上的一张图,git有工作目录、索引区(也叫暂存区)、历史区,这3个区,一定要记在脑子里,基本上git所有的操作都是操作这3个区。新建一个文件并提交的一般操作是,

  1. 新建文件
  2. git add到索引区
  3. git commit到历史区(添加-a参数会自动提交到索引区,相当于第2步+第3步)

git 入门、reset的3种模式、回滚文件、还原文件、变基、merge_重置

git reset 有3种模式

git reset 有3种模式,

git reset [--soft | --mixed [-N] | --hard | --merge | --keep] [-q] [<commit>]

git reset –mixed

这是默认的重置方式,重置索引区,保留工作区。
比如,修改了一个文件后,会提示文件被修改了,并提示add提交到索引区或者restore放弃工作目录更改。

git status
On branch feature1                                                     
Your branch is up to date with 'origin/feature1'.                      
                                                                       
Changes not staged for commit:                                         
  (use "git add <file>..." to update what will be committed)           
  (use "git restore <file>..." to discard changes in working directory)
        modified:   src/main/java/com/example/learnspringboot/LearnspringbootApplication.java
                                                                                             
no changes added to commit (use "git add" and/or "git commit -a")

接下来提交到索引区,接着看下状态,会发现有提示,待commit,或者restore索引区。

PS D:\learnspringboot> git add src/main/java/com/example/learnspringboot/LearnspringbootApplication.java
PS D:\learnspringboot> git status
On branch feature1                                 
Your branch is up to date with 'origin/feature1'.  
                                                   
Changes to be committed:                           
  (use "git restore --staged <file>..." to unstage)
        modified:   src/main/java/com/example/learnspringboot/LearnspringbootApplication.java

然后reset(默认是mixed),会重置索引区保留工作目录,所以提示中有Unstaged changes after reset,重置后与提交到索引区之前完全一样。

PS D:\learnspringboot> git reset                                                             
Unstaged changes after reset:                                                    
M       src/main/java/com/example/learnspringboot/LearnspringbootApplication.java
PS D:\learnspringboot> git status
On branch feature1                                                     
Your branch is up to date with 'origin/feature1'.                      
                                                                       
Changes not staged for commit:                                         
  (use "git add <file>..." to update what will be committed)           
  (use "git restore <file>..." to discard changes in working directory)
        modified:   src/main/java/com/example/learnspringboot/LearnspringbootApplication.java

no changes added to commit (use "git add" and/or "git commit -a")

git reset –soft

重置head指向commit,但索引区和工作区都保存,也就是说add后但未commit的和本地工作目录都会保留。这种情况适合,本地工作目录做了更改add或者没add,但现在想回滚到某个版本,并且不想丢弃这些更改,那么就用这种方式。

git reset –hard

重置head指向commit、重置索引区、工作区,如果本地做了变更,add或者没add,最后不想要了,那么就用这种方式。

总结

这3种模式用联想记忆法,比如soft是软的意思,是最软的,重置head指向commit,索引区、工作区都保留;mixed是混合的,那就是中等喽,所以重置索引区,保留工作区;hard是最硬的,重置索引区、工作区;可以看到是包含关系,har包含mixed,mixed包含soft。为什么叫soft?因为它重置的范围最小;为什么叫mixed,因为它是中等,所以重置soft的部分+索引区,为什么是hard?因为重置了mixed部分+工作区。

经常见的问题

  • 新建了文件,但还没git add,想删除,怎么办?
  • 这种情况,因为文件还未纳入索引区,所以直接在磁盘或者ide删除就可以了。
  • 新建了文件,git add了,想彻底删除,怎么办?
  • 根据上图所说,应该是用 git reset --hard或者git reset,重置索引区,然后再物理删除文件就可以了。
  • 新建了文件,commit了,不想要这个文件了,怎么办?
  • 直接删除,然后commit就可以了。

以上问题最关键的在于理解3个工作区(工作目录、索引区、历史区)理解了后,运用之妙存乎一心了。

intellij idea中git操作

还原文件

  • intellij idea中,还原文件,通常是右击git-----Rollback(ctrl+alt+z),通过日志(下方,git----Console)中可以看到使用的命令git -c core.quotepath=false -c log.showSignature=false checkout HEAD -- src/main/java/com/example/learnspringboot/LearnspringbootApplication.java ,它用的checkout,查官方文档
git checkout [-f|--ours|--theirs|-m|--conflict=<style>] [<tree-ish>] 
--pathspec-from-file=<file> [--pathspec-file-nul]

Overwrite the contents of the files that match the pathspec. When the (most often a commit) is not given, overwrite working tree with the contents in the index. When the is given, overwrite both the index and the working tree with the contents at the .

这意思就是指定了commit就会用commit版本覆盖索引区和工作目录,如果没指定,则使用索引区覆盖工作区。

  • 用索引区覆盖工作目录
  • git checkout filepath
  • 某个commit节点覆盖索引区和工作目录
  • git checkout commit filepath,比如git checkout HEAD filepath

更新

intellij idea中更新的快捷键是ctrl+T,其实它包含了两个动作,默认的是fetch+merge,可以通过File----settings----Git----Update进行更改,本人更新喜欢rebase,因为提交历史是一条线,更清晰。

git 入门、reset的3种模式、回滚文件、还原文件、变基、merge_github_02

提交(快捷键ctrl+k)

这个没啥好说的。

推送(快捷键ctrl+shift+k)

这个没啥好说的。

stash暂存

比如正在开发中,来了个紧急bug,需要赶紧修复,但当前功能未开发完,commit很可能造成编译错误或者功能错误,我们就想如果有个地方可以暂存下变更就好了,git stash (vt. 藏匿; 隐藏; 存放; 贮藏;)
就是做这个的。右击git-----stash changes,当想恢复的时候,右击git---- unstash changes–apply changes

git rebase vs merge

  • 什么时候该用rebase?
  • rebase就是变基,只要没提交到远程仓库或者提交到了远程仓库,但别人还未基于此开发就可以使用。援引原文

如果你只对不会离开你电脑的提交执行变基,那就不会有事。 如果你对已经推送过的提交执行变基,但别人没有基于它的提交,那么也不会有事。
如果你对已经推送至公用仓库的提交上执行变基命令,并因此丢失了一些别人的开发所基于的提交, 那你就有大麻烦了,你的同事也会因此鄙视你。

  • 举例
    比如,有两个分支experiment分支、master分支,experiment从master中切出来的,分别在两个分支上做了变更,现在要把experiment合并到master上,使用merge通常是这样的,

rebase相当于以master为基础,然后重放在experiment分支上做的变更

git 入门、reset的3种模式、回滚文件、还原文件、变基、merge_重置_03


可以看到rebase后的master是一条直线,更加清晰,但是rebase有风险,在你们没弄清楚他的作用之前不要轻易使用。

Merge remote-tracking branch ‘origin/feature1’ into feature1

也许你见过如上的合并,这通常发生在更新代码时,它是怎么发生的呢?发生场景

  1. 本地更改了代码commit了,但未push
  2. 其他人也更改这个分支commit了,并push了
  3. 你打算push,push之前先更新了一下,就会出现上述情况。

这是因为分支分叉了,要合并,默认的fetch+merge就会产生上述情况,提交线不清晰,可以设置使用fetch+rebase来避免这个情况,设置方法同上边intellij idea 更新小节。

使用rebase踩过的坑

比如一个master分支,在此基础上,切出了feature1分支,因为feature1分支是多个同事都要开发所以push到了远程,开发过程中,为了保持feature1比较新,要时不时将master的提交合并到feature1上,我就rebase了master,然后intellij idea就提示要update和push可以做,如果update就会莫名其妙的很多冲突。

这是因为feature1已经变基了master,此时再update,相当于本地feature1分支又要reabse 远程feature1(我的update策略设置的rebase),已经rebase了master就没必要再rebase远程feature1了,所以正确的操作方式是变基master后,直接force push覆盖远程分支,因为是force push所以这里要格外小心,好在intellij idea的force push默认是Force-with-lease,(lease vt. 租用,租借),就好像本地租借了远程分支,但不加锁,如果远程有更新则会被拒绝,可以再次选择强制推送,

git 入门、reset的3种模式、回滚文件、还原文件、变基、merge_github_04

基于以上原因所以git官方才推荐了使用变基的场景只要没提交到远程仓库或者提交到了远程仓库,但别人还未基于此开发就可以使用。…

最佳实践

  • 多个人基于同一分支开发时,更新,怎么操作?
  • 设置git pull --rebase,设置方法同上边intellij idea 更新小节,如果不设置默认是git pull --merge
  • 下游分支更新上游分支的最新提交
  • 在下游分支上执行,git rebase 上游分支,如果下游分支有对应远程分支,则force push,注意跟同事协商好,同事全部都push了,由一个人rebase;如果没有对应远程分支,那就没不用这么麻烦了。
  • 上游分支合并下游分支
  • 先执行上一步,然后切换到上游分支执行,git merge 下游分支,这样的提交线是一条直线。

head

git的命令中经常会用到head,可以将head看做一个指针,head指本地的版本库,可以将head替换为远程库,比如origin/master,origin代表一个地址,master是分支。

origin

origin代表远程仓库的地址 ,在config文件中,如

git 入门、reset的3种模式、回滚文件、还原文件、变基、merge_spring_05


也可以通过intellij idea查看,

右击git—manage remotes

git 入门、reset的3种模式、回滚文件、还原文件、变基、merge_github_06

参考

Why am I merging “remote-tracking branch ‘origin/develop’ into develop”?


标签:reset,文件,回滚,git,feature1,rebase,索引,commit,分支
From: https://blog.51cto.com/u_15561616/6571904

相关文章

  • asp.net http大文件断点续传上传
    ​ 需求:项目要支持大文件上传功能,经过讨论,初步将文件上传大小控制在500M内,因此自己需要在项目中进行文件上传部分的调整和配置,自己将大小都以501M来进行限制。 第一步:前端修改由于项目使用的是BJUI前端框架,并没有使用框架本身的文件上传控件,而使用的基于jQuery的Uploadify......
  • 最佳被虐——Android混淆文件生成器
    直接上效果!混淆前;混淆后;使用获取混淆文件自己生成规则使用intellijidea打开 proguard-creater 工程编辑Main.java根据提示填写相应参数运行即可使用已有规则前往proguard-file 下载对应的文件即可Android工程配置开启混淆buildTypes{release{minifyEnabledt......
  • VUE http大文件断点续传上传
    ​ 对于大文件的处理,无论是用户端还是服务端,如果一次性进行读取发送、接收都是不可取,很容易导致内存问题。所以对于大文件上传,采用切块分段上传,从上传的效率来看,利用多线程并发上传能够达到最大效率。 本文是基于springboot+vue实现的文件上传,本文主要介绍服务端实现文件......
  • Gitee通过本地git上传大于10M的文件教程
    Gitee通过网页端默认上传的文件需要小于10M,如上传10M-100M的文件需要通过本地git上传,此时需要使用git指令操作,步骤如下。首先创建一个文件夹,打开后右键使用gitbash功能分别进行简要步骤说明如下1-8,详细可见下图操作说明。1、gitinit初始化;2、gitremote绑定gitee仓库3、git......
  • 如何正确删除无法删除的windows系统文件
    需要两个步骤,且步骤严格按照1、2先后顺序。【删除系统文件,本地或者重新插入的盘,该操作存在风险,建议非必须勿操作】1、更改文件所有者授权为当前管理员账户:需要递归,则在以下命令后增加/R选项递归:takeown/f文件或文件夹/A2、修改管理员权限,增加所有的权限,包括读、写、修......
  • redis配置文件
    1)绑定的ip地址,多个ip用空格隔开bind127.0.0.12)端口,默认6379,一般不做修改port63793)是否以守护进程启动,默认为no,一般改为yes代表后台启动(windows系统不支持)daemonizeno4)定义日志级别,默认值为notice,有如下4种取值:debug(记录大量日志信息,适用于开发、测试阶段)verbose(较多日......
  • Git忽略部分修改的方法(.gitignore添加忽略文件不起作用的解决办法)
    背景:有时候本地修改,有些修改不需要提交上库,所以需要进行部分文件的修改进行忽略处理。 一、.gitignore方式忽略.gitignore文件中只能忽略本地新增的目录或者文件,这个文件默认会被提交上库,除非该文件添加了忽略自身。 二、.git/info/exclude方式忽略.git/info/exclude文件......
  • 如何将mp4文件解复用并且解码为单独的.yuv图像序列以及.pcm音频采样数据?
    一.初始化解复用器在音视频的解复用的过程中,有一个非常重要的结构体AVFormatContext,即输入文件的上下文句柄结构,代表当前打开的输入文件或流。我们可以将输入文件的路径以及AVFormatContext**format_ctx传入函数avformat_open_input(),就可以打开对应的音视频文件或流。接......
  • python指定大小文件生成
    使用特定大小的随机数生成,使用随机数生成器生成特定大小的字节,并将其写入文件中importosdefgenerate_file(file_path,file_size_bytes):withopen(file_path,'wb')asfile:file.write(os.urandom(file_size_bytes))if__name__=='__main__':#生......
  • Linux文件系统
    Linux文件系统查看磁盘空间dfdf以磁盘分区为单位查看文件系统,可以获取硬盘被占用了多少空间,目前还剩下多少空间等信息。例如,我们使用df-h命令来查看磁盘信息,-h选项为根据大小适当显示。显示内容参数说明:Filesystem:文件系统Size:分区大小Used:已使用容量Avail:还可以......