Как увидеть изменения между двумя коммитами без промежуточных коммитов?

Как сделать так, чтобы git diff показывал только разницу между двумя коммитами, исключая другие коммиты между ними?

Комментарии к вопросу (3)

вы можете просто передать 2 коммита в git diff, как:

-> git diff 0da94be  59ff30c > my.patch
-> git apply my.patch
Комментарии (14)

Запрос разницы /между/ двумя коммитами без включения промежуточных коммитов не имеет смысла. Коммиты - это просто снимки содержимого репозитория; запрос о разнице между двумя коммитами обязательно включает их. Поэтому вопрос заключается в том, что же вы на самом деле ищете?

Как предложил Уильям, выборка "вишни" может дать вам дельту одного коммита, наложенного поверх другого. То есть:

$ git checkout 012345
$ git cherry-pick -n abcdef
$ git diff --cached

Берётся коммит 'abcdef', сравнивается с его непосредственным предком, затем применяется эта разница поверх '012345'. Затем отображается это новое отличие - единственное изменение заключается в том, что контекст исходит от '012345', а не от 'abcdef' непосредственного предка. Конечно, могут возникнуть конфликты и т.д., так что в большинстве случаев это не очень полезный процесс.

Если вас интересует только сам abcdef, вы можете сделать следующее:

$ git log -u -1 abcdef

Это сравнит abcdef с его непосредственным предком, в одиночку, и обычно это то, что вам нужно.

И конечно

$ git diff 012345..abcdef

дает вам все различия между этими двумя коммитами.

Это поможет лучше понять, чего вы пытаетесь достичь - как я уже говорил, запрос разницы между двумя коммитами без того, что находится между ними, на самом деле не имеет смысла.

Комментарии (6)

Чтобы сравнить два git-коммита 12345 и abcdef как патчи, можно использовать команду diff как


diff 
Комментарии (6)
git diff   path

Пример:

git diff commit1 commit2 config/routes.rb

Это показывает разницу в этом файле между этими коммитами.

Комментарии (0)

Допустим, у вас есть это

A
|
B    A0
|    |
C    D
\   /
  |
 ...

И вы хотите убедиться, что A совпадает с A0.

Это поможет:

$ git diff B A > B-A.diff
$ git diff D A0 > D-A0.diff
$ diff B-A.diff D-A0.diff
Комментарии (1)

Для проверки полных изменений:

  git diff  

Для проверки только измененных / добавленных / удаленных файлов:

  git diff   --name-only

ПРИМЕЧАНИЕ : Для проверки diff без фиксации между ними вам не нужно помещать идентификаторы коммита.

Комментарии (0)

Предположим, вы хотите увидеть разницу между коммитами 012345 и abcdef. Следующее должно сделать то, что вы хотите:

$ git checkout 012345
$ git cherry-pick -n abcdef
$ git diff --cached
Комментарии (1)

Что насчет этого?

git diff abcdef 123456 | less

Удобно просто трубить его до меньшего, если вы хотите сравнить много разных различий на лету.

Комментарии (0)

Мои настройки alias в файле ~/.bashrc для git diff:

alias gdca='git diff --cached' # diff between your staged file and the last commit
alias gdcc='git diff HEAD{,^}' # diff between your recent tow commits

Английский не мой родной язык, извините за ошибки при наборе текста

Комментарии (0)

Начиная с Git 2.19, вы можете просто использовать:

`git range-diff rev1...rev2

  • сравните два дерева коммитов, начиная с их общего предка

или `git range-diff rev1 ~..rev1 rev2 ~..rev2

  • сравнить изменения, внесенные 2 заданными коммитами
Комментарии (0)

Я написал сценарий, который отображает разницу между двумя коммитами, хорошо работает на Ubuntu.

https://gist.github.com/jacobabrahamb4/a60624d6274ece7a0bd2d141b53407bc

#!/usr/bin/env python
import sys, subprocess, os

TOOLS = ['bcompare', 'meld']

def getTool():
    for tool in TOOLS:
        try:
            out = subprocess.check_output(['which', tool]).strip()
            if tool in out:
                return tool
        except subprocess.CalledProcessError:
            pass
    return None

def printUsageAndExit():
    print 'Usage: python bdiff.py   '
    print 'Example: python bdiff.py  0 1'
    print 'Example: python bdiff.py  fhejk7fe d78ewg9we'
    print 'Example: python bdiff.py  0 d78ewg9we'
    sys.exit(0)

def getCommitIds(name, first, second):
    commit1 = None
    commit2 = None
    try:
        first_index = int(first) - 1
        second_index = int(second) - 1
        if int(first) < 0 or int(second) < 0:
            print "Cannot handle negative values: "
            sys.exit(0)
        logs = subprocess.check_output(['git', '-C', name, 'log', '--oneline', '--reverse']).split('\n')
        if first_index >= 0:
            commit1 = logs[first_index].split(' ')[0]
        if second_index >= 0:
            commit2 = logs[second_index].split(' ')[0]
    except ValueError:
        if first != '0':
            commit1 = first
        if second != '0':
            commit2 = second
    return commit1, commit2

def validateCommitIds(name, commit1, commit2):
    if commit1 == None and commit2 == None:
        print "Nothing to do, exit!"
        return False
    try:
        if commit1 != None:
            subprocess.check_output(['git', '-C', name, 'cat-file', '-t', commit1]).strip()
        if commit2 != None:
            subprocess.check_output(['git', '-C', name, 'cat-file', '-t', commit2]).strip()
    except subprocess.CalledProcessError:
        return False
    return True

def cleanup(commit1, commit2):
        subprocess.check_output(['rm', '-rf', '/tmp/'+(commit1 if commit1 != None else '0'), '/tmp/'+(commit2 if commit2 != None else '0')])

def checkoutCommit(name, commit):
    if commit != None:
        subprocess.check_output(['git', 'clone', name, '/tmp/'+commit])
        subprocess.check_output(['git', '-C', '/tmp/'+commit, 'checkout', commit])
    else:
        subprocess.check_output(['mkdir', '/tmp/0'])

def compare(tool, commit1, commit2):
        subprocess.check_output([tool, '/tmp/'+(commit1 if commit1 != None else '0'), '/tmp/'+(commit2 if commit2 != None else '0')])

if __name__=='__main__':
    tool = getTool()
    if tool == None:
        print "No GUI diff tools"
        sys.exit(0)
    if len(sys.argv) != 4:
        printUsageAndExit()

    name, first, second = None, 0, 0
    try:
        name, first, second = sys.argv[1], sys.argv[2], sys.argv[3]
    except IndexError:
        printUsageAndExit()

    commit1, commit2 = getCommitIds(name, first, second)

    if not validateCommitIds(name, commit1, commit2):
        sys.exit(0)

    cleanup(commit1, commit2)
    checkoutCommit(name, commit1)
    checkoutCommit(name, commit2)

    try:
        compare(tool, commit1, commit2)
    except KeyboardInterrupt:
        pass
    finally:
        cleanup(commit1, commit2)
    sys.exit(0)
Комментарии (0)

Мои настройки alias в файле ~/.zshrc для git diff:

alias gdf='git diff HEAD{'^',}' # diff between your recent tow commits

Спасибо @Jinmiao Luo

Комментарии (0)