it-swarm.com.ru

Когда вы используете git rebase вместо git merge?

Когда рекомендуется использовать git rebase против git merge?

Нужно ли объединять после успешной перезагрузки?

1392
Coocoo4Cocoa

Укороченная версия

  • Слияние принимает все изменения в одной ветви и объединяет их в другую ветку за один коммит.
  • Ребаз говорит, что я хочу, чтобы точка, в которой я разветвился, переместилась в новую отправную точку.

Так когда вы используете один из них?

Сливаться

  • Допустим, вы создали ветку с целью разработки одной функции. Если вы хотите вернуть эти изменения в master, вы, вероятно, захотите объединить (вам не нужно поддерживать все временные коммиты).

Rebase

  • Второй сценарий: если вы начали заниматься какой-то разработкой, а другой разработчик внес несвязанные изменения. Вы, вероятно, хотите вытащить и затем rebase, чтобы основать свои изменения из текущей версии из репозитория.
1054
Rob Di Marco

Все просто, с rebase вы говорите использовать другую ветку как новую base для вашей работы.

Если у вас есть, например, ветвь master и вы создаете ветвь для реализации новой функции, скажем, вы называете ее cool-feature, конечно, основная ветвь является базой для вашей новой функции.

Теперь в определенный момент вы хотите добавить новую функцию, которую вы реализовали в ветке master. Вы можете просто переключиться на master и объединить ветвь cool-feature:

$ git checkout master
$ git merge cool-feature

но таким образом добавляется новый фиктивный коммит, если вы хотите избежать истории спагетти, вы можете rebase:

$ git checkout cool-feature
$ git rebase master

а затем объединить его в master:

$ git checkout master
$ git merge cool-feature

На этот раз, поскольку ветка темы имеет те же коммиты master и коммиты с новой функцией, слияние будет просто ускоренной перемоткой вперед.

336
Aldo 'xoen' Giambelluca

Чтобы дополнить мой собственный ответ упомянуто TSamper ,

  • перед слиянием довольно часто хорошая идея сделать перебазирование, потому что идея заключается в том, что вы интегрируете в свою ветвь Y работу ветви B, в которую вы будете сливаться.
    Но опять же, перед объединением вы разрешаете любой конфликт в вашей ветке (то есть: "rebase", как в "воспроизвести мою работу в моей ветке, начиная с недавнего точка из ветки B)
    Если все сделано правильно, последующее слияние из вашей ветви в ветку B может быть ускорено.

  • влияние слияния непосредственно на целевую ветвь B, что означает, что слияния лучше быть тривиальными, в противном случае эта ветвь B может быть долгой, чтобы вернуться в стабильное состояние (время для разрешения всех конфликтов)


точка слияния после ребазирования?

В случае, который я описываю, я перебазирую B в свою ветку, просто чтобы иметь возможность воспроизвести мою работу с более поздней точки из B, но оставаясь в моей ветви.
В этом случае все еще необходимо объединение, чтобы перенести мою "воспроизведенную" работу на B.

Другой сценарий ( описанный, например, в Git Ready ) состоит в том, чтобы перенести вашу работу непосредственно в B посредством перебазирования (которое сохраняет все ваши коммиты Nice или даже дает вам возможность переупорядочить их через интерактивный ребаз).
В этом случае (когда вы перебазируете, находясь в ветке B), вы правы: дальнейшее слияние не требуется:

Git-дерево по умолчанию, когда мы не слили и не перебазировали

rebase1

мы получаем перебазировкой:

rebase3

Этот второй сценарий полностью посвящен тому, как вернуть новую функцию в мастер.

Моя точка зрения, описывая первый сценарий перебазирования, состоит в том, чтобы напомнить всем, что перебазировка также может быть использована в качестве предварительного шага к этому (то есть, "получить новую функцию обратно в мастер").
Вы можете использовать rebase, чтобы сначала привести master "в" ветку новой функции: rebase будет воспроизводить коммиты новой функции из HEAD master, но все еще в ветке новой функции, эффективно перемещая начальную точку ветки из старый мастер коммит HEAD-master.
Это позволяет вам разрешать любые конфликты в вашей ветви (то есть изолированно, в то же время позволяя мастеру продолжать развиваться параллельно, если этап разрешения конфликта занимает слишком много времени). ).
Затем вы можете переключиться на master и объединить new-feature (или перебазировать new-feature в master, если вы хотите сохранить коммиты, сделанные в вашей ветви new-feature).

Так:

  • "rebase vs. merge" можно рассматривать как два способа импорта работы, скажем, master.
  • Но "ребаз, а затем слияние" может быть допустимым рабочим процессом, чтобы сначала разрешить конфликт изолированно, а затем вернуть свою работу.
260
VonC

TL; DR

Если у вас есть какие-либо сомнения, используйте слияние.

Короткий ответ

