DEV Community

Cover image for The Complete Guide to Resolving Git Merge Conflicts: From Beginner to Pro
Shubham Gupta
Shubham Gupta

Posted on

The Complete Guide to Resolving Git Merge Conflicts: From Beginner to Pro

If you've been working with Git for a while, you've probably encountered a merge conflict at the worst possible moment—right before a release, during a hotfix, or while merging a large feature branch.

For many developers, especially beginners, merge conflicts can be intimidating. But here's the truth:

A merge conflict doesn't mean something is broken. It simply means Git needs your help deciding which changes should be kept.

In this guide, we'll dive deep into what merge conflicts are, why they happen, how to resolve them safely, and how professional teams minimize them.

What Is a Git Merge Conflict?

A merge conflict occurs when Git cannot automatically combine changes from two branches.

Git is usually smart enough to merge changes from different files or different parts of the same file. However, when multiple developers modify the same lines of code, Git doesn't know which version is correct.

Instead of making assumptions, Git stops the merge and asks for human intervention.

Understanding Git's Three-Way Merge

Before understanding conflicts, it's important to know how Git merges branches.

Imagine this scenario:

A --- B --- C (main)
       \
        D --- E (feature)
Enter fullscreen mode Exit fullscreen mode

B is the common ancestor
C contains changes from the main branch
E contains changes from the feature branch

When merging, Git compares:

Common ancestor (B)
Main branch (C)
Feature branch (E)

This process is called a three-way merge.

If Git can combine changes safely, the merge succeeds automatically.

If not, a conflict occurs.

A Real Example

Let's say you're working on a React application.

  • Code in Main Branch
const API_URL = "https://api.production.com";
Enter fullscreen mode Exit fullscreen mode

Meanwhile, another developer modifies the same line.

  • Code in Feature Branch
const API_URL = "https://api.staging.com";
Enter fullscreen mode Exit fullscreen mode

Now Git sees:
Main wants Production URL
Feature wants Staging URL
Git cannot determine which one is correct.

Result:

CONFLICT (content): Merge conflict in config.js
Automatic merge failed; fix conflicts and then commit the result.
Enter fullscreen mode Exit fullscreen mode

Types of Merge Conflicts

  • Content Conflicts

Most common conflict type.

Example:

function login() {
  return "Login Success";
}
Enter fullscreen mode Exit fullscreen mode

Two developers modify the same line differently.
Git cannot choose automatically.

  • Delete vs Modify Conflicts

Developer A:

Deletes user.js
Enter fullscreen mode Exit fullscreen mode

Developer B:

Adds new functionality to user.js
Enter fullscreen mode Exit fullscreen mode

Git doesn't know whether the file should exist or not.

  • Rename Conflicts

One branch renames a file while another branch edits it.

Example:

auth.js → authentication.js
Enter fullscreen mode Exit fullscreen mode

At the same time another developer updates auth.js.

Git needs clarification.

  • Binary File Conflicts

Files such as:

  • Images
  • PDFs
  • Videos cannot be merged line by line. Git requires manual selection.

How to Reproduce a Conflict

Let's create one intentionally.

Step 1: Create Feature Branch

git checkout -b feature-login
Enter fullscreen mode Exit fullscreen mode

Modify:

const version = "v1";
Enter fullscreen mode Exit fullscreen mode

Commit:

git add .
git commit -m "Update version in feature branch"
Enter fullscreen mode Exit fullscreen mode

Step 2: Switch Back to Main

git checkout main
Enter fullscreen mode Exit fullscreen mode

Modify same line:

const version = "v2";
Enter fullscreen mode Exit fullscreen mode

Commit:

git add .
git commit -m "Update version in main branch"
Enter fullscreen mode Exit fullscreen mode

Step 3: Merge

git merge feature-login
Enter fullscreen mode Exit fullscreen mode

Output:

Auto-merging app.js
CONFLICT (content): Merge conflict in app.js
Automatic merge failed.
Enter fullscreen mode Exit fullscreen mode

Congratulations.
You now have a merge conflict.

Reading Conflict Markers

Git inserts markers into the file.

<<<<<<< HEAD
const version = "v2";
=======
const version = "v1";
>>>>>>> feature-login
Enter fullscreen mode Exit fullscreen mode

Understanding these markers is crucial.

HEAD

Represents your current branch.

const version = "v2";
Enter fullscreen mode Exit fullscreen mode

Incoming Changes
Represents the branch being merged.

const version = "v1";
Enter fullscreen mode Exit fullscreen mode

Separator

=======
Enter fullscreen mode Exit fullscreen mode

Separates both versions.

How to Resolve the Conflict

You have three choices.

Option 1: Keep Current Changes

const version = "v2";

Enter fullscreen mode Exit fullscreen mode

