微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

使用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)

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