Единственные различия между ребазом и слиянием:

  • Результирующая древовидная структура истории (обычно заметная только при просмотре графа коммитов) отличается (одна будет иметь ветви, а другая - нет).
  • Слияние обычно создает дополнительный коммит (например, узел в дереве).
  • Слияние и перебазировка будут обрабатывать конфликты по-разному. Rebase будет представлять конфликты по одному коммиту за раз, когда слияние представляет их все одновременно.

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

Длинный ответ

Есть несколько факторов, которые вы должны учитывать при выборе операции.

Разделяется ли ветка, от которой вы получаете изменения, с другими разработчиками за пределами вашей команды (например, с открытым исходным кодом, общедоступным)?

Если это так, не перебазируйте. Rebase уничтожает ветку, и эти разработчики будут иметь поврежденные/несовместимые репозитории, если они не используют git pull --rebase. Это хороший способ быстро расстроить других разработчиков.

Насколько опытна ваша команда разработчиков?

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

Я работал в командах, в которых все разработчики пришли из того времени, когда компании могли позволить себе выделенный персонал, чтобы заниматься ветвлением и слиянием. Эти разработчики мало знают о Git и не хотят много знать. В этих командах я бы ни за что не рискнул порекомендовать перебазирование.

Представляет ли сама ветка полезную информацию

Некоторые команды используют модель ветвления на функцию, где каждая ветвь представляет функцию (или исправление, или подфункцию и т.д.). В этой модели ветвь помогает идентифицировать наборы связанных коммитов. Например, можно быстро восстановить функцию, отменив объединение этой ветви (если честно, это редкая операция). Или отличить особенность, сравнивая две ветви (чаще). Rebase уничтожит ветку, и это не будет простым.

Я также работал над командами, которые использовали модель ветвления на разработчика (мы все были там). В этом случае сама ветка не передает никакой дополнительной информации (у коммита уже есть автор). Там не будет никакого вреда в перебазировании.

Можете ли вы отменить слияние по какой-либо причине?

Возврат (как при отмене) перебазирования является значительно трудным и/или невозможным (если у перебазирования были конфликты) по сравнению с возвратом слияния. Если вы считаете, что есть шанс вернуться, используйте merge.

Ты работаешь в команде? Если да, готовы ли вы использовать подход "все или ничего" в этой отрасли?

Операции перебазирования должны быть извлечены с соответствующим git pull --rebase. Если вы работаете самостоятельно, вы можете вспомнить, какой из них следует использовать в соответствующее время. Если вы работаете в команде, это будет очень трудно координировать. Вот почему большинство рабочих процессов rebase рекомендуют использовать rebase для всех слияний (и git pull --rebase для всех извлечений).

Общие мифы

Слияние уничтожает историю

Предполагая, что у вас есть следующее слияние:

    B -- C
   /      \
  A--------D

Некоторые люди утверждают, что слияние "уничтожает" историю коммитов, потому что если бы вы смотрели журнал только основной ветки (A - D), вы пропустили бы важные сообщения коммитов, содержащиеся в B и C.

Если бы это было правдой, у нас не было бы такие вопросы . По сути, вы увидите B и C, если явно не попросите их не видеть (используя --first-parent). Это очень легко попробовать для себя.

Rebase позволяет более безопасные/простые слияния

Два подхода объединяются по-разному, но не ясно, что один всегда лучше другого, и это может зависеть от рабочего процесса разработчика. Например, если разработчик стремится к регулярной фиксации (например, может быть, он фиксирует дважды в день при переходе с работы на дом), тогда для данной ветви может быть много коммитов. Многие из этих коммитов могут не выглядеть как конечный продукт (я склонен к рефакторингу своего подхода один или два раза для каждой функции). Если кто-то еще работает над соответствующей областью кода и пытается отменить мои изменения, это может быть довольно утомительной операцией.

Rebase круче/сексуальнее/профессиональнее

Если вы хотите использовать псевдоним rm для rm -rf, чтобы "сэкономить время", то, возможно, ребаз для вас.

Мои два цента

Я всегда думаю, что когда-нибудь я столкнусь со сценарием, в котором git rebase является отличным инструментом, который решает проблему. Как и я думаю, я столкнусь со сценарием, в котором git reflog - это замечательный инструмент, который решает мою проблему. Я работал с Git более пяти лет. Этого не случилось.

Грязные истории никогда не были для меня проблемой. Я никогда не читаю историю коммитов, как захватывающий роман. Большую часть времени мне нужна история, я в любом случае собираюсь использовать git blame или git bisect. В этом случае наличие коммита слияния на самом деле полезно для меня, потому что, если слияние привело к проблеме, которая является важной информацией для меня.

Обновление (4/2017)

