过滤SQLalchemy中的左连接

Ada*_*nst 18 python sql join sqlalchemy

使用SQLalchemy我想执行左外连接并过滤掉在连接表中具有匹配项的行.

我正在发送推送通知,所以我有一张Notification桌子.这意味着我还有一个ExpiredDeviceId表来存储不再有效的device_ids.(我不想只删除受影响的通知,因为用户以后可能会重新安装应用程序,此时通知应根据Apple的文档恢复.)

CREATE TABLE Notification (device_id TEXT, time DATETIME);
CREATE TABLE ExpiredDeviceId (device_id TEXT PRIMARY KEY, expiration_time DATETIME);
Run Code Online (Sandbox Code Playgroud)

注意:每个device_id可能有多个通知.每个设备都没有"设备"表.

所以在做的时候SELECT FROM Notification我应该相应地过滤.我可以在SQL中做到这一点:

SELECT * FROM Notification 
    LEFT OUTER JOIN ExpiredDeviceId 
    ON Notification.device_id = ExpiredDeviceId.device_id
WHERE expiration_time IS NULL
Run Code Online (Sandbox Code Playgroud)

但是我怎么能在SQLalchemy中做到这一点?

sess.query(
    Notification, 
    ExpiredDeviceId
).outerjoin(
    (ExpiredDeviceId, Notification.device_id == ExpiredDeviceId.device_id)
).filter(
    ???
)
Run Code Online (Sandbox Code Playgroud)

或者我可以用一个device_id NOT IN (SELECT device_id FROM ExpiredDeviceId)子句做到这一点,但这似乎效率低下.

Ale*_*ean 14

你需要在元组中拉回ExpiredDeviceId吗?如果你不这样做(即你只关心live device_ids),那么你不能这样做:

sess.query(
    Notification
).outerjoin(
    (ExpiredDeviceId, Notification.device_id == ExpiredDeviceId.device_id)
).filter(
    ExpiredDeviceId.expiration_time == None
)
Run Code Online (Sandbox Code Playgroud)

  • `ExpiredDeviceId.expiration_time is None`将不起作用 - 它将在传递给.filter()之前始终计算为False,从而导致意外行为(如果您希望读取SQL,则会出现意外情况).使用`==`. (3认同)