Tutorial Ruby / DBI
Bab ini mengajarkan Anda cara mengakses database menggunakan Ruby. The Ruby DBI modul menyediakan antarmuka database-independen untuk script Ruby mirip dengan modul Perl DBI.
DBI adalah singkatan dari Database Independent Interface for Ruby, yang berarti DBI menyediakan lapisan abstraksi antara kode Ruby dan database yang mendasarinya, memungkinkan Anda untuk mengganti implementasi database dengan sangat mudah. Ini mendefinisikan satu set metode, variabel, dan konvensi yang menyediakan antarmuka database yang konsisten, terlepas dari database aktual yang digunakan.
DBI dapat berinteraksi dengan berikut ini -
- ADO (Objek Data ActiveX)
- DB2
- Frontbase
- mSQL
- MySQL
- ODBC
- Oracle
- OCI8 (Oracle)
- PostgreSQL
- Proxy/Server
- SQLite
- SQLRelay
Arsitektur Aplikasi DBI
DBI tidak bergantung pada database apa pun yang tersedia di backend. Anda dapat menggunakan DBI apakah Anda bekerja dengan Oracle, MySQL atau Informix, dll. Ini jelas dari diagram arsitektur berikut.
Arsitektur umum untuk Ruby DBI menggunakan dua lapisan -
Lapisan antarmuka database (DBI). Lapisan ini adalah database independen dan menyediakan sekumpulan metode akses umum yang digunakan dengan cara yang sama terlepas dari jenis server database yang Anda gunakan untuk berkomunikasi.
Lapisan database driver (DBD). Lapisan ini bergantung pada database; driver yang berbeda memberikan akses ke mesin database yang berbeda. Ada satu driver untuk MySQL, satu lagi untuk PostgreSQL, satu lagi untuk InterBase, satu lagi untuk Oracle, dan seterusnya. Setiap driver menafsirkan permintaan dari lapisan DBI dan memetakannya ke permintaan yang sesuai untuk jenis server database tertentu.
Prasyarat
Jika Anda ingin menulis skrip Ruby untuk mengakses database MySQL, Anda harus menginstal modul Ruby MySQL.
Modul ini bertindak sebagai DBD seperti yang dijelaskan di atas dan dapat diunduh dari https://www.tmtm.org/en/mysql/ruby/
Memperoleh dan Menginstal Ruby / DBI
Anda dapat mengunduh dan menginstal modul Ruby DBI dari lokasi berikut -
https://imgur.com/NFEuWe4/embed
Sebelum memulai instalasi ini, pastikan Anda memiliki hak akses root. Sekarang, ikuti langkah-langkah yang diberikan di bawah ini -
Langkah 1
$ tar zxf dbi-0.2.0.tar.gz
Langkah 2
Masuk ke direktori distribusi dbi-0.2.0 dan konfigurasikan menggunakan skrip setup.rb di direktori itu. Perintah konfigurasi yang paling umum terlihat seperti ini, tanpa argumen yang mengikuti argumen config. Perintah ini mengkonfigurasi distribusi untuk menginstal semua driver secara default.
$ ruby setup.rb config
Untuk lebih spesifik, berikan opsi --dengan yang mencantumkan bagian tertentu dari distribusi yang ingin Anda gunakan. Misalnya, untuk hanya mengkonfigurasi modul DBI utama dan driver tingkat DBD MySQL, jalankan perintah berikut -
$ ruby setup.rb config --with = dbi,dbd_mysql
LANGKAH 3
Langkah terakhir adalah membangun driver dan menginstalnya menggunakan perintah berikut -
$ ruby setup.rb setup
$ ruby setup.rb install
Koneksi Database
Dengan asumsi kita akan bekerja dengan database MySQL, sebelum menghubungkan ke database pastikan hal berikut -
Anda telah membuat TESTDB database.
Anda telah membuat EMPLOYEE di TESTDB.
Tabel ini memiliki bidang FIRST_NAME, LAST_NAME, AGE, SEX, dan INCOME.
ID pengguna "testuser" dan kata sandi "test123" disetel untuk mengakses TESTDB.
Ruby Module DBI diinstal dengan benar di komputer Anda.
Anda telah melalui tutorial MySQL untuk memahami Dasar-Dasar MySQL.
Berikut adalah contoh koneksi dengan database MySQL "TESTDB"
#!/usr/bin/ruby -w
require "dbi"
begin
# connect to the MySQL server
dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123")
# get server version string and display it
row = dbh.select_one("SELECT VERSION()")
puts "Server version: " + row[0]
rescue DBI::DatabaseError => e
puts "An error occurred"
puts "Error code: #{e.err}"
puts "Error message: #{e.errstr}"
ensure
# disconnect from server
dbh.disconnect if dbh
end
Saat menjalankan skrip ini, ini menghasilkan hasil berikut di mesin Linux kami.
Server version: 5.0.45
Jika koneksi dibuat dengan sumber data, maka Pegangan Database dikembalikan dan disimpan ke dbh untuk penggunaan lebih lanjut sebaliknya dbhdiset ke nilai nil dan e.err dan e :: errstr mengembalikan kode kesalahan dan string kesalahan masing-masing.
Terakhir, sebelum keluar, pastikan koneksi database ditutup dan sumber daya dilepaskan.
Operasi INSERT
Operasi INSERT diperlukan saat Anda ingin membuat rekaman Anda ke dalam tabel database.
Setelah koneksi database dibuat, kita siap membuat tabel atau record ke dalam tabel database menggunakan do metode atau prepare dan execute metode.
Menggunakan do Statement
Pernyataan yang tidak mengembalikan baris dapat dikeluarkan dengan memanggil dodatabase menangani metode. Metode ini mengambil argumen string pernyataan dan mengembalikan hitungan jumlah baris yang dipengaruhi oleh pernyataan itu.
dbh.do("DROP TABLE IF EXISTS EMPLOYEE")
dbh.do("CREATE TABLE EMPLOYEE (
FIRST_NAME CHAR(20) NOT NULL,
LAST_NAME CHAR(20),
AGE INT,
SEX CHAR(1),
INCOME FLOAT )" );
Demikian pula, Anda dapat menjalankan pernyataan SQL INSERT untuk membuat record ke dalam tabel EMPLOYEE.
#!/usr/bin/ruby -w
require "dbi"
begin
# connect to the MySQL server
dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123")
dbh.do( "INSERT INTO EMPLOYEE(FIRST_NAME, LAST_NAME, AGE, SEX, INCOME)
VALUES ('Mac', 'Mohan', 20, 'M', 2000)" )
puts "Record has been created"
dbh.commit
rescue DBI::DatabaseError => e
puts "An error occurred"
puts "Error code: #{e.err}"
puts "Error message: #{e.errstr}"
dbh.rollback
ensure
# disconnect from server
dbh.disconnect if dbh
end
Menggunakan persiapkan dan eksekusi
Anda dapat menggunakan metode mempersiapkan dan menjalankan kelas DBI untuk menjalankan pernyataan SQL melalui kode Ruby.
Pembuatan rekaman mengambil langkah-langkah berikut -
Mempersiapkan pernyataan SQL dengan pernyataan INSERT. Ini akan dilakukan dengan menggunakanprepare metode.
Menjalankan kueri SQL untuk memilih semua hasil dari database. Ini akan dilakukan dengan menggunakanexecute metode.
Releasing Statement menangani. Ini akan dilakukan dengan menggunakanfinish API
Jika semuanya baik-baik saja, maka commit operasi ini jika tidak, Anda bisa rollback transaksi lengkap.
Berikut ini adalah sintaks untuk menggunakan kedua metode ini -
sth = dbh.prepare(statement)
sth.execute
... zero or more SQL operations ...
sth.finish
Kedua metode ini dapat digunakan untuk lulus bindnilai ke pernyataan SQL. Mungkin ada kasus ketika nilai yang akan dimasukkan tidak diberikan sebelumnya. Dalam kasus seperti itu, nilai mengikat digunakan. Tanda tanya (?) digunakan sebagai pengganti nilai aktual dan kemudian nilai aktual diteruskan melalui API eksekusi ().
Berikut adalah contoh untuk membuat dua record di tabel EMPLOYEE -
#!/usr/bin/ruby -w
require "dbi"
begin
# connect to the MySQL server
dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123")
sth = dbh.prepare( "INSERT INTO EMPLOYEE(FIRST_NAME, LAST_NAME, AGE, SEX, INCOME)
VALUES (?, ?, ?, ?, ?)" )
sth.execute('John', 'Poul', 25, 'M', 2300)
sth.execute('Zara', 'Ali', 17, 'F', 1000)
sth.finish
dbh.commit
puts "Record has been created"
rescue DBI::DatabaseError => e
puts "An error occurred"
puts "Error code: #{e.err}"
puts "Error message: #{e.errstr}"
dbh.rollback
ensure
# disconnect from server
dbh.disconnect if dbh
end
Jika ada beberapa INSERT dalam satu waktu, maka mempersiapkan pernyataan terlebih dahulu dan kemudian mengeksekusinya beberapa kali dalam satu loop lebih efisien daripada pemanggilan lakukan setiap kali melalui loop.
BACA Operasi
Operasi BACA pada database apa pun berarti mengambil beberapa informasi berguna dari database.
Setelah koneksi database kami dibuat, kami siap untuk membuat kueri ke dalam database ini. Kita bisa menggunakan keduanyado metode atau prepare dan execute metode untuk mengambil nilai dari tabel database.
Pengambilan rekaman mengambil langkah-langkah berikut -
Mempersiapkan kueri SQL berdasarkan kondisi yang diperlukan. Ini akan dilakukan dengan menggunakanprepare metode.
Menjalankan kueri SQL untuk memilih semua hasil dari database. Ini akan dilakukan dengan menggunakanexecute metode.
Mengambil semua hasil satu per satu dan mencetak hasil tersebut. Ini akan dilakukan dengan menggunakanfetch metode.
Releasing Statement menangani. Ini akan dilakukan dengan menggunakanfinish metode.
Berikut prosedur untuk query semua record dari tabel KARYAWAN yang memiliki gaji lebih dari 1000.
#!/usr/bin/ruby -w
require "dbi"
begin
# connect to the MySQL server
dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123")
sth = dbh.prepare("SELECT * FROM EMPLOYEE WHERE INCOME > ?")
sth.execute(1000)
sth.fetch do |row|
printf "First Name: %s, Last Name : %s\n", row[0], row[1]
printf "Age: %d, Sex : %s\n", row[2], row[3]
printf "Salary :%d \n\n", row[4]
end
sth.finish
rescue DBI::DatabaseError => e
puts "An error occurred"
puts "Error code: #{e.err}"
puts "Error message: #{e.errstr}"
ensure
# disconnect from server
dbh.disconnect if dbh
end
Ini akan menghasilkan hasil sebagai berikut -
First Name: Mac, Last Name : Mohan
Age: 20, Sex : M
Salary :2000
First Name: John, Last Name : Poul
Age: 25, Sex : M
Salary :2300
Ada lebih banyak metode pintas untuk mengambil catatan dari database. Jika Anda tertarik, lanjutkan ke bagian Pengambilan Hasil jika tidak lanjutkan ke bagian berikutnya.
Perbarui Operasi
UPDATE Operasi pada database apa pun berarti memperbarui satu atau lebih rekaman, yang sudah tersedia dalam database. Berikut adalah prosedur untuk memperbarui semua record yang memiliki SEX sebagai 'M'. Di sini, kami akan meningkatkan USIA semua pria satu tahun. Ini akan mengambil tiga langkah -
Mempersiapkan kueri SQL berdasarkan kondisi yang diperlukan. Ini akan dilakukan dengan menggunakanprepare metode.
Menjalankan kueri SQL untuk memilih semua hasil dari database. Ini akan dilakukan dengan menggunakanexecute metode.
Releasing Statement menangani. Ini akan dilakukan dengan menggunakanfinish metode.
Jika semuanya baik-baik saja commit operasi ini jika tidak, Anda bisa rollback transaksi lengkap.
#!/usr/bin/ruby -w
require "dbi"
begin
# connect to the MySQL server
dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123")
sth = dbh.prepare("UPDATE EMPLOYEE SET AGE = AGE + 1 WHERE SEX = ?")
sth.execute('M')
sth.finish
dbh.commit
rescue DBI::DatabaseError => e
puts "An error occurred"
puts "Error code: #{e.err}"
puts "Error message: #{e.errstr}"
dbh.rollback
ensure
# disconnect from server
dbh.disconnect if dbh
end
HAPUS Operasi
Operasi HAPUS diperlukan saat Anda ingin menghapus beberapa rekaman dari database Anda. Berikut adalah prosedur untuk menghapus semua catatan dari KARYAWAN di mana USIA lebih dari 20. Operasi ini akan mengambil langkah-langkah berikut.
Mempersiapkan kueri SQL berdasarkan kondisi yang diperlukan. Ini akan dilakukan dengan menggunakanprepare metode.
Menjalankan kueri SQL untuk menghapus rekaman yang diperlukan dari database. Ini akan dilakukan dengan menggunakanexecute metode.
Releasing Statement menangani. Ini akan dilakukan dengan menggunakanfinish metode.
Jika semuanya baik-baik saja commit operasi ini jika tidak, Anda bisa rollback transaksi lengkap.
#!/usr/bin/ruby -w
require "dbi"
begin
# connect to the MySQL server
dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123")
sth = dbh.prepare("DELETE FROM EMPLOYEE WHERE AGE > ?")
sth.execute(20)
sth.finish
dbh.commit
rescue DBI::DatabaseError => e
puts "An error occurred"
puts "Error code: #{e.err}"
puts "Error message: #{e.errstr}"
dbh.rollback
ensure
# disconnect from server
dbh.disconnect if dbh
end
Melakukan Transaksi
Transaksi adalah mekanisme yang memastikan konsistensi data. Transaksi harus memiliki empat properti berikut -
Atomicity - Transaksi selesai atau tidak terjadi sama sekali.
Consistency - Transaksi harus dimulai dalam keadaan konsisten dan membiarkan sistem dalam keadaan konsisten.
Isolation - Hasil antara transaksi tidak terlihat di luar transaksi saat ini.
Durability - Setelah transaksi dilakukan, efeknya tetap ada, bahkan setelah kegagalan sistem.
DBI menyediakan dua metode untuk melakukan atau membatalkan transaksi. Ada satu metode lagi yang disebut transaksi yang dapat digunakan untuk mengimplementasikan transaksi. Ada dua pendekatan sederhana untuk mengimplementasikan transaksi -
Pendekatan I
Pendekatan pertama menggunakan metode commit dan rollback DBI untuk secara eksplisit melakukan atau membatalkan transaksi -
dbh['AutoCommit'] = false # Set auto commit to false.
begin
dbh.do("UPDATE EMPLOYEE SET AGE = AGE+1 WHERE FIRST_NAME = 'John'")
dbh.do("UPDATE EMPLOYEE SET AGE = AGE+1 WHERE FIRST_NAME = 'Zara'")
dbh.commit
rescue
puts "transaction failed"
dbh.rollback
end
dbh['AutoCommit'] = true
Pendekatan II
Pendekatan kedua menggunakan metode transaksi . Ini lebih sederhana, karena dibutuhkan blok kode yang berisi pernyataan yang membentuk transaksi. The transaksi Metode mengeksekusi blok, kemudian memanggil komit atau rollback secara otomatis, tergantung pada apakah blok berhasil atau gagal -
dbh['AutoCommit'] = false # Set auto commit to false.
dbh.transaction do |dbh|
dbh.do("UPDATE EMPLOYEE SET AGE = AGE+1 WHERE FIRST_NAME = 'John'")
dbh.do("UPDATE EMPLOYEE SET AGE = AGE+1 WHERE FIRST_NAME = 'Zara'")
end
dbh['AutoCommit'] = true
Operasi COMMIT
Komit adalah operasi, yang memberikan sinyal hijau ke database untuk menyelesaikan perubahan, dan setelah operasi ini, tidak ada perubahan yang dapat dikembalikan.
Berikut adalah contoh sederhana untuk memanggil file commit metode.
dbh.commit
Operasi ROLLBACK
Jika Anda tidak puas dengan satu atau beberapa perubahan dan Anda ingin mengembalikan perubahan itu sepenuhnya, gunakan rollback metode.
Berikut adalah contoh sederhana untuk memanggil file rollback metode.
dbh.rollback
Memutus Database
Untuk memutuskan koneksi Database, gunakan API putuskan.
dbh.disconnect
Jika koneksi ke database ditutup oleh pengguna dengan metode putuskan, setiap transaksi yang beredar dibatalkan oleh DBI. Namun, daripada bergantung pada detail implementasi DBI mana pun, aplikasi Anda akan lebih baik memanggil commit atau rollback secara eksplisit.
Penanganan Error
Ada banyak sumber kesalahan. Beberapa contoh adalah kesalahan sintaks dalam pernyataan SQL yang dieksekusi, kegagalan koneksi, atau memanggil metode pengambilan untuk pegangan pernyataan yang sudah dibatalkan atau selesai.
Jika metode DBI gagal, DBI memunculkan pengecualian. Metode DBI dapat memunculkan salah satu dari beberapa tipe pengecualian tetapi dua kelas pengecualian yang paling penting adalah DBI :: InterfaceError dan DBI :: DatabaseError .
Objek pengecualian dari kelas-kelas ini memiliki tiga atribut bernama err , errstr , dan state , yang merepresentasikan nomor kesalahan, string kesalahan deskriptif, dan kode kesalahan standar. Atribut dijelaskan di bawah -
err- Mengembalikan representasi integer dari kesalahan yang terjadi atau nihil jika ini tidak didukung oleh DBD. Oracle DBD misalnya mengembalikan bagian numerik dari pesan kesalahan ORA-XXXX .
errstr - Mengembalikan representasi string dari kesalahan yang terjadi.
state- Mengembalikan kode SQLSTATE dari kesalahan yang terjadi. SQLSTATE adalah string sepanjang lima karakter. Kebanyakan DBD tidak mendukung ini dan mengembalikan nihil sebagai gantinya.
Anda telah melihat kode berikut di atas di sebagian besar contoh -
rescue DBI::DatabaseError => e
puts "An error occurred"
puts "Error code: #{e.err}"
puts "Error message: #{e.errstr}"
dbh.rollback
ensure
# disconnect from server
dbh.disconnect if dbh
end
Untuk mendapatkan informasi debug tentang apa yang dilakukan skrip Anda saat dijalankan, Anda dapat mengaktifkan pelacakan. Untuk melakukannya, Anda harus memuat modul dbi / trace terlebih dahulu, lalu memanggil metode pelacakan yang mengontrol mode pelacakan dan tujuan keluaran -
require "dbi/trace"
..............
trace(mode, destination)
Nilai mode mungkin 0 (mati), 1, 2, atau 3, dan tujuan harus berupa objek IO. Nilai default masing-masing adalah 2 dan STDERR.
Blok Kode dengan Metode
Ada beberapa metode yang membuat pegangan. Metode ini dapat dipanggil dengan blok kode. Keuntungan menggunakan blok kode bersama dengan metode adalah mereka menyediakan pegangan ke blok kode sebagai parameternya dan secara otomatis membersihkan pegangan ketika blok berakhir. Ada beberapa contoh untuk memahami konsep tersebut.
DBI.connect- Metode ini menghasilkan pegangan database dan direkomendasikan untuk memanggil putuskan sambungan di akhir blok untuk memutuskan database.
dbh.prepare- Metode ini menghasilkan sebuah pegangan pernyataan dan direkomendasikan untuk diselesaikan di akhir blok. Di dalam blok, Anda harus menjalankan metode eksekusi untuk mengeksekusi pernyataan tersebut.
dbh.execute- Metode ini serupa kecuali kita tidak perlu menjalankan eksekusi di dalam blok. Pegangan pernyataan dijalankan secara otomatis.
Contoh 1
DBI.connect dapat mengambil blok kode, meneruskan pegangan database ke sana, dan secara otomatis memutuskan pegangan di akhir blok sebagai berikut.
dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123") do |dbh|
Contoh 2
dbh.prepare dapat mengambil blok kode, meneruskan pegangan pernyataan padanya, dan secara otomatis panggilan selesai di akhir blok sebagai berikut.
dbh.prepare("SHOW DATABASES") do |sth|
sth.execute
puts "Databases: " + sth.fetch_all.join(", ")
end
Contoh 3
dbh.execute dapat mengambil blok kode, meneruskan pegangan pernyataan padanya, dan secara otomatis panggilan selesai di akhir blok sebagai berikut -
dbh.execute("SHOW DATABASES") do |sth|
puts "Databases: " + sth.fetch_all.join(", ")
end
Metode transaksi DBI juga mengambil blok kode yang telah dijelaskan di atas.
Fungsi dan Atribut khusus pengemudi
DBI memungkinkan driver database menyediakan fungsi khusus database tambahan, yang dapat dipanggil oleh pengguna melalui metode func dari objek Handle apa pun.
Atribut khusus driver didukung dan dapat disetel atau didapatkan menggunakan []= atau [] metode.
Sr.No. | Fungsi & Deskripsi |
---|---|
1 | dbh.func(:createdb, db_name) Membuat database baru. |
2 | dbh.func(:dropdb, db_name) Menjatuhkan database. |
3 | dbh.func(:reload) Melakukan operasi muat ulang. |
4 | dbh.func(:shutdown) Matikan server. |
5 | dbh.func(:insert_id) => Fixnum Mengembalikan nilai AUTO_INCREMENT terbaru untuk koneksi. |
6 | dbh.func(:client_info) => String Mengembalikan informasi klien MySQL dalam hal versi. |
7 | dbh.func(:client_version) => Fixnum Mengembalikan informasi klien dalam hal versi. Ini mirip dengan: client_info tetapi mengembalikan fixnum bukannya sting. |
8 | dbh.func(:host_info) => String Mengembalikan informasi host. |
9 | dbh.func(:proto_info) => Fixnum Mengembalikan protokol yang digunakan untuk komunikasi. |
10 | dbh.func(:server_info) => String Mengembalikan informasi server MySQL dalam hal versi. |
11 | dbh.func(:stat) => String Mengembalikan status database saat ini. |
12 | dbh.func(:thread_id) => Fixnum Mengembalikan ID utas saat ini. |
Contoh
#!/usr/bin/ruby
require "dbi"
begin
# connect to the MySQL server
dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123")
puts dbh.func(:client_info)
puts dbh.func(:client_version)
puts dbh.func(:host_info)
puts dbh.func(:proto_info)
puts dbh.func(:server_info)
puts dbh.func(:thread_id)
puts dbh.func(:stat)
rescue DBI::DatabaseError => e
puts "An error occurred"
puts "Error code: #{e.err}"
puts "Error message: #{e.errstr}"
ensure
dbh.disconnect if dbh
end
Ini akan menghasilkan hasil sebagai berikut -
5.0.45
50045
Localhost via UNIX socket
10
5.0.45
150621
Uptime: 384981 Threads: 1 Questions: 1101078 Slow queries: 4 \
Opens: 324 Flush tables: 1 Open tables: 64 \
Queries per second avg: 2.860