Я чувствую себя обязанным упомянуть, что лично смягчил использование rebase, хотя мой общий совет остается в силе. Недавно я много взаимодействовал с проектом Angular 2 Material . Они использовали rebase, чтобы сохранить очень чистую историю коммитов. Это позволило мне очень легко увидеть, что фиксация исправила данный дефект и была ли эта фиксация включена в релиз. Это служит хорошим примером правильного использования rebase.

173
Pace

Многие ответы здесь говорят, что слияние превращает все ваши коммиты в один, и поэтому предлагают использовать rebase для сохранения ваших коммитов. Это неверно. И плохая идея, если вы уже выдвинули свои коммиты .

Слияние не стирает ваши коммиты. Слияние сохраняет историю! (просто посмотрите на gitk) Rebase переписывает историю, что является плохой вещью после того, как вы нажали ее.

Используйте merge, а не rebase всякий раз, когда вы уже нажали.

Вот Линус '(автор git) возьми это на себя . Это действительно хорошее чтение. Или вы можете прочитать мою собственную версию той же идеи ниже.

Перебазирование ветки на мастера:

  • дает неверное представление о том, как были созданы коммиты
  • загрязняет хозяина кучей промежуточных коммитов, которые, возможно, не были хорошо протестированы
  • может на самом деле вводить разрывы сборки для этих промежуточных коммитов из-за изменений, которые были внесены в мастерскую между тем, когда исходная ветка темы была создана и когда она была перебазирована.
  • затрудняет поиск хороших мест в мастере для оформления заказа.
  • Заставляет временные метки коммитов не выравниваться с их хронологическим порядком в дереве. Таким образом, вы увидите, что коммит A предшествует коммиту B в master, но коммит B был создан первым. (Какие?!)
  • Создает больше конфликтов, потому что отдельные коммиты в ветке темы могут включать конфликты слияния, которые должны быть разрешены индивидуально (дальнейшая история о том, что происходило в каждом коммите).
  • это переписать историю. Если перебазируемая ветвь была перенесена куда-либо (предоставлена ​​кому-либо, кроме вас), то вы облажались с каждым, у кого есть эта ветка, с тех пор, как вы переписали историю.

Напротив, объединение ветки темы в мастер:

  • сохраняет историю создания веток темы, включая любые слияния от главной ветки к теме, чтобы поддерживать ее актуальность. Вы действительно получите точное представление о том, с каким кодом работал разработчик, когда создавал.
  • master - это ветвь, состоящая в основном из слияний, и каждый из этих коммитов слияния, как правило, является "хорошей точкой" в истории, которую можно безопасно проверить, потому что именно там ветка темы была готова для интеграции.
  • все отдельные коммиты ветки темы сохраняются, в том числе тот факт, что они были в ветке темы, поэтому изоляция этих изменений является естественной, и вы можете углубиться в нее при необходимости.
  • конфликты слияния должны быть разрешены только один раз (в точке слияния), поэтому изменения промежуточной фиксации, сделанные в ветке темы, не должны решаться независимо.
  • можно сделать несколько раз плавно. Если вы периодически интегрируете ветку своей темы в мастеринг, люди могут продолжать строить ветку тем и объединять ее независимо.
171
Andrew Arnott

Слияние означает: создать один новый коммит, который объединит мои изменения в место назначения.

Rebase означает: создать новую серию коммитов, используя мой текущий набор коммитов в качестве подсказок. Другими словами, посчитайте, как бы выглядели мои изменения, если бы я начал их вносить с того момента, к которому я перебираю. Таким образом, после перебазирования вам может потребоваться повторно протестировать ваши изменения, и во время перебазирования у вас может возникнуть несколько конфликтов.

Учитывая это, почему бы вам сделать ребаз? Просто чтобы история развития была понятной. Допустим, вы работаете над функцией X, и когда вы закончите, вы объедините свои изменения. У места назначения теперь будет один коммит, который будет что-то говорить в духе "Добавленной функции X". Теперь, вместо слияния, если вы перебазировали, а затем слили, целевая история разработки будет содержать все отдельные коммиты в одной логической последовательности. Это значительно упрощает последующее рассмотрение изменений. Представьте, как трудно было бы просмотреть историю разработки, если бы 50 разработчиков постоянно объединяли различные функции.

Тем не менее, если вы уже выдвинули ветку, над которой работаете, вам не нужно делать ребаз, а вместо этого объединять. Для веток, которые не были переданы вверх по потоку, выполните ребазирование, тестирование и объединение.

В другой раз вы, возможно, захотите сделать ребазинг, когда вы хотите избавиться от коммитов из вашей ветки, прежде чем переходить вверх по течению. Например: коммиты, которые вводят некоторый код отладки на ранних этапах, а другие коммиты в дальнейшем очищают этот код. Единственный способ сделать это - выполнить интерактивную перебазировку: git rebase -i <branch/commit/tag>

