10

首先这是不同的,因为这适用于所有分支,而不是一个。(我发现的所有这些一次只指定一个分支)

我在一个大约有 20-30 个分支的仓库上工作(我知道我知道的长故事不好的做法)

无论如何,我正在使用 git bfg 清理器删除一些旧提交,使用它后,您需要删除并重新克隆代码或重置每个分支。

我知道如何使用

git fetch origin
git reset --hard origin/master

但是有没有办法用一个命令重置每个分支,还是我必须一次只做一个分支?

我有很多本地文件被忽略,我不想处理复制和重写。(如 IDE 文件、计算机配置文件等...)

4

5 回答 5

7

首先这是不同的,因为这适用于所有分支而不是一个

您可以使用一个脚本,该脚本将遍历您的分支,然后对每个分支执行您想要的操作:

# Loop on all the desired branches
for ref in $(git branch -a)
do
   # ... DO ANYTHING YOU NEED HERE
   # You have the $ref which stores the branch name
   # ex: remotes/origin/master
done

有没有更好更简单的解决方案?

是的,只需再次克隆项目,您将拥有与原始项目相同的内容。
这正是您为每个分支手动执行的操作。


笔记:

您可以使用以下命令一次获取所有数据:

# Fetch all remotes, branches and remove deledted content
git fetch --all --prune
于 2016-03-15T18:39:29.377 回答
5

为什么不直接删除所有本地分支,然后每当远程分支重新签出时,您就会被同步?

 # BE CAREFUL!!!
 # THIS COMMAND WIPES OUT ALL LOCAL BRANCHES
 ​git branch | xargs git branch -D

xargs 针对标准输入的每一行运行提供的命令,在本例中,针对原始“git branch”命令的每一行执行“git branch -D”。

当前分支的“git branch -D”命令将失败,但所有其他分支都将被擦除。

或者,您可以使用运行“git branch -f”的“xargs”来制作一些东西,以强制将每个本地分支设置为其远程副本,但我认为上面的“git branch -D”方法是最简单的,也是最简单的就后果而言,直截了当,涉及潜在的危险。

于 2016-03-15T18:43:42.770 回答
3

刚刚写了一个小工具来做到这一点。

该工具的优点:

  1. 在一个存储库中支持多个遥控器。
  2. 支持本地和跟踪分支具有不同的名称。
  3. 没有容易出错的正则表达式解析/编辑。

这是脚本:

#!/bin/bash

ENOENT=2
EEXIST=17

curref=$(git symbolic-ref HEAD 2>&-)

fetch_all() {
  git fetch --all
  ret=$?
  [ "$ret" -eq 0 ] || {
    return "$ret"
  }
}

hard_reset_all() {
  ret=0

  while read -r line; do
    branch=$(echo "$line" | cut -c 8- | rev | cut -c 8- | rev)
    remote=$(git config --local --get "branch.$branch.remote")
    upstream=$(git config --local --get "branch.$branch.merge" | cut -c 12-)

    if ! git rev-parse --verify --quiet "$remote/$upstream" 1>&- 2>&-; then
      ret=$(( $ret | $ENOENT ))
      echo "Branch $branch"
      echo "  skipped: upstream is absent"
      continue
    fi

    if [ -z "$(git log --oneline -1 "$branch...$remote/$upstream")" ]; then
      continue
    fi

    echo "Branch $branch"

    echo "  $(git log --oneline -1 $branch)"
    if [ "refs/heads/$branch" = "$curref" ]; then
      git reset --hard "$remote/$upstream" 1>&- 2>&-
    else
      git update-ref "refs/heads/$branch" "$remote/$upstream" 1>&- 2>&-
    fi
    echo "  -> $(git log --oneline -1 $remote/$upstream)"
  done < <(git config --local --name-only --get-regex '^branch.*remote$')

  return "$ret"
}

fetch_all
excode=$?
[ "$excode" -eq 0 ] || {
  exit "$excode"
}

if output=$(git status --porcelain) && [ -n "$output" ]; then
  echo "Skipped: repo is not clean"
  exit $EEXIST
fi

hard_reset_all
excode=$?
[ "$excode" -eq 0 ] || {
  exit "$excode"
}
于 2019-10-14T07:54:54.893 回答
1

您可以使用以下 oneliner(基于其他答案):

git branch --format='%(refname:short)' | xargs -i sh -c "git checkout {} && git reset --hard origin/{}"

  • --format='%(refname:short)'消除烦人*git branch
  • -i允许{}用作分支的占位符

警告:我没有经常使用该命令,因此它可能并不完全安全,如果您想让它更安全,请查看该答案。

于 2019-11-26T16:23:08.013 回答
0

完全删除目录,然后克隆存储库。

rm -R repodir
git clone https://github.com/repodir/repodir
于 2020-08-24T18:42:23.760 回答