Itu halus.
Jika persyaratan bisnis adalah "Saya ingin mengaudit perubahan data - siapa yang melakukan apa dan kapan?", Anda biasanya dapat menggunakan tabel audit (sesuai contoh pemicu yang diposting Keethanjan). Saya bukan penggemar berat pemicu, tetapi memiliki manfaat besar karena relatif tidak menyakitkan untuk diterapkan - kode Anda yang ada tidak perlu tahu tentang pemicu dan audit.
Jika persyaratan bisnisnya adalah "tunjukkan keadaan data pada tanggal tertentu di masa lalu", itu berarti aspek perubahan dari waktu ke waktu telah memasuki solusi Anda. Meskipun Anda dapat, hampir, merekonstruksi status database hanya dengan melihat tabel audit, itu sulit dan rawan kesalahan, dan untuk logika database yang rumit, itu menjadi sulit. Misalnya, jika bisnis ingin mengetahui "menemukan alamat surat yang seharusnya kami kirimkan kepada pelanggan yang memiliki faktur terutang dan belum dibayar pada hari pertama bulan itu", Anda mungkin harus menjaring setengah lusin tabel audit.
Sebaliknya, Anda dapat memasukkan konsep perubahan dari waktu ke waktu ke dalam desain skema Anda (ini adalah opsi kedua yang disarankan Keethanjan). Ini adalah perubahan pada aplikasi Anda, pasti pada logika bisnis dan tingkat ketekunan, jadi tidak sepele.
Misalnya, jika Anda memiliki tabel seperti ini:
CUSTOMER
---------
CUSTOMER_ID PK
CUSTOMER_NAME
CUSTOMER_ADDRESS
dan Anda ingin terus memantaunya, Anda harus mengubahnya sebagai berikut:
CUSTOMER
------------
CUSTOMER_ID PK
CUSTOMER_VALID_FROM PK
CUSTOMER_VALID_UNTIL PK
CUSTOMER_STATUS
CUSTOMER_USER
CUSTOMER_NAME
CUSTOMER_ADDRESS
Setiap kali Anda ingin mengubah catatan pelanggan, alih-alih memperbarui catatan, Anda menetapkan VALID_UNTIL pada catatan saat ini ke SEKARANG (), dan memasukkan catatan baru dengan VALID_FROM (sekarang) dan null VALID_UNTIL. Anda menyetel status "CUSTOMER_USER" ke ID login pengguna saat ini (jika Anda perlu menyimpannya). Jika pelanggan perlu dihapus, Anda menggunakan bendera CUSTOMER_STATUS untuk menunjukkan ini - Anda tidak boleh menghapus catatan dari tabel ini.
Dengan begitu, Anda selalu dapat menemukan status tabel pelanggan untuk tanggal tertentu - apa alamatnya? Apakah mereka sudah mengganti nama? Dengan bergabung ke tabel lain dengan tanggal valid_from dan valid_until yang serupa, Anda dapat merekonstruksi keseluruhan gambar secara historis. Untuk menemukan status saat ini, Anda mencari record dengan tanggal null VALID_UNTIL.
Ini sulit (sebenarnya, Anda tidak memerlukan valid_from, tetapi ini membuat kueri sedikit lebih mudah). Ini memperumit desain dan akses database Anda. Tapi itu membuat rekonstruksi dunia jauh lebih mudah.
Berikut cara langsung untuk melakukan ini:
Pertama, buat tabel riwayat untuk setiap tabel data yang ingin Anda lacak (contoh kueri di bawah). Tabel ini akan memiliki entri untuk setiap kueri penyisipan, pembaruan, dan penghapusan yang dilakukan pada setiap baris dalam tabel data.
Struktur tabel sejarah akan sama dengan tabel data yang dilacak kecuali tiga kolom tambahan: kolom untuk menyimpan operasi yang terjadi (sebut saja 'tindakan'), tanggal dan waktu operasi, dan kolom untuk menyimpan nomor urut ('revisi'), yang bertambah per operasi dan dikelompokkan berdasarkan kolom kunci utama dari tabel data.
Untuk melakukan perilaku pengurutan ini, indeks dua kolom (komposit) dibuat pada kolom kunci utama dan kolom revisi. Perhatikan bahwa Anda hanya dapat melakukan pengurutan dengan cara ini jika mesin yang digunakan oleh tabel riwayat adalah MyISAM ( Lihat 'Catatan MyISAM' di halaman ini)
Tabel riwayat cukup mudah dibuat. Di kueri ALTER TABLE di bawah (dan di kueri pemicu di bawahnya), ganti 'primary_key_column' dengan nama sebenarnya dari kolom itu di tabel data Anda.
Dan kemudian Anda membuat pemicu:
Dan Anda sudah selesai. Sekarang, semua penyisipan, pembaruan, dan penghapusan di 'MyDb.data' akan disimpan di 'MyDb.data_history', memberi Anda tabel riwayat seperti ini (minus kolom 'data_columns' yang dibuat-buat)
Untuk menampilkan perubahan pada kolom atau kolom tertentu dari pembaruan ke pembaruan, Anda harus menggabungkan tabel riwayat ke tabel itu sendiri pada kunci utama dan kolom urutan. Anda dapat membuat tampilan untuk tujuan ini, misalnya:
Sunting: Oh wow, orang-orang menyukai tabel sejarah saya dari 6 tahun yang lalu: P
Implementasi saya tentang hal itu masih terus berlanjut, menjadi lebih besar dan lebih berat, saya kira. Saya menulis tampilan dan UI yang cukup bagus untuk melihat sejarah dalam database ini, tapi saya rasa itu tidak pernah banyak digunakan. Begitu seterusnya.
Untuk menjawab beberapa komentar tanpa urutan tertentu:
Saya melakukan implementasi saya sendiri di PHP yang sedikit lebih terlibat, dan menghindari beberapa masalah yang dijelaskan dalam komentar (memiliki indeks ditransfer, secara signifikan. Jika Anda mentransfer indeks unik ke tabel riwayat, semuanya akan rusak. Ada solusi untuk ini di komentar). Mengikuti posting ini ke surat itu bisa menjadi petualangan, tergantung pada seberapa mapan database Anda.
Jika hubungan antara kunci utama dan kolom revisi tampak tidak aktif, biasanya kunci komposit mengalami gangguan. Pada beberapa kesempatan langka, saya mengalami hal ini dan kehilangan tujuan.
Saya menemukan solusi ini cukup berkinerja, menggunakan pemicu seperti itu. Juga, MyISAM cepat dalam menyisipkan, yang dilakukan oleh semua pemicu. Anda dapat meningkatkan ini lebih jauh dengan pengindeksan cerdas (atau kurangnya ...). Memasukkan satu baris ke dalam tabel MyISAM dengan kunci utama seharusnya tidak menjadi operasi yang perlu Anda optimalkan, sungguh, kecuali Anda memiliki masalah signifikan yang terjadi di tempat lain. Selama saya menjalankan database MySQL, implementasi tabel riwayat ini, tidak pernah menjadi penyebab dari (banyak) masalah performa yang muncul.
jika Anda mendapatkan penyisipan berulang, periksa lapisan perangkat lunak Anda untuk INSERT IGNORE jenis kueri. Hrmm, tidak dapat mengingatnya sekarang, tetapi saya pikir ada masalah dengan skema dan transaksi ini yang akhirnya gagal setelah menjalankan beberapa tindakan DML. Sesuatu yang harus diperhatikan, setidaknya.
Bidang di tabel riwayat dan tabel data harus cocok. Atau, tabel data Anda tidak memiliki lebih banyak kolom daripada tabel riwayat. Jika tidak, kueri sisipkan / perbarui / del pada tabel data akan gagal, saat penyisipan ke tabel riwayat meletakkan kolom dalam kueri yang tidak ada (karena d. * Dalam kueri pemicu), dan pemicu gagal. Ini akan menjadi luar biasa jika MySQL memiliki sesuatu seperti pemicu skema, di mana Anda dapat mengubah tabel riwayat jika kolom ditambahkan ke tabel data. Apakah MySQL memilikinya sekarang? Saya melakukan React hari ini: P
sumber
CREATE TABLE MyDB.data_history as select * from MyDB.data limit 0;
owner
bidang, dan untuk pembaruan saya dapat menambahkanupdatedby
bidang, tetapi untuk menghapus saya tidak yakin bagaimana saya bisa melakukannya melalui pemicu. memperbaruidata_history
baris dengan id pengguna di terasa kotor: PAnda bisa membuat pemicu untuk menyelesaikan ini. Berikut adalah tutorial untuk melakukannya (tautan yang diarsipkan).
Solusi lain adalah menyimpan bidang Revisi dan memperbarui bidang ini saat disimpan. Anda dapat memutuskan bahwa maks adalah revisi terbaru, atau 0 adalah baris terbaru. Terserah kamu.
sumber
Inilah cara kami menyelesaikannya
tabel Pengguna terlihat seperti ini
Dan persyaratan bisnis berubah dan kami harus memeriksa semua alamat dan nomor telepon sebelumnya yang pernah dimiliki pengguna. skema baru terlihat seperti ini
Untuk menemukan alamat saat ini dari setiap pengguna, kami mencari UserData dengan revisi DESC dan LIMIT 1
Untuk mendapatkan alamat pengguna antara periode waktu tertentu kita dapat menggunakan create_on bewteen (tanggal1, tanggal 2)
sumber
revision=1
paraid_user=1
? Pertama saya mengira bahwa penghitungan Anda adalah0,2,3,...
tetapi kemudian saya melihat bahwa untukid_user=2
penghitungan revisi adalah0,1, ...
id
danid_user
kolom. Just use a group ID of
id` (ID pengguna) danrevision
.MariaDB mendukung Pembuatan Versi Sistem sejak 10.3 yang merupakan fitur SQL standar yang melakukan apa yang Anda inginkan: ia menyimpan riwayat catatan tabel dan menyediakan akses ke sana melalui
SELECT
kueri. MariaDB adalah fork pengembangan terbuka dari MySQL. Anda dapat menemukan lebih lanjut tentang Pembuatan Versi Sistem melalui tautan ini:https://mariadb.com/kb/en/library/system-versioned-tables/
sumber
Mengapa tidak menggunakan file log bin? Jika replikasi diatur di server Mysql, dan format file binlog diatur ke ROW, maka semua perubahan dapat ditangkap.
Pustaka python yang bagus bernama noplay dapat digunakan. Info selengkapnya di sini .
sumber
Hanya 2 sen saya. Saya akan membuat solusi yang mencatat dengan tepat apa yang berubah, sangat mirip dengan solusi transien.
My ChangesTable sederhana menjadi:
DateTime | WhoChanged | TableName | Action | ID |FieldName | OldValue
1) Ketika seluruh baris diubah di tabel utama, banyak entri akan masuk ke tabel ini, TAPI itu sangat tidak mungkin, jadi bukan masalah besar (orang biasanya hanya mengubah satu hal) 2) OldVaue (dan NewValue jika Anda ingin) harus menjadi semacam "jenis apa saja" epik karena bisa berupa data apa pun, mungkin ada cara untuk melakukan ini dengan jenis RAW atau hanya menggunakan string JSON untuk mengonversi masuk dan keluar.
Penggunaan data minimum, menyimpan semua yang Anda butuhkan dan dapat digunakan untuk semua tabel sekaligus. Saya sedang meneliti ini sendiri sekarang, tetapi ini mungkin akhirnya menjadi cara saya pergi.
Untuk Buat dan Hapus, cukup ID baris, tidak perlu bidang. Jika menghapus sebuah bendera di meja utama (aktif?) Akan bagus.
sumber
Cara langsung untuk melakukan ini adalah dengan membuat pemicu di atas tabel. Tetapkan beberapa kondisi atau metode pemetaan. Ketika pembaruan atau penghapusan terjadi, itu akan dimasukkan ke dalam tabel 'ubah' secara otomatis.
Tetapi bagian terbesar adalah bagaimana jika kita mendapat banyak kolom dan banyak tabel. Kami harus mengetikkan setiap nama kolom dari setiap tabel. Jelas sekali, Ini hanya membuang-buang waktu.
Untuk menangani ini dengan lebih baik, kita dapat membuat beberapa prosedur atau fungsi untuk mengambil nama kolom.
Kami juga dapat menggunakan alat bagian ke-3 hanya untuk melakukan ini. Di sini, saya menulis program java Mysql Tracker
sumber
create table like table
saya pikir mereplikasi semua kolom dengan mudah