ОБНОВЛЕНИЕ: Вы также хотите использовать rebase, когда используете Git для взаимодействия с системой контроля версий, которая не поддерживает нелинейную историю (например, Subversion). При использовании моста git-svn очень важно, чтобы изменения, которые вы объединяете обратно в Subversion, представляли собой последовательный список изменений поверх самых последних изменений в транке. Есть только два способа сделать это: (1) заново создать изменения вручную и (2) использовать команду rebase, что намного быстрее.

ОБНОВЛЕНИЕ 2: Еще один способ думать о перебазировании заключается в том, что он обеспечивает своего рода сопоставление вашего стиля разработки со стилем, принятым в репозитории, к которому вы привязываетесь. Допустим, вам нравится совершать небольшие, маленькие кусочки. У вас есть один коммит, чтобы исправить опечатку, один коммит, чтобы избавиться от неиспользуемого кода и так далее. К тому времени, как вы закончите то, что вам нужно сделать, у вас будет длинный ряд коммитов. Теперь давайте предположим, что репозиторий, который вы делаете, поощряет большие коммиты, поэтому для работы, которую вы делаете, можно ожидать один или, может быть, два коммита. Как вы берете свою строку коммитов и сжимаете их до ожидаемого? Вы бы использовали интерактивную перебазировку и разделили свои крошечные коммиты на меньшее количество больших кусков. То же самое верно, если нужно было обратное - если ваш стиль был несколькими большими коммитами, но репо требовало длинных строк небольших коммитов. Вы должны использовать ребаз для этого. Если вы слились вместо этого, вы теперь перенесли свой стиль коммитов в основной репозиторий. Если разработчиков много, вы можете себе представить, как трудно было бы проследить историю с несколькими различными стилями фиксации через некоторое время.

ОБНОВЛЕНИЕ3: Does one still need to merge after a successful rebase? Да, вы делаете. Причина в том, что ребаз по сути включает в себя "сдвиг" коммитов. Как я уже говорил выше, эти коммиты рассчитываются, но если у вас было 14 коммитов с точки ветвления, то при условии, что с вашей перебазировкой все в порядке, вы будете на 14 коммитов вперед (от точки, на которую вы перебазируете) после ребаз сделан. У вас была ветка перед ребазой. У вас будет ветвь такой же длины после. Вам все еще нужно объединиться, прежде чем публиковать свои изменения. Другими словами, сделайте ребаз столько раз, сколько хотите (опять же, только если вы не выдвинули свои изменения вверх по течению). Объединять только после того, как вы сделаете ребаз.

69
Carl

до слияния/ребазирования:

A <- B <- C    [master]
^
 \
  D <- E       [branch]

после git merge master:

A <- B <- C
^         ^
 \         \
  D <- E <- F

после git rebase master:

A <- B <- C <- D' <- E'

(A, B, C, D, E и F являются коммитами)

этот пример и гораздо более хорошо иллюстрированная информация о git можно найти здесь: http://excess.org/article/2008/07/ogre-git-tutorial/

60
guybrush

Хотя объединение, безусловно, самый простой и распространенный способ интеграции изменений, он не единственный: Rebase является альтернативным средством интеграции.

Понимание слияния лучше

Когда Git выполняет слияние, он ищет три коммита:

  • (1) Общий предковый коммит Если вы следите за историей двух ветвей в проекте, у них всегда есть хотя бы одна общая фиксация: на данный момент обе ветви имели одинаковое содержание, а затем развивались по-разному.
  • (2) + (3) Конечные точки каждой ветви Цель интеграции - объединить текущие состояния двух ветвей. Поэтому их соответствующие последние редакции представляют особый интерес. Объединение этих трех коммитов приведет к интеграции, к которой мы стремимся.

Fast Forward или Merge Commit

В очень простых случаях одна из двух ветвей не имеет никаких новых коммитов с тех пор, как произошло ветвление - ее последний коммит все еще остается общим предком.

enter image description here

В этом случае выполнить интеграцию очень просто: Git может просто добавить все коммиты другой ветки поверх коммитов общего предка. В Git эта простейшая форма интеграции называется слиянием "ускоренной перемотки вперед". Затем обе ветви имеют одинаковую историю.

enter image description here

Однако во многих случаях обе ветви двигались вперед по отдельности. enter image description here

Чтобы сделать интеграцию, Git должен будет создать новый коммит, содержащий различия между ними - коммит слияния.

enter image description here

Human Commits и Merge Commits

Обычно коммит тщательно создается человеком. Это значимый модуль, который оборачивает только связанные изменения и комментирует их.

Коммит слияния немного отличается: вместо того, чтобы быть созданным разработчиком, он создается Git автоматически. И вместо того, чтобы оборачивать набор связанных изменений, его цель - соединить две ветви, как узел. Если вы хотите понять операцию слияния позже, вам нужно взглянуть на историю обеих ветвей и соответствующий граф коммитов.

Интеграция с Rebase

