使用IFNULL会阻止SQLite使用索引吗?

假设我有一个表T和一个在字段f上的索引.我想基于一些整数myF对f进行过滤.如果整数为0,则我希望所有记录为f为空.可以肯定的写法是:

db.rawQuery("SELECT someField FROM T WHERE "
        + (myF == 0 ? "f IS NULL" : "f = ?"),
        (myF == 0 ? new String[] {}, new String[] {String.valueOf(myF)}));

这有点不方便;特别是如果查询比这更复杂并且具有其他参数.所以我以为我会写

db.rawQuery("SELECT someField FROM T WHERE IFNULL(f, 0) = ?", 
        new String[] {String.valueOf(myF)});

相反,它更加简单,易于阅读并且易于维护.¹

我的问题是:如果f上有一个索引,sqlite仍会使用该索引还是将其用于表扫描?我问是因为在后一种情况下,我不是比较字段和参数,而是比较表达式和参数,而且我不确定sqlite查询优化器的“智能程度”.

¹请注意,数据库中没有f = 0的记录.

解决方法:

这将导致表扫描.

使用sqlite3命令行客户端的示例:

sqlite> create table t(f);
sqlite> create index tf on t(f);
sqlite> explain query plan select * from t where ifnull(f,0)=1;
0|0|0|SCAN TABLE t (~500000 rows)
sqlite> explain query plan select * from t where f is null;
0|0|0|SEARCH TABLE t USING COVERING INDEX tf (f=?) (~10 rows)
sqlite> explain query plan select * from t where f=1;
0|0|0|SEARCH TABLE t USING COVERING INDEX tf (f=?) (~10 rows)

相关文章

SQLite架构简单,又有Json计算能力,有时会承担Json文件/RES...
使用Python操作内置数据库SQLite以及MySQL数据库。
破解微信数据库密码,用python导出微信聊天记录
(Unity)SQLite 是一个软件库,实现了自给自足的、无服务器...
安卓开发,利用SQLite实现登陆注册功能