Apakah mungkin untuk membatalkan perubahan yang disebabkan oleh perintah berikut? Jika ya, bagaimana caranya?
git reset --hard HEAD~1
sumber
Apakah mungkin untuk membatalkan perubahan yang disebabkan oleh perintah berikut? Jika ya, bagaimana caranya?
git reset --hard HEAD~1
Pat Notz benar. Anda bisa mendapatkan komit kembali selama itu sudah dalam beberapa hari. git hanya mengumpulkan sampah setelah sekitar satu bulan atau lebih kecuali Anda secara eksplisit mengatakannya untuk menghapus gumpalan yang lebih baru.
$ git init
Initialized empty Git repository in .git/
$ echo "testing reset" > file1
$ git add file1
$ git commit -m 'added file1'
Created initial commit 1a75c1d: added file1
1 files changed, 1 insertions(+), 0 deletions(-)
create mode 100644 file1
$ echo "added new file" > file2
$ git add file2
$ git commit -m 'added file2'
Created commit f6e5064: added file2
1 files changed, 1 insertions(+), 0 deletions(-)
create mode 100644 file2
$ git reset --hard HEAD^
HEAD is now at 1a75c1d... added file1
$ cat file2
cat: file2: No such file or directory
$ git reflog
1a75c1d... HEAD@{0}: reset --hard HEAD^: updating HEAD
f6e5064... HEAD@{1}: commit: added file2
$ git reset --hard f6e5064
HEAD is now at f6e5064... added file2
$ cat file2
added new file
Anda dapat melihat dalam contoh bahwa file2 telah dihapus sebagai hasil dari hard reset, tetapi diletakkan kembali pada tempatnya ketika saya reset melalui reflog.
git log -g
bisa menjadi cara yang sedikit lebih baik untuk melihat reflog daripada git reflog
.
git reflog
daripada git log -g
hanya karena Anda mendapatkan semua informasi pada satu baris dengan sha1, info KEPALA dan melakukan pesan semua berbaris. Jauh lebih mudah dibaca.
Yang ingin Anda lakukan adalah menentukan sha1 dari commit yang ingin Anda pulihkan. Anda bisa mendapatkan sha1 dengan memeriksa reflog ( git reflog
) dan kemudian melakukannya
git reset --hard <sha1 of desired commit>
Tapi jangan menunggu terlalu lama ... setelah beberapa minggu git akhirnya akan melihat komit itu sebagai tidak direferensikan dan menghapus semua gumpalan.
Jawabannya tersembunyi dalam respons terperinci di atas, Anda cukup melakukannya:
$> git reset --hard HEAD@{1}
(Lihat output dari git reflog show )
Dimungkinkan untuk memulihkannya jika Git belum mengumpulkan sampah.
Dapatkan ikhtisar komitmen menjuntai dengan fsck
:
$ git fsck --lost-found
dangling commit b72e67a9bb3f1fc1b64528bcce031af4f0d6fcbf
Memulihkan komit yang menggantung dengan rebase:
$ git rebase b72e67a9bb3f1fc1b64528bcce031af4f0d6fcbf
Jika Anda benar-benar beruntung, seperti saya sebelumnya, Anda dapat kembali ke editor teks Anda dan menekan 'batalkan'.
Saya tahu itu bukan jawaban yang tepat, tetapi itu menyelamatkan saya dari pekerjaan setengah hari jadi semoga itu akan melakukan hal yang sama untuk orang lain!
sejauh yang saya tahu, --hard
akan membuang perubahan yang tidak dikomit. Karena ini tidak dilacak oleh git. tetapi Anda dapat membatalkan discarded commit
.
$ git reflog
akan mendaftar:
b0d059c HEAD@{0}: reset: moving to HEAD~1
4bac331 HEAD@{1}: commit: added level introduction....
....
dimana 4bac331
itu discarded commit
.
Sekarang cukup gerakkan kepala ke komit itu ::
$ git reset --hard 4bac331
$ git fsck --lost-found
Checking object directories: 100% (256/256), done.
Checking objects: 100% (3/3), done.
dangling blob 025cab9725ccc00fbd7202da543f556c146cb119
dangling blob 84e9af799c2f5f08fb50874e5be7fb5cb7aa7c1b
dangling blob 85f4d1a289e094012819d9732f017c7805ee85b4
dangling blob 8f654d1cd425da7389d12c17dd2d88d318496d98
dangling blob 9183b84bbd292dcc238ca546dab896e073432933
dangling blob 1448ee51d0ea16f259371b32a557b60f908d15ee
dangling blob 95372cef6148d980ab1d7539ee6fbb44f5e87e22
dangling blob 9b3bf9fb1ee82c6d6d5ec9149e38fe53d4151fbd
dangling blob 2b21002ca449a9e30dbb87e535fbd4e65bac18f7
dangling blob 2fff2f8e4ea6408ac84a8560477aa00583002e66
dangling blob 333e76340b59a944456b4befd0e007c2e23ab37b
dangling blob b87163c8def315d40721e592f15c2192a33816bb
dangling blob c22aafb90358f6bf22577d1ae077ad89d9eea0a7
dangling blob c6ef78dd64c886e9c9895e2fc4556e69e4fbb133
dangling blob 4a71f9ff8262701171d42559a283c751fea6a201
dangling blob 6b762d368f44ddd441e5b8eae6a7b611335b49a2
dangling blob 724d23914b48443b19eada79c3eb1813c3c67fed
dangling blob 749ffc9a412e7584245af5106e78167b9480a27b
dangling commit f6ce1a403399772d4146d306d5763f3f5715cb5a <- it's this one
$ git show f6ce1a403399772d4146d306d5763f3f5715cb5a
commit f6ce1a403399772d4146d306d5763f3f5715cb5a
Author: Stian Gudmundsen Høiland <stian@Stians-Mac-mini.local>
Date: Wed Aug 15 08:41:30 2012 +0200
*MY COMMIT MESSAGE IS DISPLAYED HERE*
diff --git a/Some.file b/Some.file
new file mode 100644
index 0000000..15baeba
--- /dev/null
+++ b/Some.file
*THE WHOLE COMMIT IS DISPLAYED HERE*
$ git rebase f6ce1a403399772d4146d306d5763f3f5715cb5a
First, rewinding head to replay your work on top of it...
Fast-forwarded master to f6ce1a403399772d4146d306d5763f3f5715cb5a.
Bergantung pada keadaan repositori Anda ketika Anda menjalankan perintah, efek dari git reset --hard
can berkisar dari sepele ke membatalkan, pada dasarnya tidak mungkin.
Di bawah ini saya telah membuat daftar berbagai skenario yang mungkin berbeda, dan bagaimana Anda dapat pulih dari itu.
Situasi ini biasanya terjadi ketika Anda menjalankan git reset
dengan argumen, seperti pada git reset --hard HEAD~
. Jangan khawatir, ini mudah untuk dipulihkan!
Jika Anda hanya berlari git reset
dan tidak melakukan hal lain sejak itu, Anda dapat kembali ke posisi semula dengan one-liner ini:
git reset --hard @{1}
Ini akan mereset cabang Anda saat ini dalam keadaan apa pun sebelum diubah terakhir kali (dalam kasus Anda, modifikasi terbaru pada cabang akan menjadi hard reset yang ingin Anda batalkan).
Namun, jika Anda telah melakukan modifikasi lain pada cabang Anda sejak reset, one-liner di atas tidak akan berfungsi. Alih-alih, Anda harus berlari untuk melihat daftar semua perubahan terbaru yang dilakukan pada cabang Anda (termasuk me-reset). Daftar itu akan terlihat seperti ini:git reflog
<branchname>
7c169bd master@{0}: reset: moving to HEAD~
3ae5027 master@{1}: commit: Changed file2
7c169bd master@{2}: commit: Some change
5eb37ca master@{3}: commit (initial): Initial commit
Temukan operasi dalam daftar ini yang ingin Anda "batalkan". Pada contoh di atas, itu akan menjadi baris pertama, yang bertuliskan "reset: pindah ke HEAD ~". Kemudian salin representasi komit sebelum (di bawah) operasi itu. Dalam kasus kami, itu akan menjadi master@{1}
(atau 3ae5027
, keduanya mewakili komit yang sama), dan jalankan git reset --hard <commit>
untuk mereset cabang Anda saat ini kembali ke komit itu.
git add
, tetapi tidak pernah berkomitmen. Sekarang perubahan saya hilang!Ini agak sulit untuk dipulihkan. git memang memiliki salinan dari file yang Anda tambahkan, tetapi karena salinan ini tidak pernah terikat pada komit tertentu, Anda tidak dapat mengembalikan perubahan sekaligus. Sebagai gantinya, Anda harus mencari file individual di database git dan mengembalikannya secara manual. Anda dapat melakukan ini menggunakan git fsck
.
Untuk detailnya, lihat Undo git reset --hard dengan file yang tidak dikomit di area stage .
git add
, dan tidak pernah komit. Sekarang perubahan saya hilang!Uh oh. Aku benci mengatakan ini padamu, tapi kamu mungkin kurang beruntung. git tidak menyimpan perubahan yang tidak Anda tambahkan atau komit padanya, dan sesuai dengan dokumentasi untukgit reset
:
--keras
Atur ulang indeks dan pohon kerja. Setiap perubahan pada file yang dilacak di pohon kerja sejak
<commit>
dibuang.
Ada kemungkinan bahwa Anda mungkin dapat memulihkan perubahan Anda dengan semacam utilitas pemulihan disk atau layanan pemulihan data profesional, tapi pada saat ini itu mungkin masalah lebih dari itu layak.
Jika Anda belum mengumpulkan sampah repositori Anda (misalnya menggunakan git repack -d
atau git gc
, tetapi perhatikan bahwa pengumpulan sampah juga dapat terjadi secara otomatis), maka komit Anda masih ada - itu tidak lagi dapat dijangkau melalui HEAD.
Anda dapat mencoba menemukan komit Anda dengan melihat melalui output dari git fsck --lost-found
.
Versi Git yang lebih baru memiliki sesuatu yang disebut "reflog", yang merupakan log dari semua perubahan yang dilakukan pada referensi (tidak seperti perubahan yang dilakukan pada konten repositori). Jadi, misalnya, setiap kali Anda mengganti KEPALA Anda (yaitu setiap kali Anda melakukan git checkout
untuk beralih cabang) yang akan dicatat. Dan, tentu saja, Anda git reset
juga memanipulasi KEPALA, jadi itu juga dicatat. Anda dapat mengakses status lama dari referensi Anda dengan cara yang sama seperti Anda dapat mengakses status lama dari repositori Anda, dengan menggunakan @
tanda alih-alih ~
, seperti git reset HEAD@{1}
.
Butuh beberapa saat untuk memahami apa perbedaan antara HEAD @ {1} dan HEAD ~ 1, jadi inilah sedikit penjelasan:
git init
git commit --allow-empty -mOne
git commit --allow-empty -mTwo
git checkout -b anotherbranch
git commit --allow-empty -mThree
git checkout master # This changes the HEAD, but not the repository contents
git show HEAD~1 # => One
git show HEAD@{1} # => Three
git reflog
Jadi, HEAD~1
berarti "pergi ke komit sebelum komit yang ditunjuk HEAD saat ini", sementara HEAD@{1}
berarti "pergi ke komit yang ditunjuk HEAD sebelum menunjuk di tempat yang saat ini menunjuk pada".
Itu akan dengan mudah memungkinkan Anda menemukan komit Anda yang hilang dan memulihkannya.
Sebelum menjawab, mari tambahkan beberapa latar belakang, jelaskan apa ini HEAD
.
First of all what is HEAD?
HEAD
hanyalah referensi ke komit saat ini (terbaru) di cabang saat ini.
Hanya ada satu HEAD
pada waktu tertentu. (tidak termasuk git worktree
)
Konten HEAD
disimpan di dalam .git/HEAD
dan berisi 40 byte SHA-1 dari komit saat ini.
detached HEAD
Jika Anda tidak menggunakan komit terbaru - artinya HEAD
menunjuk komit sebelumnya dalam sejarah yang disebut detached HEAD
.
Pada baris perintah akan terlihat seperti ini- SHA-1 bukan nama cabang karena HEAD
tidak menunjuk ke ujung cabang saat ini
git checkout
git checkout <commit_id>
git checkout -b <new branch> <commit_id>
git checkout HEAD~X // x is the number of commits t go back
Ini akan mengecek cabang baru yang menunjuk ke komit yang diinginkan.
Perintah ini akan checkout ke komit yang diberikan.
Pada titik ini Anda dapat membuat cabang dan mulai bekerja sejak saat ini.
# Checkout a given commit.
# Doing so will result in a `detached HEAD` which mean that the `HEAD`
# is not pointing to the latest so you will need to checkout branch
# in order to be able to update the code.
git checkout <commit-id>
# create a new branch forked to the given commit
git checkout -b <branch name>
git reflog
Anda selalu dapat menggunakan itu reflog
juga.
git reflog
akan menampilkan perubahan apa pun yang memperbarui HEAD
dan memeriksa entri reflog yang diinginkan akan mengatur HEAD
kembali komitmen ini.
Setiap kali KEPALA diubah akan ada entri baru di reflog
git reflog
git checkout HEAD@{...}
Ini akan membuat Anda kembali ke komit yang Anda inginkan
git reset HEAD --hard <commit_id>
"Gerakkan" kepala Anda kembali ke komit yang diinginkan.
# This will destroy any local modifications.
# Don't do it if you have uncommitted work you want to keep.
git reset --hard 0d1d7fc32
# Alternatively, if there's work to keep:
git stash
git reset --hard 0d1d7fc32
git stash pop
# This saves the modifications, then reapplies that patch after resetting.
# You could get merge conflicts, if you've modified things which were
# changed since the commit you reset to.
git rebase --no-autostash
.git revert <sha-1>
"Undo" rentang komit atau komit yang diberikan.
Perintah reset akan "membatalkan" setiap perubahan yang dilakukan dalam komit yang diberikan.
Komit baru dengan undo patch akan dilakukan sementara komit asli akan tetap ada dalam sejarah juga.
# add new commit with the undo of the original one.
# the <sha-1> can be any commit(s) or commit range
git revert <sha-1>
Skema ini menggambarkan perintah mana melakukan apa.
Seperti yang Anda lihat ada reset && checkout
modifikasi HEAD
.
git reset HEAD --hard <commit_id>
contoh Anda diambil dari stackoverflow.com/questions/4114095/... - Jika itu masalahnya, bisakah Anda mengedit dalam atribusi?
git reflog
git cherry-pick <the sha>
Saya tahu ini adalah utas lama ... tetapi karena banyak orang mencari cara untuk membatalkan hal-hal di Git, saya masih berpikir itu mungkin ide yang baik untuk terus memberikan kiat di sini.
Ketika Anda melakukan "git add" atau memindahkan apa pun dari kiri atas ke kiri bawah di git gui, isi file disimpan dalam gumpalan dan konten file dimungkinkan untuk pulih dari gumpalan itu.
Jadi dimungkinkan untuk memulihkan file bahkan jika itu tidak dilakukan tetapi harus ditambahkan.
git init
echo hello >> test.txt
git add test.txt
Sekarang gumpalan dibuat tetapi direferensikan oleh indeks sehingga tidak akan terdaftar dengan git fsck sampai kita mengatur ulang. Jadi kami mengatur ulang ...
git reset --hard
git fsck
Anda akan mendapatkan gumpalan menggantung ce013625030ba8dba906f756967f9e9ca394464a
git show ce01362
akan memberi Anda konten file "halo" kembali
Untuk menemukan komitmen yang tidak direferensikan, saya menemukan tip di suatu tempat menyarankan ini.
gitk --all $(git log -g --pretty=format:%h)
Saya memilikinya sebagai alat di git gui dan itu sangat berguna.
git fsck --lost-found
dapat membantu.
Jika Anda menggunakan JetBrains IDE (apa pun yang berbasis IntelliJ), Anda bahkan dapat memulihkan perubahan yang tidak dikomit melalui fitur "Sejarah Lokal" mereka.
Klik kanan pada direktori tingkat atas Anda di pohon file Anda, cari "Sejarah Lokal" di menu konteks, dan pilih "Tampilkan Riwayat". Ini akan membuka tampilan tempat pengeditan terakhir Anda dapat ditemukan, dan setelah Anda menemukan revisi yang ingin Anda kembalilah, klik kanan padanya dan klik "Kembalikan".
Saya baru saja melakukan reset keras pada proyek yang salah. Yang menyelamatkan hidup saya adalah sejarah lokal Eclipse. IntelliJ Idea dikatakan memiliki satu juga, dan semoga editor Anda, ada baiknya memeriksa:
Buat skrip kecil agar lebih mudah menemukan komit yang dicari:
git fsck --lost-found | grep commit | cut -d ' ' -f 3 | xargs -i git show \{\} | egrep '^commit |Date:'
Ya, itu bisa dibuat jauh lebih cantik dengan awk atau sesuatu seperti itu, tetapi sederhana dan saya hanya membutuhkannya. Mungkin menghemat orang lain 30 detik.
Masalah saya hampir mirip. Saya memiliki file yang tidak berkomitmen sebelum saya masuk git reset --hard
.
Untungnya Saya berhasil melewati semua sumber daya ini. Setelah saya perhatikan bahwa saya bisa membatalkan ( ctrl-z
). 😊 Saya hanya ingin menambahkan ini ke semua jawaban di atas.
Catatan. Tidak mungkin untuk ctrl-z
membuka file.
Ini telah menyelamatkan hidup saya:
https://medium.com/@CarrieGuss/how-to-recover-from-a-git-hard-reset-b830b5e3f60c
Pada dasarnya Anda perlu menjalankan:
for blob in $(git fsck --lost-found | awk ‘$2 == “blob” { print $3 }’); do git cat-file -p $blob > $blob.txt; done
Kemudian secara manual melalui rasa sakit untuk mengatur ulang file Anda ke struktur yang benar.
Takeaway: Jangan pernah gunakan git reset --hard
jika Anda tidak sepenuhnya 100% mengerti cara kerjanya, lebih baik tidak menggunakannya.
--hard
buang perubahan yang tidak dikomit. Karena ini tidak dilacak oleh git, tidak ada cara untuk mengembalikannya melalui git.