How do I split a file in Git?

3.3k Views Asked by At

I have source file A, which is getting long and contains two large classes. I want to split it into files A and B, preserving history and "blame" data.

In Subversion, this is very easy: do an svn copy A B, delete the lines from A that should only be in B and vice versa, then commit. But the project's owner sadly wants to switch to Git, and this is one of the workflows that I don't know how to do.

When you do svn log B, you get B's history from after the split, plus A's history from before the split. Additionally, svn blame B will attribute pre-split changes to the correct authors in the history of A, rather than attribute them to whomever did the split.

3

There are 3 best solutions below

2
On BEST ANSWER

Git does not explicitly track renames. Instead where needed it calculates if a rename has occurred. For example, commands like git blame, git diff and git merge can make use of Git's rename detection machinery.

For this reason you should just make the change you need and commit it as normal.

So the correct steps for this case would be:

<make the needed changes to the files>
git add A B
git commit

Given that renaming is not explicitly tracked means that the rename detection algorithm sometimes needs a little bit of tuning to achieve what you need. For example, when doing a merge you might need to adjust the rename-threshold=<n> to get a better auto-merge.

If the similarity of the "renamed" files is below what Git detects, or can be configured to detect, then there is no way for Git to consider it a rename.

0
On

First, copy the original file to a new filename:

$ cp A B

Next, delete unwanted lines from B and A, then add the new file to git and commit:

$ git add -A
$ git commit
0
On

You don't have to do anything special when committing. Just commit the new and deleted files as usual, then use the -C option with git blame when you want to track copied file contents. git will figure everything out automatically. The same goes for moved files.

See also: Record file copy operation with Git