我如何可靠地在SQLite中检查特定的用户表是否存在?

我并不是要求使用不可靠的方法,比如检查表上的“select *”是否返回错误(这是一个好主意吗?)

原因如下:

在我的程序中,我需要创建并填充一些表,如果它们还不存在的话。

如果它们已经存在,我需要更新一些表。

我是否应该采取其他路径来表示已经创建了相关的表-例如,通过在磁盘上的程序初始化/设置文件中创建/放置/设置某个标志?

或者我的方法有意义吗?


当前回答

在swift的数据库中,表是否存在

func tableExists(_ tableName:String) -> Bool {
        sqlStatement = "SELECT name FROM sqlite_master WHERE type='table' AND name='\(tableName)'"
        if sqlite3_prepare_v2(database, sqlStatement,-1, &compiledStatement, nil) == SQLITE_OK {
            if sqlite3_step(compiledStatement) == SQLITE_ROW {
                return true
            }
            else {
                return false
            }
        }
        else {
            return false
        }
            sqlite3_finalize(compiledStatement)
    }

其他回答

一种变体是使用SELECT COUNT(*)而不是SELECT NAME,即。

SELECT count(*) FROM sqlite_master WHERE type='table' AND name='table_name';

如果表不存在,返回0,如果存在,返回1。这可能在你的编程中很有用,因为数值结果处理起来更快/更容易。下面说明了如何在Android中使用SQLiteDatabase, Cursor, rawQuery带参数来做到这一点。

boolean tableExists(SQLiteDatabase db, String tableName)
{
    if (tableName == null || db == null || !db.isOpen())
    {
        return false;
    }
    Cursor cursor = db.rawQuery(
       "SELECT COUNT(*) FROM sqlite_master WHERE type = ? AND name = ?",
       new String[] {"table", tableName}
    );
    if (!cursor.moveToFirst())
    {
        cursor.close();
        return false;
    }
    int count = cursor.getInt(0);
    cursor.close();
    return count > 0;
}

我喜欢的方法是:

SELECT "name" FROM pragma_table_info("table_name") LIMIT 1;

如果您得到一个行结果,则该表存在。这是更好的(对我),然后检查sqlite_master,因为它也将检查附加和临时数据库。

我想补充Diego Vélez关于PRAGMA声明的回答。

从https://sqlite.org/pragma.html我们可以得到一些有用的函数,可以返回关于我们数据库的信息。 在此,我引用以下内容:

例如,索引中有关列的信息可以使用index_info pragma读取,如下所示: 编译指示index_info(“idx52”); 或者,可以使用以下方法读取相同的内容: SELECT * FROM pragma_index_info('idx52'); 表值函数格式的优点是查询可以只返回PRAGMA列的一个子集,可以包含WHERE子句,可以使用聚合函数,并且表值函数可以只是连接中的几个数据源之一……

Diego的回答给了PRAGMA table_info(table_name)一个选项,但这在您的其他查询中没有多大用处。

所以,要回答OPs问题并改进diego的回答,你可以做

SELECT * FROM pragma_table_info('table_name');

或者更好,

SELECT name FROM pragma_table_list('table_name');

如果你想模仿poorluzer投票最多的答案。

注意,要检查TEMP数据库中是否存在表,必须使用sqlite_temp_master而不是sqlite_master:

SELECT name FROM sqlite_temp_master WHERE type='table' AND name='table_name';

我错过了FAQ条目。

不管怎样,为了将来的参考,完整的查询是:

SELECT name FROM sqlite_master WHERE type='table' AND name='{table_name}';

其中{table_name}是要检查的表的名称。

参考文档部分:数据库文件格式。SQL数据库模式的存储

这将返回指定名称的表列表;也就是说,游标的计数将为0(不存在)或1(确实存在)