Некоторые люди предпочитают обходиться без таких автоматических коммитов слияния. Вместо этого они хотят, чтобы история проекта выглядела так, как будто он развивался по одной прямой линии. Не осталось никаких признаков того, что он был разделен на несколько веток в какой-то момент.

enter image description here

Давайте шаг за шагом пройдемся по операции rebase. Сценарий такой же, как и в предыдущих примерах: мы хотим интегрировать изменения из ветви B в ветку A, но теперь с помощью rebase.

enter image description here

Мы сделаем это в три этапа

  1. git rebase branch-A // syncs the history with branch-A
  2. git checkout branch-A // change the current branch to branch-A
  3. git merge branch-B // merge/take the changes from branch-B to branch-A

Во-первых, Git "отменит" все коммиты на ветви-A, которые произошли после того, как линии начали ветвиться (после коммитов общего предка). Однако, конечно, он не откажется от них: вместо этого вы можете думать об этих коммитах как о "временно спасенных".

enter image description here

Затем он применяет коммиты из Branch-B, которые мы хотим интегрировать. На данный момент обе ветви выглядят одинаково.

enter image description here

На последнем шаге новые коммиты в ответвлении A теперь повторно применяются - но на новой позиции, поверх интегрированных коммитов из ответвления B (они переоснованы). Результат выглядит так, как будто развитие произошло по прямой линии. Вместо фиксации слияния, содержащей все объединенные изменения, была сохранена исходная структура фиксации.

enter image description here

Наконец, вы получаете чистую ветвь branch-A без нежелательных и автоматически генерируемых коммитов.

Примечание: Взято с удивительного postgit-tower . недостатки rebase также хорошо читаются в том же посте.

52
Abdullah Khan

Это предложение получает это:

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

Источник: http://www.git-scm.com/book/en/v2/Git-Branching-Rebasing#Rebase-vs.-Merge

27
Joaquin Sargiotto

Этот ответ широко ориентирован на Git Flow . Таблицы были сгенерированы с помощью Nice ASCII Table Generator и деревьев истории с помощью этой замечательной команды ( с псевдонимом as git lg):

git log --graph --abbrev-commit --decorate --date=format:'%Y-%m-%d %H:%M:%S' --format=format:'%C(bold blue)%h%C(reset) - %C(bold cyan)%ad%C(reset) %C(bold green)(%ar)%C(reset)%C(bold yellow)%d%C(reset)%n''          %C(white)%s%C(reset) %C(dim white)- %an%C(reset)'

Таблицы расположены в обратном хронологическом порядке, чтобы быть более совместимыми с деревьями истории. Сначала посмотрите также разницу между git merge и git merge --no-ff (обычно вы хотите использовать git merge --no-ff, так как ваша история выглядит ближе к реальности):

git merge

Команды:

Time          Branch "develop"             Branch "features/foo"
------- ------------------------------ -------------------------------
15:04   git merge features/foo
15:03                                  git commit -m "Third commit"
15:02                                  git commit -m "Second commit"
15:01   git checkout -b features/foo
15:00   git commit -m "First commit"

Результат:

* 142a74a - YYYY-MM-DD 15:03:00 (XX minutes ago) (HEAD -> develop, features/foo)
|           Third commit - Christophe
* 00d848c - YYYY-MM-DD 15:02:00 (XX minutes ago)
|           Second commit - Christophe
* 298e9c5 - YYYY-MM-DD 15:00:00 (XX minutes ago)
            First commit - Christophe

git merge --no-ff

Команды:

Time           Branch "develop"              Branch "features/foo"
------- -------------------------------- -------------------------------
15:04   git merge --no-ff features/foo
15:03                                    git commit -m "Third commit"
15:02                                    git commit -m "Second commit"
15:01   git checkout -b features/foo
15:00   git commit -m "First commit"

Результат:

*   1140d8c - YYYY-MM-DD 15:04:00 (XX minutes ago) (HEAD -> develop)
|\            Merge branch 'features/foo' - Christophe
| * 69f4a7a - YYYY-MM-DD 15:03:00 (XX minutes ago) (features/foo)
| |           Third commit - Christophe
| * 2973183 - YYYY-MM-DD 15:02:00 (XX minutes ago)
|/            Second commit - Christophe
* c173472 - YYYY-MM-DD 15:00:00 (XX minutes ago)
            First commit - Christophe

git merge vs git rebase

Первое замечание: всегда объединяйте функции в разработке, никогда не перебирайте разработки из функций . Это является следствием Золотое правило перебазирования :

Золотое правило git rebase - никогда не использовать его в публичных ветвях.

другими словами :

Никогда не опровергайте ничего, что вы куда-то толкнули.

Я бы лично добавил: , если это не ветка функций, а вы и ваша команда не знаете о последствиях .

