SQLite:错误调用sqlite3_prepare_v2 - > SIGSEGV

时间:2012-10-20 18:11:33

标签: c sqlite sigsegv

我被SQLite困在家庭作业上。我用2列;第一个用于产品,第二个用于计数。用户添加新产品,更新计数。我们必须控制,用户不会再次添加相同的产品,或阻止他选择比可用的更多的单位。我们必须经常使用它,所以我创建了函数:

int exists(char *param, sqlite3** ppDb) //0 if product exists
{
    int error = 0;
    char *a = NULL;
    sqlite3_stmt **ppStmt = NULL;
    const char **pzTail = NULL;
    char *zSQL = sqlite3_mprintf("SELECT 'products' FROM 'table' WHERE 'products' LIKE '%q'", param);
//HERE IT FALS
    error = sqlite3_prepare_v2(
      *ppDb,                /* Database handle */
      zSQL,                 /* SQL statement, UTF-8 encoded */
      (sizeof(zSQL)+1),         /* Maximum length of zSql in bytes. */
      ppStmt,               /* OUT: Statement handle */
      pzTail                /* OUT: Pointer to unused portion of zSql */
    );
    sqlite3_free(zSQL);
    a = (char*) sqlite3_column_text(*ppStmt, 0);
    return strcmp(a, param); //0 if same -> product is in db yet
}
//similar one for count

呼叫

int main(int argc, const char *argv[])
{
    sqlite3 *pDb;
    int error = 0;
//parsing input
    error = sqlite3_open(argv[1], &pDb);
    if (error == 0)
    {
        sqlite3_exec(
          pDb,      /* An open database */
          "CREATE TABLE 'table' ('products', 'quantity')",  /* SQL */
          0,        /* Callback function */
          NULL,     /* 1st argument to callback */
          NULL      /* Error msg written here */
        );

        if (exists(param[1], &pDb) == 0) 
        {
            fprintf (stderr, "ERROR: Product exists yet\n");
        }
        else
        {
            char *zSQL = sqlite3_mprintf("INSERT INTO 'table' VALUES ('%q', '0')", param[1]);
            error = sqlite3_exec(
              pDb,      /* An open database */
              zSQL,     /* SQL to be evaluated */
              0,        /* Callback function */
              NULL,     /* 1st argument to callback */
              NULL      /* Error msg written here */
            );
            sqlite3_free(zSQL);
            if (error == 0) printf("Added\n");
            else printf("%i", error);
        }
    }
    else return 1;
    return 0;
}

sqlite3_prepare_v2失败。我希望pDb上的指针有问题,但是我无法修复它(我不是指针的粉丝 - 对于初学者来说太强大了)。当它失败时,调试器堆叠在sqlite3.c中的第93396行(* ppStmt = 0; - 它写在某处,不应该在哪里)。

在linux x64上编译:

gcc -std=c99 -Wall -pedantic -Wextra -Werror -DSQLITE_THREADSAFE=0 -ldl -o sqlite main.c sqlite3.c

没有错(如果我复制错误的括号,忽略它 - 这不是问题),SQLite 3.7.14.1

对不起我的英语,我来自捷克语。

1 个答案:

答案 0 :(得分:0)

sqlite3_prepare_v2想要将语句指针写入输出变量,但是你没有给它指向这个指针变量的指针,你给它一个NULL指针。 使用:

sqlite3_stmt *pStmt;
sqlite3_prepare_v2(..., &pStmt, ...);

另请注意,标识符应引用"double quotes"[brackets]

`backticks`

但不是'single quotes',它们用于文字字符串。