Use when current branch is correct.

Option 2: Keep Incoming Changes

const version = "v1";
Enter fullscreen mode Exit fullscreen mode

Use when merged branch contains desired code.

Option 3: Combine Both

Often the best solution.

const version = process.env.VERSION || "v2";
Enter fullscreen mode Exit fullscreen mode

This preserves both ideas.

Completing the Merge

After editing:

git add app.js
Enter fullscreen mode Exit fullscreen mode

Then:

git commit
Enter fullscreen mode Exit fullscreen mode

Git creates a merge commit.

Verify:

git log --oneline --graph
Enter fullscreen mode Exit fullscreen mode

Resolving Conflicts Using VS Code

VS Code provides built-in conflict resolution tools.
You'll see buttons such as:

  • Accept Current Change
  • Accept Incoming Change
  • Accept Both Changes
  • Compare Changes

Example:

Current Change
Incoming Change

Enter fullscreen mode Exit fullscreen mode

Instead of manually editing markers, simply choose the appropriate option.
This is often the fastest approach.

Handling Conflicts During Rebase

Many teams prefer rebasing.

Example:

git rebase main
Enter fullscreen mode Exit fullscreen mode

Conflict appears:

CONFLICT (content)
Enter fullscreen mode Exit fullscreen mode

Resolve file.
Then continue:

git add .
git rebase --continue
Enter fullscreen mode Exit fullscreen mode

If needed:

git rebase --abort
Enter fullscreen mode Exit fullscreen mode

This restores branch to its original state.

Advanced Conflict Resolution Commands

View Conflicted Files

git status
Enter fullscreen mode Exit fullscreen mode

Output:

both modified: app.js
both modified: config.js
Enter fullscreen mode Exit fullscreen mode

See Exact Differences

git diff
Enter fullscreen mode Exit fullscreen mode

Useful for understanding what changed.

Abort Entire Merge

git merge --abort
Enter fullscreen mode Exit fullscreen mode

Returns repository to pre-merge state.

View Merge History

git log --graph --all --decorate
Enter fullscreen mode Exit fullscreen mode

Helps visualize branch history.

Best Practices Used by Professional Teams

1. Pull Latest Changes Frequently

Don't work on stale code.

git pull origin main
Enter fullscreen mode Exit fullscreen mode

Regular updates reduce conflicts.

2. Keep Branches Short-Lived

Bad:

Feature branch open for 3 weeks
Enter fullscreen mode Exit fullscreen mode

Good:

Feature branch merged within 1-3 days
Enter fullscreen mode Exit fullscreen mode

3. Create Smaller Pull Requests

Smaller PRs:

  • Easier reviews
  • Faster merges
  • Fewer conflicts

4. Use Feature Flags

Instead of long-lived branches:

if (featureFlagEnabled) {
  // new feature
}

Enter fullscreen mode Exit fullscreen mode

This minimizes divergence.

5. Communicate Early

If multiple developers are modifying:

  • Authentication
  • Payments
  • Shared components Coordinate before coding. A 5-minute discussion can save hours of conflict resolution.

Common Mistakes Developers Make

Deleting Conflict Markers Incorrectly
Wrong:

<<<<<<< HEAD
Enter fullscreen mode Exit fullscreen mode

Leaving markers causes syntax errors.
Always remove them completely.

Resolving Without Understanding Code

Never blindly click:

Accept Current
Enter fullscreen mode Exit fullscreen mode

or

Accept Incoming
Enter fullscreen mode Exit fullscreen mode

Understand the business logic first.

Skipping Testing

After resolving:

npm test
Enter fullscreen mode Exit fullscreen mode

or

npm run build
Enter fullscreen mode Exit fullscreen mode

Always verify functionality.

Real Production Story

A team member once resolved a conflict by keeping only their own changes.
The application compiled successfully.
However, the conflict resolution removed a critical authentication check added by another developer.
The bug wasn't discovered until production deployment.
The lesson:
Successful merging doesn't guarantee correct merging.
Always review the final code carefully.

Key Takeaways

✅ Merge conflicts are normal.
✅ They happen when Git cannot decide between competing changes.
✅ Learn to read conflict markers confidently.
✅ Use tools like VS Code to simplify resolution.
✅ Pull frequently and keep branches small.
✅ Test thoroughly after resolving conflicts.

Conclusion

Merge conflicts are a fundamental part of collaborative software development. Every professional developer encounters them regularly.

The difference between a beginner and an experienced engineer isn't avoiding merge conflicts—it's resolving them confidently and safely.

Once you understand how Git thinks, merge conflicts become less of a roadblock and more of a routine step in your development workflow.

Have you ever faced a merge conflict that took hours to resolve? Share your story in the comments—I'd love to hear how you handled it! 🚀

Top comments (0)