Таким образом, вопрос git merge vs git rebase применим почти только к ветвям объектов (в следующих примерах --no-ff всегда использовался при объединении). Обратите внимание, что, поскольку я не уверен, что есть одно лучшее решение ( дискуссия существует ), я лишь расскажу, как ведут себя обе команды. В моем случае я предпочитаю использовать git rebase, поскольку он создает более красивое дерево истории :)

Между функциональными ветвями

git merge

Команды:

Time           Branch "develop"              Branch "features/foo"           Branch "features/bar"
------- -------------------------------- ------------------------------- --------------------------------
15:10   git merge --no-ff features/bar
15:09   git merge --no-ff features/foo
15:08                                                                    git commit -m "Sixth commit"
15:07                                                                    git merge --no-ff features/foo
15:06                                                                    git commit -m "Fifth commit"
15:05                                                                    git commit -m "Fourth commit"
15:04                                    git commit -m "Third commit"
15:03                                    git commit -m "Second commit"
15:02   git checkout -b features/bar
15:01   git checkout -b features/foo
15:00   git commit -m "First commit"

Результат:

*   c0a3b89 - YYYY-MM-DD 15:10:00 (XX minutes ago) (HEAD -> develop)
|\            Merge branch 'features/bar' - Christophe
| * 37e933e - YYYY-MM-DD 15:08:00 (XX minutes ago) (features/bar)
| |           Sixth commit - Christophe
| *   eb5e657 - YYYY-MM-DD 15:07:00 (XX minutes ago)
| |\            Merge branch 'features/foo' into features/bar - Christophe
| * | 2e4086f - YYYY-MM-DD 15:06:00 (XX minutes ago)
| | |           Fifth commit - Christophe
| * | 31e3a60 - YYYY-MM-DD 15:05:00 (XX minutes ago)
| | |           Fourth commit - Christophe
* | |   98b439f - YYYY-MM-DD 15:09:00 (XX minutes ago)
|\ \ \            Merge branch 'features/foo' - Christophe
| |/ /
|/| /
| |/
| * 6579c9c - YYYY-MM-DD 15:04:00 (XX minutes ago) (features/foo)
| |           Third commit - Christophe
| * 3f41d96 - YYYY-MM-DD 15:03:00 (XX minutes ago)
|/            Second commit - Christophe
* 14edc68 - YYYY-MM-DD 15:00:00 (XX minutes ago)
            First commit - Christophe

git rebase

Команды:

Time           Branch "develop"              Branch "features/foo"           Branch "features/bar"
------- -------------------------------- ------------------------------- -------------------------------
15:10   git merge --no-ff features/bar
15:09   git merge --no-ff features/foo
15:08                                                                    git commit -m "Sixth commit"
15:07                                                                    git rebase features/foo
15:06                                                                    git commit -m "Fifth commit"
15:05                                                                    git commit -m "Fourth commit"
15:04                                    git commit -m "Third commit"
15:03                                    git commit -m "Second commit"
15:02   git checkout -b features/bar
15:01   git checkout -b features/foo
15:00   git commit -m "First commit"

Результат:

*   7a99663 - YYYY-MM-DD 15:10:00 (XX minutes ago) (HEAD -> develop)
|\            Merge branch 'features/bar' - Christophe
| * 708347a - YYYY-MM-DD 15:08:00 (XX minutes ago) (features/bar)
| |           Sixth commit - Christophe
| * 949ae73 - YYYY-MM-DD 15:06:00 (XX minutes ago)
| |           Fifth commit - Christophe
| * 108b4c7 - YYYY-MM-DD 15:05:00 (XX minutes ago)
| |           Fourth commit - Christophe
* |   189de99 - YYYY-MM-DD 15:09:00 (XX minutes ago)
|\ \            Merge branch 'features/foo' - Christophe
| |/
| * 26835a0 - YYYY-MM-DD 15:04:00 (XX minutes ago) (features/foo)
| |           Third commit - Christophe
| * a61dd08 - YYYY-MM-DD 15:03:00 (XX minutes ago)
|/            Second commit - Christophe
* ae6f5fc - YYYY-MM-DD 15:00:00 (XX minutes ago)
            First commit - Christophe

От develop к функциональной ветви

git merge

Команды:

Time           Branch “develop"              Branch "features/foo"           Branch "features/bar"
------- -------------------------------- ------------------------------- -------------------------------
15:10   git merge --no-ff features/bar
15:09                                                                    git commit -m “Sixth commit"
15:08                                                                    git merge --no-ff development
15:07   git merge --no-ff features/foo
15:06                                                                    git commit -m “Fifth commit"
15:05                                                                    git commit -m “Fourth commit"
15:04                                    git commit -m “Third commit"
15:03                                    git commit -m “Second commit"
15:02   git checkout -b features/bar
15:01   git checkout -b features/foo
15:00   git commit -m “First commit"

Результат:

*   9e6311a - YYYY-MM-DD 15:10:00 (XX minutes ago) (HEAD -> develop)
|\            Merge branch 'features/bar' - Christophe
| * 3ce9128 - YYYY-MM-DD 15:09:00 (XX minutes ago) (features/bar)
| |           Sixth commit - Christophe
| *   d0cd244 - YYYY-MM-DD 15:08:00 (XX minutes ago)
| |\            Merge branch 'develop' into features/bar - Christophe
| |/
|/|
* |   5bd5f70 - YYYY-MM-DD 15:07:00 (XX minutes ago)
|\ \            Merge branch 'features/foo' - Christophe
| * | 4ef3853 - YYYY-MM-DD 15:04:00 (XX minutes ago) (features/foo)
| | |           Third commit - Christophe
| * | 3227253 - YYYY-MM-DD 15:03:00 (XX minutes ago)
|/ /            Second commit - Christophe
| * b5543a2 - YYYY-MM-DD 15:06:00 (XX minutes ago)
| |           Fifth commit - Christophe
| * 5e84b79 - YYYY-MM-DD 15:05:00 (XX minutes ago)
|/            Fourth commit - Christophe
* 2da6d8d - YYYY-MM-DD 15:00:00 (XX minutes ago)
            First commit - Christophe

git rebase

Команды:

Time           Branch “develop"              Branch "features/foo"           Branch "features/bar"
------- -------------------------------- ------------------------------- -------------------------------
15:10   git merge --no-ff features/bar
15:09                                                                    git commit -m “Sixth commit"
15:08                                                                    git rebase development
15:07   git merge --no-ff features/foo
15:06                                                                    git commit -m “Fifth commit"
15:05                                                                    git commit -m “Fourth commit"
15:04                                    git commit -m “Third commit"
15:03                                    git commit -m “Second commit"
15:02   git checkout -b features/bar
15:01   git checkout -b features/foo
15:00   git commit -m “First commit"

Результат:

*   b0f6752 - YYYY-MM-DD 15:10:00 (XX minutes ago) (HEAD -> develop)
|\            Merge branch 'features/bar' - Christophe
| * 621ad5b - YYYY-MM-DD 15:09:00 (XX minutes ago) (features/bar)
| |           Sixth commit - Christophe
| * 9cb1a16 - YYYY-MM-DD 15:06:00 (XX minutes ago)
| |           Fifth commit - Christophe
| * b8ddd19 - YYYY-MM-DD 15:05:00 (XX minutes ago)
|/            Fourth commit - Christophe
*   856433e - YYYY-MM-DD 15:07:00 (XX minutes ago)
|\            Merge branch 'features/foo' - Christophe
| * 694ac81 - YYYY-MM-DD 15:04:00 (XX minutes ago) (features/foo)
| |           Third commit - Christophe
| * 5fd94d3 - YYYY-MM-DD 15:03:00 (XX minutes ago)
|/            Second commit - Christophe
* d01d589 - YYYY-MM-DD 15:00:00 (XX minutes ago)
            First commit - Christophe

Примечания стороны

git cherry-pick

Когда вам нужен только один конкретный коммит, git cherry-pick - это решение Nice (опция -x добавляет строку с надписью " (вишня выбрана из коммита ...) " к исходному сообщению коммита body, поэтому, как правило, неплохо использовать его - git log <commit_sha1>, чтобы увидеть его):

Команды:

Time           Branch “develop"              Branch "features/foo"                Branch "features/bar"           
------- -------------------------------- ------------------------------- -----------------------------------------
15:10   git merge --no-ff features/bar                                                                            
15:09   git merge --no-ff features/foo                                                                            
15:08                                                                    git commit -m “Sixth commit"             
15:07                                                                    git cherry-pick -x <second_commit_sha1>  
15:06                                                                    git commit -m “Fifth commit"             
15:05                                                                    git commit -m “Fourth commit"            
15:04                                    git commit -m “Third commit"                                             
15:03                                    git commit -m “Second commit"                                            
15:02   git checkout -b features/bar                                                                              
15:01   git checkout -b features/foo                                                                              
15:00   git commit -m “First commit"                                                                              

Результат:

*   50839cd - YYYY-MM-DD 15:10:00 (XX minutes ago) (HEAD -> develop)
|\            Merge branch 'features/bar' - Christophe
| * 0cda99f - YYYY-MM-DD 15:08:00 (XX minutes ago) (features/bar)
| |           Sixth commit - Christophe
| * f7d6c47 - YYYY-MM-DD 15:03:00 (XX minutes ago)
| |           Second commit - Christophe
| * dd7d05a - YYYY-MM-DD 15:06:00 (XX minutes ago)
| |           Fifth commit - Christophe
| * d0d759b - YYYY-MM-DD 15:05:00 (XX minutes ago)
| |           Fourth commit - Christophe
* |   1a397c5 - YYYY-MM-DD 15:09:00 (XX minutes ago)
|\ \            Merge branch 'features/foo' - Christophe
| |/
|/|
| * 0600a72 - YYYY-MM-DD 15:04:00 (XX minutes ago) (features/foo)
| |           Third commit - Christophe
| * f4c127a - YYYY-MM-DD 15:03:00 (XX minutes ago)
|/            Second commit - Christophe
* 0cf894c - YYYY-MM-DD 15:00:00 (XX minutes ago)
            First commit - Christophe

