如何删除已合并的分支?我可以一次删除所有分支,而不是逐个删除每个分支吗?


当前回答

我已经用亚当的答案很多年了。也就是说,在某些情况下,它的行为并没有达到我的预期:

包含单词“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)。

其他回答

您可以将提交添加到--merged选项。这样,您可以确保只删除合并到原始/主文件中的分支

以下命令将从原点删除合并的分支。

git branch -r --merged origin/master | grep -v "^.*master" | sed s:origin/:: |xargs -n 1 git push origin --delete 

您可以测试将删除哪些分支来替换git推送原点--delete with echo

git branch -r --merged origin/master | grep -v "^.*master" | sed s:origin/:: |xargs -n 1 echo

请尝试以下命令:

git branch -d $(git branch --merged | grep -vw $(git rev-parse --abbrev-ref HEAD))

通过使用gitrevparse,将获取当前分支名称以排除它。如果出现错误,则意味着没有要删除的本地分支。

要对远程分支执行相同操作(使用远程名称更改源),请尝试:

git push origin -vd $(git branch -r --merged | grep -vw $(git rev-parse --abbrev-ref HEAD) | cut -d/ -f2)

如果您有多个遥控器,请在剪切之前添加grep-origin |,以仅过滤源。

如果上述命令失败,请尝试先删除合并的远程跟踪分支:

git branch -rd $(git branch -r --merged | grep -vw $(git rev-parse --abbrev-ref HEAD))

然后git再次获取远程,并再次使用前面的gitpush-vd命令。

如果您经常使用它,请考虑将别名添加到~/.gitconfig文件中。

如果您错误地删除了一些分支,请使用gitreflog查找丢失的提交。

对于我来说,gitbranch--merged不显示通过GitHub PR合并的分支。我不确定原因,但我使用以下行删除所有没有远程跟踪分支的本地分支:

diff <(git branch --format "%(refname:short)") <(git branch -r | grep -v HEAD | cut -d/ -f2-) | grep '<' | cut -c 3- | xargs git branch -D

说明:

gitbranch--格式“%(refname:short)”提供本地分支列表gitbranch-r | grep-v HEAD | cut-d/-f2-提供远程分支列表,过滤掉HEADdiff<(…)<(grep“<”筛选第一个列表中存在但第二个列表中不存在的分支cut-c 3-给出从第3个字符开始的行,从而删除前缀<xargs git branch-D针对每个分支名称执行git branch-D

或者,可以这样避免grep-v“<”:

diff --old-line-format="%L" --new-line-format="" --unchanged-line-format="" <(git branch --format "%(refname:short)") <(git branch -r | grep -v HEAD | cut -d/ -f2-) | xargs git branch -D

要删除远程上已合并的所有分支,请执行以下操作:

git branch -r --merged | grep -v master | sed 's/origin\//:/' | xargs -n 1 git push origin

在Git的最新版本中

git branch -r --merged | grep -v master | sed 's/origin\///' | xargs -n 1 git push --delete origin

UPDATE(通过@oliver;因为不适合注释,但已经有足够的答案了):如果您在分支ABC上,那么ABC将出现在git branch-r--merged的结果中,因为未指定分支,所以分支默认为当前分支,分支始终符合自身合并的条件(因为分支和自身之间没有区别!)。

因此,指定分支:

git branch -r --merged master | grep -v master ...

OR第一个结账主机:

git checkout master | git branch -r --merged | grep -v ...

我最喜欢的简单脚本:

git branch --merged | grep -E -v "(master|main|develop|other)" | xargs git branch -d