如果我想将仅对特定提交中更改的部分文件(包括对多个文件的更改)进行的更改合并到Git分支中,如何实现这一点?

假设名为stuff的Git提交对文件A、B、C和D进行了更改,但我只想将stuff的更改合并到文件A和B。这听起来像Git cherry-pick的工作,但cherry pick只知道如何合并整个提交,而不是文件的子集。


当前回答

自动化程度更高:

#!/usr/bin/env bash

filter_commit_to_files() {
    FILES="$1"
    SHA="$2"
    git show "$SHA" -- $FILES | git apply --index -
    git commit -c "$SHA"
}

示例用法:

filter_commit_to_files "file1.txt file2.txt" 07271c5e

我通过复制并粘贴到外壳中来定义它。您不需要here文档。

其他回答

使用修补程序标志签出

完成您所描述内容的一个简单方法是通过签出补丁。修改我在顶部提供的四个变量。脚本执行以下操作:

克隆包含提交的分支(源分支)签出要将文件移动到的分支(目标分支)在所需的提交版本签出源分支,提供补丁标志和文件的相对位置作为参数。添加、提交、推送

对我来说,这一直是最简单的方法。步骤#3确实创建了一个交互式shell,但如果你想破坏目标,你可以默认整个选项。

source_branch=branch_with_file_you_want
destination_branch=file_go_here
rev=757c47d4
relative_file_path=structures/serializers/node.py
message="Patching a file from $source_branch to $destination_branch"

git clone https://github.com/yourepo/app.git -b $source_branch $source_branch
cd $source_branch
git checkout $destination_branch
git checkout $rev --patch $source_branch $relative_file_path
git add $relative_file_path
git commit -m "$message"
git push

如果您是远程GitLab,如果您需要MR,可以使用git push-o merge_request.create

使用gitmerge--squashbranch_name,这将从其他分支获取所有更改,并为您准备提交。现在删除所有不需要的更改并保留所需的更改。git不会知道有合并。

我会选择一切,然后这样做:

git reset --soft HEAD^

然后我会还原我不想要的更改,然后进行新的提交。

我找到了另一种方法来防止樱桃采摘上的任何冲突合并,IMO很容易记住和理解。因为您实际上不是在挑选提交,而是其中的一部分,所以您需要先拆分它,然后创建一个符合您需要的提交,然后再挑选它。

首先从要拆分并签出的提交创建分支:

$ git checkout COMMIT-TO-SPLIT-SHA -b temp

然后恢复上一次提交:

$ git reset HEAD~1

然后添加您要选择的文件/更改:

$ git add FILE

并提交:

$ git commit -m "pick me"

请注意提交散列,让我们将其称为PICK-SHA,然后返回主分支,例如master强制结账:

$ git checkout -f master

和樱桃选择承诺:

$ git cherry-pick PICK-SHA

现在可以删除临时分支:

$ git branch -d temp -f

其他方法对我不起作用,因为提交时对很多其他文件进行了大量更改和冲突。我想到的只是

git show SHA -- file1.txt file2.txt | git apply -

它实际上并没有为您添加文件或进行提交,因此您可能需要使用

git add file1.txt file2.txt
git commit -c SHA

或者,如果您想跳过add,可以使用--cached参数来git-apply

git show SHA -- file1.txt file2.txt | git apply --cached -

您也可以对整个目录执行相同的操作

git show SHA -- dir1 dir2 | git apply -