如何删除已合并的分支?我可以一次删除所有分支,而不是逐个删除每个分支吗?
当前回答
如何在PowerShell控制台中删除合并的分支
git branch --merged | %{git branch -d $_.Trim()}
如果要排除master或任何其他分支名称,可以使用PowerShell Select String这样进行管道,并将结果传递给git branch-d:
git branch -d $(git branch --merged | Select-String -NotMatch "master" | %{$_.ToString().Trim()})
其他回答
我已经用亚当的答案很多年了。也就是说,在某些情况下,它的行为并没有达到我的预期:
包含单词“master”的分支被忽略,例如“notmaster”或“masterful”,而不仅仅是master分支包含单词“dev”的分支被忽略,例如“dev test”,而不仅仅是dev分支删除可从当前分支的HEAD(即,不一定是主分支)访问的分支在分离HEAD状态下,删除当前提交中可访问的每个分支
1和2很容易解决,只需更改正则表达式。3取决于所需内容的上下文(即仅删除尚未合并到主分支或当前分支的分支)。如果您无意中在分离的HEAD状态下运行,4可能会造成灾难(尽管可以使用git-relog恢复)。
最后,我希望这一切都在一个不需要单独(Bash|Ruby|Python)脚本的一行中。
TL;博士
创建一个接受可选-f标志的git别名“sweep”:
git config --global alias.sweep '!git branch --merged $([[ $1 != "-f" ]] \
&& git rev-parse master) | egrep -v "(^\*|^\s*(master|develop)$)" \
| xargs git branch -d'
并调用它:
git sweep
or:
git sweep -f
长而详细的答案
对我来说,创建一个带有一些分支和提交的git repo示例来测试正确的行为是最简单的:
通过一次提交创建新的git repo
mkdir sweep-test && cd sweep-test && git init
echo "hello" > hello
git add . && git commit -am "initial commit"
创建一些新分支
git branch foo && git branch bar && git branch develop && git branch notmaster && git branch masterful
git branch --list
酒吧发展食品*船长,船长有驾驭能力的非主人翁
期望的行为:选择所有合并的分支,但主分支、开发分支或当前分支除外
原始正则表达式缺少分支“masterful”和“notmaster”:
git checkout foo
git branch --merged | egrep -v "(^\*|master|dev)"
酒吧
使用更新的正则表达式(现在不包括“develop”而不是“dev”):
git branch --merged | egrep -v "(^\*|^\s*(master|develop)$)"
酒吧有驾驭能力的非主人翁
切换到分支foo,进行新的提交,然后根据foo签出新的分支foobar:
echo "foo" > foo
git add . && git commit -am "foo"
git checkout -b foobar
echo "foobar" > foobar
git add . && git commit -am "foobar"
我当前的分支是foobar,如果我重新运行上面的命令来列出我要删除的分支,即使分支“foo”还没有合并到master中,它也会包含在内:
git branch --merged | egrep -v "(^\*|^\s*(master|develop)$)"
酒吧食品有驾驭能力的非主人翁
但是,如果我在master上运行相同的命令,则不包含分支“foo”:
git checkout master && git branch --merged | egrep -v "(^\*|^\s*(master|develop)$)"
酒吧有驾驭能力的非主人翁
这仅仅是因为如果没有另外指定,gitbranch--merged默认为当前分支的HEAD。至少对于我的工作流程来说,我不想删除本地分支,除非它们已合并到主分支,所以我更喜欢使用git-rev-parse的以下变体:
git checkout foobar
git branch --merged $(git rev-parse master) | egrep -v "(^\*|^\s*(master|develop)$)"
酒吧有驾驭能力的非主人翁
分离的HEAD状态
依赖gitbranch的默认行为--merged在分离HEAD状态下会产生更严重的后果:
git checkout foobar
git checkout HEAD~0
git branch --merged | egrep -v "(^\*|^\s*(master|develop)$)"
酒吧食品脚踏板有驾驭能力的非主人翁
这会删除我刚才打开的分支,“foobar”和“foo”,这几乎肯定不是理想的结果。然而,通过我们修改的命令:
git branch --merged $(git rev-parse master) | egrep -v "(^\*|^\s*(master|develop)$)"
酒吧有驾驭能力的非主人翁
一行,包括实际删除
git branch --merged $(git rev-parse master) | egrep -v "(^\*|^\s*(master|develop)$)" | xargs git branch -d
全部打包成一个git别名“sweep”:
git config --global alias.sweep '!git branch --merged $([[ $1 != "-f" ]] \
&& git rev-parse master) | egrep -v "(^\*|^\s*(master|develop)$)" \
| xargs git branch -d'
别名接受可选的-f标志。默认行为是只删除已合并到主分支的分支,但-f标志将删除已合并至当前分支的分支。
git sweep
删除了分支条(为9a56952)。删除了分支masterful(为9a56952)。删除了分支notmaster(为9a56952)。
git sweep -f
已删除分支foo(为2cea1ab)。
我使用git流式命名方案,因此这对我来说非常安全:
git branch --merged | grep -e "^\s\+\(fix\|feature\)/" | xargs git branch -d
它主要查找以字符串fix/或feature/开头的合并提交。
如果要删除合并的分支,则只需删除远程跟踪分支,除非您另有说明。
因此,要删除这些分支,您可以通过
git branch--remote--合并原始/主|egrep-v“(^\*|master|development)”|cut-b 10-|xargs git push--删除原始
这将删除除主分支和开发分支外的所有合并分支(合并到主分支)。
库布恩的回答没有删除分支名称中包含单词master的分支。以下内容改进了他的回答:
git branch -r --merged | grep -v "origin/master$" | sed 's/\s*origin\///' | xargs -n 1 git push --delete origin
当然,它不会删除“master”分支本身:)
Adam更新答案的别名版本:
[alias]
branch-cleanup = "!git branch --merged | egrep -v \"(^\\*|master|dev)\" | xargs git branch -d #"
此外,请参阅此答案以获得有关转义复杂别名的实用提示。
推荐文章
- Visual Studio代码如何解决合并冲突与git?
- 无法推送到远程分支,无法解析到分支
- Git:如何将数据库重置为特定的提交?
- 如何在合并期间使用Git和命令行保存本地文件或远程文件?
- 能够用一个命令推到所有git遥控器?
- 重新基于Git合并提交
- 忽略已经签入目录的内容?
- 如何完全删除TFS绑定
- 哪些Eclipse文件属于版本控制?
- 如何从windows cmd保存git提交消息?
- (Mac) -bash: __git_ps1:命令未找到
- 如何删除多个已删除的文件在Git仓库
- 使用vimdiff查看所有' git diff '
- 如何拉特定的目录与git
- 本地存储库中的文件与源文件之间的差异