SQLAlchemy ORM-관련 개체 삭제

단일 테이블에서 삭제 작업을 수행하는 것은 쉽습니다. 세션에서 매핑 된 클래스의 개체를 삭제하고 작업을 커밋하기 만하면됩니다. 그러나 여러 관련 테이블에 대한 삭제 작업은 약간 까다 롭습니다.

sales.db 데이터베이스에서 Customer 및 Invoice 클래스는 일대 다 유형의 관계로 고객 및 송장 테이블에 매핑됩니다. Customer 개체를 삭제하고 결과를 확인합니다.

빠른 참조로, 아래는 고객 및 송장 클래스의 정의입니다-

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")

세션을 설정하고 아래 프로그램을 사용하여 기본 ID로 쿼리하여 Customer 개체를 얻습니다.

from sqlalchemy.orm import sessionmaker
Session = sessionmaker(bind=engine)
session = Session()
x = session.query(Customer).get(2)

샘플 테이블에서 x.name은 'Gopal Krishna'입니다. 세션에서이 x를 삭제하고이 이름의 발생 횟수를 계산해 보겠습니다.

session.delete(x)
session.query(Customer).filter_by(name = 'Gopal Krishna').count()

결과 SQL 표현식은 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

그러나 x의 관련 송장 개체는 여전히 있습니다. 다음 코드로 확인할 수 있습니다-

session.query(Invoice).filter(Invoice.invno.in_([10,14])).count()

여기서 10과 14는 고객 Gopal Krishna의 송장 번호입니다. 위 쿼리의 결과는 2로 관련 개체가 삭제되지 않았 음을 의미합니다.

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

이는 SQLAlchemy가 캐스케이드 삭제를 가정하지 않기 때문입니다. 삭제 명령을 내려야합니다.

동작을 변경하기 위해 User.addresses 관계에 계단식 옵션을 구성합니다. 진행중인 세션을 닫고 new declarative_base ()를 사용하고 User 클래스를 다시 선언하여 캐스케이드 구성을 포함한 주소 관계를 추가하겠습니다.

관계 함수의 캐스케이드 속성은 세션 작업이 부모에서 자식으로 "캐스케이드"되어야하는 방법을 결정하는 쉼표로 구분 된 캐스케이드 규칙 목록입니다. 기본적으로 False이며 "save-update, merge"를 의미합니다.

사용 가능한 캐스케이드는 다음과 같습니다.

  • save-update
  • merge
  • expunge
  • delete
  • delete-orphan
  • refresh-expire

자주 사용되는 옵션은 "all, delete-orphan"으로 관련 개체가 모든 경우에 상위 개체와 함께 따라야하며 연결 ​​해제시 삭제되어야 함을 나타냅니다.

따라서 재 선언 된 고객 클래스는 다음과 같습니다.

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" 
   )

아래 프로그램을 사용하여 Gopal Krishna 이름으로 고객을 삭제하고 관련 송장 개체 수를 확인하겠습니다.

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()

카운트는 이제 위의 스크립트에 의해 생성 된 다음 SQL과 함께 0입니다.

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