SQLite查询给出了不同的结果

时间:2014-05-01 19:15:45

标签: sql c database sqlite

我正在用C开发一个小应用程序,我想使用SQLite对数据库执行一些查询。问题是我得到了一个奇怪的行为。

如果我在Firefox插件的同一个数据库上运行相同的查询" SQLite Manager"我得到了预期的结果,但如果我在我的程序中通过C接口运行它,我得到一个错误的结果元组的顺序。

您可以找到数据库here(CSV或SQLite格式),而有问题的查询是这样的:

SELECT name, slot, id, species_id 
FROM type_names T, pokemon_types PT, pokemon P 
WHERE T.type_id = PT.type_id 
AND P.id = PT.pokemon_id AND T.local_language_id = 9 
AND P.identifier LIKE '%char%' 
ORDER BY P.species_id;

在Firefox上我得到了预期的结果:

+--------+------+-------+------------+
|  name  | slot |  id   | species_id |
+--------+------+-------+------------+
| Fire   |    1 |     4 |          4 |
| Fire   |    1 |     5 |          5 |
| Fire   |    1 |     6 |          6 |
| Flying |    2 |     6 |          6 |
| Fire   |    1 | 10034 |          6 |
| Dragon |    2 | 10034 |          6 |
| Fire   |    1 | 10035 |          6 |
| Flying |    2 | 10035 |          6 |
| Fire   |    1 |   390 |        390 |
+--------+------+-------+------------+

在C中我得到了这个:

+--------+------+-------+------------+
|  name  | slot |  id   | species_id |
+--------+------+-------+------------+
| Fire   |    1 |     4 |          4 |
| Fire   |    1 |     5 |          5 |
| Flying |    2 |     6 |          6 |
| Flying |    2 | 10035 |          6 |
| Fire   |    1 |     6 |          6 |
| Fire   |    1 | 10034 |          6 |
| Fire   |    1 | 10035 |          6 |
| Dragon |    2 | 10034 |          6 |
| Fire   |    1 |   390 |        390 |
+--------+------+-------+------------+

我还尝试了一些只使用该查询访问数据库的简单代码,我总是遇到同样的问题,这里是代码:

#include <stdio.h>
#include <sqlite3.h>

// argument 1 is path to db, argument 2 is query

static int callback(void *NotUsed, int argc, char **argv, char **azColName){

    int i;
    for(i=0; i<argc; i++){
        printf("%s\t", argv[i] ? argv[i] : "NULL");
    }
    printf("\n");
    return 0;
}

int main(int argc, char **argv){
    sqlite3 *db;
    char *zErrMsg = 0;
    int rc;

    if( argc!=3 ){
        fprintf(stderr, "Usage: %s DATABASE SQL-STATEMENT\n", argv[0]);
        return(1);
    }
    rc = sqlite3_open(argv[1], &db);
    if( rc ){
        fprintf(stderr, "Can't open database: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return(1);
    }
    rc = sqlite3_exec(db, argv[2], callback, 0, &zErrMsg);
    if( rc!=SQLITE_OK ){
        fprintf(stderr, "SQL error: %s\n", zErrMsg);
        sqlite3_free(zErrMsg);
    }
    sqlite3_close(db);
    return 0;
}

2 个答案:

答案 0 :(得分:0)

您的ORDER BY条款的全部内容为ORDER BY P.species_id。因此,两个输出都是100%正确的(它们都以P.species_id值递增的方式排序)。如果您希望按其他列对结果进行排序,则需要在ORDER BY子句中包含这些列。

答案 1 :(得分:0)

好像你期待的那样:

SELECT name, slot, id, species_id 
FROM type_names T, pokemon_types PT, pokemon P 
WHERE T.type_id = PT.type_id 
AND P.id = PT.pokemon_id 
AND T.local_language_id = 9 
AND P.identifier LIKE '%char%' ORDER BY P.species_id, id, slot ASC;

进一步限定您的ORDER BY能够完全复制结果。