SQLAlchemy ORM-조인 작업

이제 두 개의 테이블이 있으므로 두 테이블에 대해 동시에 쿼리를 만드는 방법을 살펴 보겠습니다. Customer와 Invoice 간의 단순한 암시 적 조인을 구성하기 위해 Query.filter ()를 사용하여 관련 열을 서로 동일시 할 수 있습니다. 아래에서는이 방법을 사용하여 고객 및 송장 엔터티를 한 번에로드합니다.

from sqlalchemy.orm import sessionmaker
Session = sessionmaker(bind = engine)
session = Session()

for c, i in session.query(Customer, Invoice).filter(Customer.id == Invoice.custid).all():
   print ("ID: {} Name: {} Invoice No: {} Amount: {}".format(c.id,c.name, i.invno, i.amount))

SQLAlchemy에서 내 보낸 SQL 표현식은 다음과 같습니다.

SELECT customers.id 
AS customers_id, customers.name 
AS customers_name, customers.address 
AS customers_address, customers.email 
AS customers_email, invoices.id 
AS invoices_id, invoices.custid 
AS invoices_custid, invoices.invno 
AS invoices_invno, invoices.amount 
AS invoices_amount
FROM customers, invoices
WHERE customers.id = invoices.custid

그리고 위의 코드 줄의 결과는 다음과 같습니다.

ID: 2 Name: Gopal Krishna Invoice No: 10 Amount: 15000
ID: 2 Name: Gopal Krishna Invoice No: 14 Amount: 3850
ID: 3 Name: Govind Pant Invoice No: 3 Amount: 10000
ID: 3 Name: Govind Pant Invoice No: 4 Amount: 5000
ID: 4 Name: Govind Kala Invoice No: 7 Amount: 12000
ID: 4 Name: Govind Kala Invoice No: 8 Amount: 8500
ID: 5 Name: Abdul Rahman Invoice No: 9 Amount: 15000
ID: 5 Name: Abdul Rahman Invoice No: 11 Amount: 6000

실제 SQL JOIN 구문은 다음과 같이 Query.join () 메서드를 사용하여 쉽게 얻을 수 있습니다.

session.query(Customer).join(Invoice).filter(Invoice.amount == 8500).all()

조인에 대한 SQL 표현식이 콘솔에 표시됩니다.

SELECT customers.id 
AS customers_id, customers.name 
AS customers_name, customers.address 
AS customers_address, customers.email 
AS customers_email
FROM customers JOIN invoices ON customers.id = invoices.custid
WHERE invoices.amount = ?

for 루프를 사용하여 결과를 반복 할 수 있습니다.

result = session.query(Customer).join(Invoice).filter(Invoice.amount == 8500)
for row in result:
   for inv in row.invoices:
      print (row.id, row.name, inv.invno, inv.amount)

bind 매개 변수로 8500을 사용하면 다음 출력이 표시됩니다.

4 Govind Kala 8 8500

Query.join ()은 이러한 테이블 사이에 외래 키가 하나만 있기 때문에 이러한 테이블을 조인하는 방법을 알고 있습니다. 외래 키가 없거나 더 많은 외래 키가 없으면 Query.join ()은 다음 형식 중 하나를 사용할 때 더 잘 작동합니다.

query.join (Invoice, id == Address.custid) 명시 적 조건
query.join (Customer.invoices) 왼쪽에서 오른쪽으로 관계 지정
query.join (Invoice, Customer.invoices) 동일, 명시 적 대상
query.join ( 'invoices') 동일, 문자열 사용

마찬가지로 outerjoin () 함수를 사용하여 왼쪽 외부 조인을 수행 할 수 있습니다.

query.outerjoin(Customer.invoices)

subquery () 메서드는 별칭에 포함 된 SELECT 문을 나타내는 SQL 식을 생성합니다.

from sqlalchemy.sql import func

stmt = session.query(
   Invoice.custid, func.count('*').label('invoice_count')
).group_by(Invoice.custid).subquery()

stmt 객체는 아래와 같이 SQL 문을 포함합니다-

SELECT invoices.custid, count(:count_1) AS invoice_count FROM invoices GROUP BY invoices.custid

진술이 있으면 Table 구조처럼 동작합니다. 명령문의 열은 아래 코드와 같이 c라는 속성을 통해 액세스 할 수 있습니다.

for u, count in session.query(Customer, stmt.c.invoice_count).outerjoin(stmt, Customer.id == stmt.c.custid).order_by(Customer.id):
   print(u.name, count)

위의 for 루프는 다음과 같이 이름 별 인보이스 개수를 표시합니다.

Arjun Pandit None
Gopal Krishna 2
Govind Pant 2
Govind Kala 2
Abdul Rahman 2