git pull --rebase

Не уверен, что могу объяснить это лучше, чем Дерек Гурлэй ... По сути, используйте git pull --rebase вместо git pull :) Однако в статье не хватает того, что вы можете включить его по умолчанию знак равно

git config --global pull.rebase true

git rerere

Опять же, хорошо объяснил здесь . Но попросту говоря, если вы включите его, вам больше не придется разрешать один и тот же конфликт несколько раз.

19
sp00m

Книга про git в качестве действительно хорошего объяснения на страница перебазирования .

В основном слияние займет 2 коммита и объединит их.

Ребаз будет идти к общему предку на 2 и постепенно применять изменения друг на друге. Это делает "чище" и более линейной истории.

Но когда вы делаете ребазинг, вы отказываетесь от предыдущих коммитов и создаете новые. Таким образом, вы никогда не должны отказываться от публичного репо. Другие люди, работающие в репо, будут ненавидеть вас.

Только по этой причине я почти исключительно сливаюсь. В 99% случаев мои ветви не сильно отличаются друг от друга, поэтому, если возникают конфликты, это только в одном или двух местах.

15
xero

Git rebase используется для того, чтобы сделать пути ветвления в истории более чистыми и структуру хранилища линейной.

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

После выполнения ребазинга мы также избавляемся от дополнительного коммита, который мы использовали, чтобы увидеть, выполняем ли мы нормальное слияние.

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

4
cvibha

Некоторые практические примеры, в некоторой степени связанные с крупномасштабной разработкой, где gerrit используется для обзора и интеграции поставки.

Я сливаюсь, когда поднимаю свою ветвь функций на новый удаленный мастер. Это дает минимальную работу по подъему и легко отслеживать историю развития функции, например, в gitk.

git fetch
git checkout Origin/my_feature
git merge Origin/master
git commit
git Push Origin HEAD:refs/for/my_feature

Я объединяюсь, когда готовлю коммит доставки.

git fetch
git checkout Origin/master
git merge --squash Origin/my_feature
git commit
git Push Origin HEAD:refs/for/master

Я перезагружаюсь, когда моя фиксация доставки не проходит интеграцию по любой причине, и мне нужно обновить ее до нового удаленного мастера.

git fetch
git fetch <gerrit link>
git checkout FETCH_HEAD
git rebase Origin/master
git Push Origin HEAD:refs/for/master
3
Martin G

Много раз объяснялось, что такое ребаз и что такое слияние, но когда и что использовать?

Когда использовать rebase?

  • когда вы не нажали ветку/над ней никто больше не работает
  • ты хочешь полную историю
  • вы хотите избежать всех автоматически сгенерированных сообщений "слияния .."

Как git rebase меняет историю. Поэтому вы не должны использовать его, когда кто-то еще работает над той же веткой/если вы нажали ее. Но если у вас есть локальная ветвь, вы можете выполнить мастер слияния ребаз перед слиянием своей ветки с главной, чтобы сохранить более чистую историю. Делая это, после слияния с мастер-веткой не будет видно, что вы использовали ветку в мастер-ветке - история "чище", так как у вас нет автоматически сгенерированного "слияния ..", но все еще есть полный история в вашей главной ветке без автоматической генерации коммитов "merged ..". Тем не менее, убедитесь, что вы используете git merge feature-branch --ff-only, чтобы избежать конфликтов при создании одного коммита при слиянии вашей функции с основной. Это интересно, если вы используете ветви функций для каждой задачи, над которой работаете, так как вы получаете историю ветви функций, но не коммит "объединено .."

Второй сценарий будет, если вы разветвляетесь из ветки и хотите знать, что изменилось в основной ветке. rebase дает вам информацию, поскольку она включает в себя каждый коммит.

Когда использовать слияние?

  • когда вы нажали на ветку/другие тоже работают над этим
  • вам не нужна полная история
  • достаточно просто слияния

Когда вам не нужно или вы хотите иметь всю историю ветви функций в вашей основной ветке или если другие работают в той же ветке/вы добавили ее. Если вы все еще хотите иметь историю, просто объедините мастер с ветвью объектов, прежде чем объединять ветвь объектов с мастером. Это приведет к ускоренному слиянию, при котором у вас будет история ветви объектов в вашем мастере (включая коммит слияния, который был в вашей ветви функций, потому что вы слили мастер в него).

1
Jeremy Benks