5

我有一个 SQLite 查询,例如:

SELECT max(e), url, branch FROM (SELECT max(T1.entry) e, T1.url, T1.branch FROM repo_history T1
                                                        WHERE (SELECT active FROM repos T2 WHERE url = T1.url AND branch = T1.branch AND project = ?1)
                                                        GROUP BY T1.url, T1.branch
                                                        UNION
                                                        SELECT null, T3.url, T3.branch FROM repos T3 WHERE active AND project = ?1 )
                               GROUP BY url ORDER BY e

?1请注意,该参数出现了两次。无论如何,在某些情况下,它可以为空(None在 python 中,据我所知,NULL在 SQLite 中)。这是一个问题,因为我不了解 null 处理,但基本上我没有得到任何回报。

那么,我该如何处理where "project" = ?1何时?1为 NULL?我想避免对它进行 2 个单独的查询。我环顾四周,但我只能找到关于IS NULL/的东西IS NOT NULL,这对我不起作用,因为我不是要检查列是否为空或不为空,而是要检查两个可空值是否匹配,无论它们是否为空。

4

1 回答 1

11

在 SQLite 中,您可以使用IS运算符而不是 = 进行 NULL tolarant 比较。也适用于?插入(与 MikeT 不同)。

Python 示例:

>>> c.execute('SELECT * FROM mytable WHERE userid = ? AND recipe = ?', (3, None)).fetchall()
[]

>>> c.execute('SELECT * FROM mytable WHERE userid = ? AND recipe IS ?', (3, None)).fetchall()
[<Row object>, <Row object>]

>>> c.execute('SELECT * FROM mytable WHERE userid = ? AND recipe is ?', (3, 'TestRecipe')).fetchall()
[<Row object>]

IS 和 IS NOT 运算符的工作方式与 = 和 != 类似,除非其中一个或两个操作数为 NULL。在这种情况下,如果两个操作数都为 NULL,则 IS 运算符的计算结果为 1(真),而 IS NOT 运算符的计算结果为 0(假)。如果一个操作数为 NULL 而另一个不是,则 IS 运算符的计算结果为 0(假)并且 IS NOT 运算符为 1(真)。IS 或 IS NOT 表达式不可能计算为 NULL。运算符 IS 和 IS NOT 与 = 具有相同的优先级。

对于较旧的 MySQL/Mariadb 版本,至少 NULL tolarant 比较运算符是<=>,在 PostgreSQL 中是IS NOT DISTINCT FROM.

PostgreSQL 变体在 SQL:2003 标准中定义。为了临时兼容性,可以从 Python 字典中插入合适的运算符 ...

于 2019-12-19T18:04:44.170 回答