Выберите значения NULL в SQLAlchemy
Здесь моя таблица (PostgreSQL) -
test=> create table people (name varchar primary key,
marriage_status varchar) ;
test=> insert into people values ('Ken', 'married');
test=> insert into people values ('May', 'single');
test=> insert into people values ('Joe', NULL);
Я хочу выбрать всех людей, которые не известны как состоящие в браке, т.е. в том числе с NULL marriage_status.
Это работает не -
test=> select * from people where marriage_status != 'married' ;
name | marriage_status
------+-----------------
May | single
(1 row)
Конечно, это делает -
test=> select * from people where marriage_status != 'married'
or marriage_status is NULL ;
name | marriage_status
------+-----------------
May | single
Joe |
Проблема в том, что я обращаюсь к нему из SQLAlchemy с помощью
...filter(or_(people.marriage_status!='married',
people.marriage_status is None))
который переводится на
SELECT people.name as name,
people.marriage_status as marriage_status
FROM people
WHERE people.marriage_status != %(status_1)s OR False
sqlalchemy.engine.base.Engine.... {'status_1': 'married'}
И работает не -
test=> select * from people where marriage_status != 'married'
or False;
name | marriage_status
------+-----------------
May | single
(1 row)
не делает -
test=> select * from people where marriage_status != 'married'
or NULL;
name | marriage_status
------+-----------------
May | single
(1 row)
Как выбрать значения NULL через SQLAlchemy?
Ответы
Ответ 1
(как указано @augurar): Поскольку sqlalchemy использует магические методы (перегрузка операторов) для создания конструкций SQL
, он может обрабатывать только оператор, например !=
или ==
, но не может работать с is
(это очень эффективная конструкция Python).
Поэтому, чтобы заставить его работать с sqlalchemy, а не:
...filter(or_(people.marriage_status!='married', people.marriage_status is None))
вы должны использовать:
...filter(or_(people.marriage_status!='married', people.marriage_status == None))
в основном замените is None
на == None
. В этом случае ваш запрос будет правильно переведен на следующий SQL:
SELECT people.name AS people_name, people.marriage_status AS people_marriage_status
FROM people
WHERE people.marriage_status IS NULL OR people.marriage_status != ?
См. IS NULL
в документации.
Ответ 2
С SQLAlchemy 0.7.9 вы можете использовать метод is_
для столбца.
Выражение фильтра типа:
filter(or_(people.marriage_status!='married', people.marriage_status.is_(None)))
создаст параметризованный SQL:
WHERE people.marriage_status != %(status_1)s OR people.marriage_status IS NULL
Ответ 3
i столкнулся с аналогичной проблемой
https://groups.google.com/forum/?fromgroups#!topic/sqlalchemy/EVpxsNp5Ifg%5B1-25%5D
короткий ответ:
- теперь нет оператора столбца для IS (NOT) NULL, но будет
тем временем вы можете использовать либо:
filter (tablename.is_deleted.op( "IS NOT" ) (True))
фильтр (coalesce (tablename.is_deleted, False)!= True)