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