第一次插入后,sqlite3_last_insert_rowid返回0

时间:2013-08-16 23:42:23

标签: ios c sqlite

我正在使用以下代码将行插入数据库。出于某种原因,sqlite3_last_insert_rowid返回插入的第一行的正确行ID,但之后它总是返回0.

插入之间可能已发生其他数据库操作,如删除行。

_scoresDB是一个实例var。

这是我的插入代码:

-(void)insertScore:(OKScore*)score
{
    const char *dbpath = [[self dbPath] UTF8String];

    if(sqlite3_open(dbpath, &_scoresDB) == SQLITE_OK) {

        // Setup the SQL Statement
        if(insertScoreStatement == nil) {
            //OKLog(@"Preparing statement for cache score");
            const char *insertSQL = "INSERT INTO OKCACHE(leaderboardID,scoreValue,metadata,displayString,submitted) VALUES(?,?,?,?,?);";

            if(sqlite3_prepare_v2(_scoresDB, insertSQL, -1, &insertScoreStatement, NULL) != SQLITE_OK) {
                OKLog(@"Failed to prepare score insert statement with message: '%s'", sqlite3_errmsg(_scoresDB));
                return;
            }
        }

        // Bind the score values to the statement
        sqlite3_bind_int(insertScoreStatement, 1, [score OKLeaderboardID]);
        sqlite3_bind_int64(insertScoreStatement, 2, [score scoreValue]);
        sqlite3_bind_int(insertScoreStatement, 3, [score metadata]);

        if([score displayString]) {
            sqlite3_bind_text(insertScoreStatement, 4, [[score displayString] UTF8String], -1, SQLITE_TRANSIENT);
        } else {
            sqlite3_bind_null(insertScoreStatement, 4);
        }
        sqlite3_bind_int(insertScoreStatement, 5, (int)[score submitted]);

        //Execute the SQL statement
        if(sqlite3_step(insertScoreStatement) == SQLITE_DONE) {
            int scoreID = sqlite3_last_insert_rowid(_scoresDB);
            [score setOKScoreID:scoreID];
            OKLog(@"Cached score : %@",score);
        } else {
            OKLog(@"Failed to store score in cache wihth error message: %s",sqlite3_errmsg(_scoresDB));
        }


        sqlite3_reset(insertScoreStatement);
        sqlite3_clear_bindings(insertScoreStatement);
        sqlite3_close(_scoresDB);

    } else {
        OKLog(@"Could not open cache DB insertScore");
    }
}

2 个答案:

答案 0 :(得分:1)

我认为您不应该保留insertScoreStatement。你已经绑定了它的值,现在你绑定了更多的东西。我会把它变成一个局部变量(不是一个类,也不是一个属性),如下所示:

    // Setup the SQL Statement
    sqlite3_stmt *insertScoreStatement
    //OKLog(@"Preparing statement for cache score");
    const char *insertSQL = "INSERT INTO OKCACHE(leaderboardID,scoreValue,metadata,displayString,submitted) VALUES(?,?,?,?,?);";

   if(sqlite3_prepare_v2(_scoresDB, insertSQL, -1, &insertScoreStatement, NULL) != SQLITE_OK) {
        OKLog(@"Failed to prepare score insert statement with message: '%s'", sqlite3_errmsg(_scoresDB));
            return;
        }
    }

答案 1 :(得分:1)

您无法为已关闭的数据库保留语句。 如果您尝试重用该旧语句,则不会插入任何内容(既不会插入旧数据库,也不会插入新数据库,因为该语句不知道它)。

在关闭数据库之前,必须使用sqlite3_finalize释放语句。