SQLite常见问题及解决方法
1. 如何建立自动增长字段?
简短回答:声明为 INTEGER PRIMARY KEY
的列将会自动增长。
2. SQLite支持哪些数据类型?
SQLite使用动态类型。内容可以存储为 INTEGER
、REAL
、TEXT
、BLOB
或 NULL
。
3. 为什么SQLite不允许在同一个表不同的两行上使用0和0.0作主键?
主键必须是数值类型,将主键改为 TEXT
型将不起作用。每一行必须有一个唯一的主键。对于一个数值型列,SQLite认为 0
和 0.0
是相同的,因为它们在作为整数比较时是相等的。
4. 多个应用程序或一个应用程序的多个实例可以同时访问同一个数据库文件吗?
多个进程可同时打开同一个数据库。多个进程可以同时进行 SELECT
操作,但在任一时刻,只能有一个进程对数据库进行更改。SQLite使用读、写锁控制对数据库的访问。但使用时要注意:如果数据库文件存放于一个NFS文件系统上,这种锁机制可能不能正常工作。
5. SQLite线程安全吗?
SQLite 是线程安全的。由于很多用户会忽略我们在上一段中给出的建议,我们做出了这种让步。但是,为了达到线程安全,SQLite在编译时必须将 SQLITE_THREADSAFE
预处理宏置为1。在Windows和Linux上,已编译的好的二进制发行版中都是这样设置的。如果不确定你所使用的库是否是线程安全的,可以调用 sqlite3_threadsafe()
接口找出。
6. 在SQLite数据库中如何列出所有的表和索引?
如果你运行 sqlite3
命令行来访问你的数据库,可以键入 .tables
来获得所有表的列表。或者,你可以输入 .schema
来看整个数据库模式,包括所有的表的索引。输入这些命令,后面跟一个 LIKE
模式匹配可以限制显示的表。
7. SQLite数据库有已知的大小限制吗?
在Windows和Unix下,版本2.7.4的SQLite可以达到2的41次方字节(2T字节)。老版本的为2的31次方字节(2G字节)。SQLite版本2.8限制一个记录的容量为1M。SQLite版本3.0则对单个记录容量没有限制。表名、索引表名、视图名、触发器名和字段名没有长度限制。但SQL函数的名称(由 sqlite3_create_function()
API函数创建)不得超过255个字符。
8. 在SQLite中,VARCHAR字段最长是多少?
SQLite不强制 VARCHAR
的长度。你可以在SQLITE中声明一个 VARCHAR(10)
,SQLite还是可以很高兴地允许你放入500个字符。并且这500个字符是原封不动的,它永远不会被截断。
9. 在SQLite中,如何在一个表上添加或删除一列?
SQLite有有限地 ALTER TABLE
支持。你可以使用它来在表的末尾增加一列,可更改表的名称。如果需要对表结构做更复杂的改变,则必须重新建表。重建时可以先将已存在的数据放到一个临时表中,删除原表,创建新表,然后将数据从临时表中复制回来。
10. 插入太慢——我只能每秒做几十次插入
实际上,SQLite在普通的桌面电脑上可以每秒执行50000或更多的插入语句。但它只能每秒做几十个事务。事务被磁盘驱动器的转速限制了。一个事务正常需要磁盘的两次完整的循环(rotatioins),这对于一个7200转每分钟的磁盘驱动器来说,大约每秒限制在60个事务。默认地,每个插入语句本身都有自己的事务,但如果把多个插入语句用 BEGIN…COMMIT
包围,那么所有的插入就被组合到单个事务中。提交所需要的时间被所有的包含的插入语句分摊,所以每个插入语句的时间都会大大减少。
11. 我意外地删除了一些重要信息,如何恢复?
如果有数据库的备份拷贝,那么从备份可以恢复信息。如果没有,那么恢复就非常困难了。
12. SQLite支持外键吗?
从3.6.19版本,SQLite开始支持外键约束。之前的版本能够解析外键约束,但不强制实施。
13. 使用sqlite视图遇到的no such column问题
出现这个问题是因为在建立视图时,从两个关联表查询出的字段,有名字相同的列。解决办法:在查询这个有相同列表的字段时,加一个别名即可。
14. 如果其中有两个表的字段名称相同,在插入的时候也会出现一些异常情况
比如 terminal
表和 vehicle
表都有字段 vehicleid
,将数据 "b413e9f2-d551-462f-93b7-53058e65d7e4" 插入 terminal
表中的 vehicleid
的时候,只显示0。这个问题很奇怪。
15. 使用游标的地方一定要关闭。不然如果多次运行的时候会报异常。
16. SQLite数据库文件损坏
由于各种原因,SQLite数据库文件可能会损坏,导致无法正常访问数据库。可能的原因包括系统崩溃、存储介质故障或者数据库操作错误等。当数据库文件损坏时,常见的表现是出现类似以下错误:sqlite[s10]: disk i/o error
。这种错误通常意味着数据库文件发生了I/O错误,即输入/输出错误。在遇到这种情况时,可以尝试以下方法来解决:
- 使用SQLite官方工具进行修复:SQLite提供了一些官方工具,如
sqlite3_analyzer
和 sqlite3
等,可以用于检查和修复损坏的数据库文件。可以尝试使用这些工具来修复数据库文件。 - 恢复备份文件:如果有数据库的备份文件,可以尝试使用备份文件来替换损坏的数据库文件。这样可以避免数据丢失。
- 重新创建数据库文件:如果以上方法无法解决问题,可以尝试重新创建数据库文件。这将会丢失数据库中的所有数据,所以务必在执行此操作前做好备份工作。
17. 数据库锁问题
在多线程或多进程环境下,SQLite数据库可能会遇到锁问题,导致数据库操作阻塞或者出现死锁。常见的表现包括:database is locked
(表示数据库被其他进程或线程锁定,当前操作无法继续)、database table is locked
(表示数据库表被锁定,当前操作无法继续)。当遇到数据库锁问题时,可以采取以下方法来解决:
18. 数据库性能问题
在处理大量数据或复杂查询时,SQLite数据库可能会遇到性能问题,导致数据库操作变慢或者程序运行效率低下。常见的表现包括数据库查询时间过长、数据库操作频繁导致程序卡顿。当遇到数据库性能问题时,可以采取以下方法来优化:
19. SQLiteOpenHelper错误
SQLiteOpenHelper是Android开发中常用的数据库助手类,用于创建和管理SQLite数据库。在使用SQLiteOpenHelper的过程中,可能会遇到一些错误,下面列举了几个常见的问题及其解决办法:
- SQLiteOpenHelper的子类未正确实现
onCreate
或 onUpgrade
方法:在自定义的SQLiteOpenHelper子类中,需要重写 onCreate
和 onUpgrade
方法。如果忘记了实现这些方法,或者方法中的执行逻辑有误,将会导致数据库创建或升级失败。解决办法是确保在子类中正确实现这两个方法,并根据具体需求编写正确的逻辑。 - 数据库文件已损坏或不完整:有时,可能会因为一些原因导致数据库文件损坏或不完整,这会导致SQLiteOpenHelper抛出异常。通常情况下,这种问题可以通过删除数据库文件并重新创建来解决。
- 数据库版本号与实际版本不匹配:使用SQLiteOpenHelper时,需要指定数据库的版本号。如果在应用升级后没有更新版本号,那么SQLiteOpenHelper将无法正确处理数据库的升级操作。解决办法是在应用升级时更新数据库版本号,并在SQLiteOpenHelper子类的
onUpgrade
方法中编写正确的升级逻辑。 - 表格创建语句有误或与实际表结构不匹配:在SQLiteOpenHelper的
onCreate
方法中,需要执行创建表格的操作。如果在创建表格时的SQL语句有误,或者与实际表结构不匹配,将会导致数据库操作失败。解决办法是仔细检查创建表格的SQL语句,并确保其与实际表结构一致。 - 数据库连接未关闭:在使用SQLiteOpenHelper时,需要注意及时关闭数据库连接,否则可能导致资源泄露或无法打开数据库。解决办法是在适当的时候调用
close
方法来关闭数据库连接。
20. 如何从存在的表中添加或者删除列?
SQLite支持有限的 ALTER TABLE
语句,你可以用来在表的末尾添加列或者更改表的名字。如果要做更复杂的修改,那么不得不要重新创建表,可以先把数据存到一个临时表中,接着删除表,再创建新表,最后把数据从临时表复制到新表。示例:
BEGIN TRANSACTION;
CREATE TEMPORARY TABLE t1_backup(a,b);
INSERT INTO t1_backup SELECT a,b FROM t1;
DROP TABLE t1; CREATE TABLE t1(a,b);
INSERT INTO t1 SELECT a,b FROM t1_backup;
DROP TABLE t1_backup;
COMMIT;