假设我有一个表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)