在SQLite中使用索引优化查询速度

本文通过实验展示了在SQLite数据库中如何利用索引来优化多表联合查询的效率。实验对比了不同索引配置下查询速度的变化,揭示了索引在查询优化中的关键作用。

本文来自 sqlitechina.org  作者: 工友

 

在进行多个表联合查询的时候,使用索引可以显著的提高速度:

建立三个表:

create table t1
(id integer primary key,
num integer not null,
word1 text not null,
word2 text not null);
create table t2
(id integer primary key,
num integer not null,
word1 text not null,
word2 text not null);
create table t3
(id integer primary key,
num integer not null,
word1 text not null,
word2 text not null);



建立若干索引:
t1表:在num,word1,word2上有复合索引
t2表:在num,word1,word2上各有一个索引
t3表:在word1上有一个索引

create index idxT1 on t1(num,word1,word2);
create index idxT2Num on t2(num);
create index idxT2Word1 on t2(word1);
create index idxT2Word2 on t2(word2);
create index idxT3Word1 on t3(word1);



向三个表中各插入10000行数据,其中num项为随机数字,word1和word2中是英文单词,三个表中对应的num,word1和word2列中都包含有部分相同值,但是它们在表中出现的顺序不同。

速度测试结果:

1) select count(*) from t1,t3 where t1.word2=t3.word2;
很慢(t3.word2上没有索引)
2) select count(*) from t3,t1 where t1.word2=t3.word2;
很慢(t1.word2上没有独立索引)
3) select count(*) from t1,t2 where t1.word2=t2.word2;
很快(t2.word2上有索引)
4) select count(*) from t2,t1 where t1.word2=t2.word2;
很慢(t1.word2上没有独立索引)
5) select count(*) from t1,t2 where t1.num=t2.num;
很快(t2.num上有索引)
6) select count(*) from t2,t1 where t1.num=t2.num;
很快(t1的复合索引中,第一个列是num)
7) select count(*) from t1,t3 where t1.num=t3.num;
很慢(t3.num上没有索引)
8) select count(*) from t3,t1 where t1.num=t3.num;
很快(t1的复合索引中,第一个列是num)



结 论:在from子句后面的两个表中,如果第2个表中要查询的列里面带有索引,这个查询的速度就快,反之就慢。比如第三个查询,from后面的第2个表是 t2,t2在word2上有索引,所以这个查询就快,当输入SQL命令并回车后,查询结果就立即显示出来了,但是如果使用第4个查询命令(即把t1和t2 的位置互换),查询起来却用了1分零6秒。

可见索引的建立对于提高数据库查询的速度是非常重要的。

更多关于SQLite查询优化的知识可以参考《Chris Newman》写的《SQLite》一书的第四章:《Query Optimization》

### 在C++中优化SQLite查询性能的方法 在C++中使用SQLite时,查询性能的优化可以通过多种方式实现。以下是一些关键的技术和策略: #### 1. 使用索引 为频繁查询的列创建索引可以显著提高查询速度[^1]。例如,如果经常根据`Name`字段进行查询,则可以创建一个索引: ```sql CREATE INDEX idx_users_name ON Users(Name); ``` #### 2. 批量处理数据 减少数据库交互次数可以提升性能。通过批量插入或更新数据,而不是逐条操作,能够有效减少开销[^2]。例如: ```cpp std::string sql = "BEGIN TRANSACTION; "; for (int i = 0; i < 1000; ++i) { sql += "INSERT INTO Users (Name, Age) VALUES ('User" + std::to_string(i) + "', " + std::to_string(i % 50) + ");"; } sql += "COMMIT;"; sqlite3_exec(db, sql.c_str(), nullptr, nullptr, nullptr); ``` #### 3. 避免SQL注入并使用预处理语句 预处理语句不仅提高了安全性,还可以通过缓存编译后的SQL语句来提升性能[^3]。例如: ```cpp const char* sql = "INSERT INTO Users (Name, Age) VALUES (?, ?);"; sqlite3_stmt* stmt; sqlite3_prepare_v2(db, sql, -1, &stmt, nullptr); for (int i = 0; i < 1000; ++i) { sqlite3_bind_text(stmt, 1, ("User" + std::to_string(i)).c_str(), -1, SQLITE_STATIC); sqlite3_bind_int(stmt, 2, i % 50); sqlite3_step(stmt); sqlite3_reset(stmt); } sqlite3_finalize(stmt); ``` #### 4. 调整缓存大小 通过调整页面缓存大小,可以优化内存使用查询性能[^4]。例如: ```cpp sqlite3_exec(db, "PRAGMA cache_size = 10000;", nullptr, nullptr, nullptr); ``` #### 5. 启用WAL模式 写前日志(WAL)模式可以提高并发性能,尤其是在多线程环境中[^5]。启用WAL模式的代码如下: ```cpp sqlite3_exec(db, "PRAGMA journal_mode = WAL;", nullptr, nullptr, nullptr); ``` #### 6. 减少不必要的计算 尽量避免在SQL语句中进行复杂的计算,尤其是那些可以在应用程序层面完成的操作[^6]。例如,将日期格式化逻辑从SQL移至C++代码。 #### 7. 确保适当的数据类型 选择合适的数据类型以减少存储空间需求并加快查询速度[^7]。例如,使用`INTEGER`代替`TEXT`存储数字。 #### 8. 分析查询计划 使用`EXPLAIN QUERY PLAN`分析查询执行计划,识别潜在瓶颈并加以改进[^8]。例如: ```sql EXPLAIN QUERY PLAN SELECT * FROM Users WHERE Age > 30; ``` ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值