SQLAlchemy ORM - Menghapus Objek Terkait
Mudah untuk melakukan operasi hapus pada satu tabel. Yang harus Anda lakukan adalah menghapus objek kelas yang dipetakan dari sesi dan melakukan tindakan. Namun, operasi hapus pada beberapa tabel terkait sedikit rumit.
Dalam database sales.db kami, kelas Pelanggan dan Faktur dipetakan ke tabel pelanggan dan faktur dengan satu jenis hubungan ke banyak. Kami akan mencoba untuk menghapus objek Pelanggan dan melihat hasilnya.
Sebagai referensi cepat, di bawah ini adalah definisi kelas Pelanggan dan Faktur -
from sqlalchemy import create_engine, ForeignKey, Column, Integer, String
engine = create_engine('sqlite:///sales.db', echo = True)
from sqlalchemy.ext.declarative import declarative_base
Base = declarative_base()
from sqlalchemy.orm import relationship
class Customer(Base):
__tablename__ = 'customers'
id = Column(Integer, primary_key = True)
name = Column(String)
address = Column(String)
email = Column(String)
class Invoice(Base):
__tablename__ = 'invoices'
id = Column(Integer, primary_key = True)
custid = Column(Integer, ForeignKey('customers.id'))
invno = Column(Integer)
amount = Column(Integer)
customer = relationship("Customer", back_populates = "invoices")
Customer.invoices = relationship("Invoice", order_by = Invoice.id, back_populates = "customer")
Kami menyiapkan sesi dan mendapatkan objek Pelanggan dengan menanyakannya dengan ID utama menggunakan program di bawah ini -
from sqlalchemy.orm import sessionmaker
Session = sessionmaker(bind=engine)
session = Session()
x = session.query(Customer).get(2)
Dalam tabel contoh kami, x.name kebetulan adalah 'Gopal Krishna'. Mari kita hapus x ini dari sesi dan hitung kemunculan nama ini.
session.delete(x)
session.query(Customer).filter_by(name = 'Gopal Krishna').count()
Ekspresi SQL yang dihasilkan akan mengembalikan 0.
SELECT count(*)
AS count_1
FROM (
SELECT customers.id
AS customers_id, customers.name
AS customers_name, customers.address
AS customers_address, customers.email
AS customers_email
FROM customers
WHERE customers.name = ?)
AS anon_1('Gopal Krishna',) 0
Namun, objek Faktur terkait x masih ada. Itu dapat diverifikasi dengan kode berikut -
session.query(Invoice).filter(Invoice.invno.in_([10,14])).count()
Di sini, 10 dan 14 adalah nomor faktur milik pelanggan Gopal Krishna. Hasil query di atas adalah 2 yang berarti objek terkait belum dihapus.
SELECT count(*)
AS count_1
FROM (
SELECT invoices.id
AS invoices_id, invoices.custid
AS invoices_custid, invoices.invno
AS invoices_invno, invoices.amount
AS invoices_amount
FROM invoices
WHERE invoices.invno IN (?, ?))
AS anon_1(10, 14) 2
Ini karena SQLAlchemy tidak mengasumsikan penghapusan cascade; kita harus memberi perintah untuk menghapusnya.
Untuk mengubah perilaku, kita mengkonfigurasi opsi bertingkat pada hubungan User.addresses. Mari kita tutup sesi yang sedang berlangsung, gunakan new declarative_base () dan deklarasikan ulang kelas User, tambahkan dalam hubungan alamat termasuk konfigurasi cascade.
Atribut kaskade dalam fungsi hubungan adalah daftar aturan kaskade yang dipisahkan koma yang menentukan bagaimana operasi Sesi harus "bertingkat" dari induk ke turunan. Secara default, ini adalah False, yang berarti "simpan-perbarui, gabungkan".
Kaskade yang tersedia adalah sebagai berikut -
- save-update
- merge
- expunge
- delete
- delete-orphan
- refresh-expire
Opsi yang sering digunakan adalah "semua, hapus-orphan" untuk menunjukkan bahwa objek terkait harus mengikuti objek induk dalam semua kasus, dan dihapus saat tidak terkait.
Karenanya kelas Pelanggan yang dideklarasikan ulang ditunjukkan di bawah ini -
class Customer(Base):
__tablename__ = 'customers'
id = Column(Integer, primary_key = True)
name = Column(String)
address = Column(String)
email = Column(String)
invoices = relationship(
"Invoice",
order_by = Invoice.id,
back_populates = "customer",
cascade = "all,
delete, delete-orphan"
)
Mari kita hapus Pelanggan dengan nama Gopal Krishna menggunakan program di bawah ini dan melihat hitungan objek Faktur terkait -
from sqlalchemy.orm import sessionmaker
Session = sessionmaker(bind = engine)
session = Session()
x = session.query(Customer).get(2)
session.delete(x)
session.query(Customer).filter_by(name = 'Gopal Krishna').count()
session.query(Invoice).filter(Invoice.invno.in_([10,14])).count()
Hitungannya sekarang 0 dengan mengikuti SQL yang dipancarkan oleh skrip di atas -
SELECT customers.id
AS customers_id, customers.name
AS customers_name, customers.address
AS customers_address, customers.email
AS customers_email
FROM customers
WHERE customers.id = ?
(2,)
SELECT invoices.id
AS invoices_id, invoices.custid
AS invoices_custid, invoices.invno
AS invoices_invno, invoices.amount
AS invoices_amount
FROM invoices
WHERE ? = invoices.custid
ORDER BY invoices.id (2,)
DELETE FROM invoices
WHERE invoices.id = ? ((1,), (2,))
DELETE FROM customers
WHERE customers.id = ? (2,)
SELECT count(*)
AS count_1
FROM (
SELECT customers.id
AS customers_id, customers.name
AS customers_name, customers.address
AS customers_address, customers.email
AS customers_email
FROM customers
WHERE customers.name = ?)
AS anon_1('Gopal Krishna',)
SELECT count(*)
AS count_1
FROM (
SELECT invoices.id
AS invoices_id, invoices.custid
AS invoices_custid, invoices.invno
AS invoices_invno, invoices.amount
AS invoices_amount
FROM invoices
WHERE invoices.invno IN (?, ?))
AS anon_1(10, 14)
0