7

当我使用此查询启动代码时,有时我可以看到列出的所有表,有时只有一个,并且总是出现此错误:

查询错误:错误:SQLITE_MISUSE:未知错误

我读过当 SQLITE API 使用不当时会发生 SQLITE_MISUSE 。你能帮我吗,因为我找不到这段代码有什么问题。

编辑。我已经对代码进行了更改以摆脱种族问题。

SQLITE_MISUSE wrror 的消息仍然出现,但是消失的表问题已经消失。我的查询中的种族就是这种情况。

这是代码。

var sqlite3 = require("node-sqlite3");
var fs = require('fs');

var query_count;
var init = function (response) {
    var db = new sqlite3.Database("test.db", function() {

        fs.readFile('./assets/sql/initDB.sql', function(err,data){
            if(err) {
                console.error("Could not open file: %s", err);
                return;
            }

            var query = data.toString('utf8');
            queries = query.split(";");

            db.serialize(function() {
                query_count = queries.length;
                for(var i=0; i<queries.length; i++) {
                    queries[i] = queries[i].replace("\r\n","");
                    db.run(queries[i], function(error) {
                        if(error) {
                            console.log("Query Error: "+error);
                        }

                        query_count--;

                        if( query_count <= 0 ) {
                            db.close();
                            listAllTables(response);
                        }
                    });
                }
            });
        });
    });
};

function listAllTables(response) {
    var db = new sqlite3.Database("./assets/sql/test.db", function () {
        db.all("SELECT name FROM sqlite_master WHERE type = 'table'", function (error, records) {
            for(var record in records) {
                response.write(record+": "+records[record]+"\n");
                for(var prop in records[record]) {
                    response.write("\t"+prop+": "+records[record][prop]+"\n");
                }
            }

            response.end();
            db.close();
        });
    });
}

exports.load_customers = function(response) {
    init(response);
};

查询文件 initDB.sql 是这样的:

CREATE TABLE IF NOT EXISTS TemporaryAuthTokens (
  authToken TEXT PRIMARY KEY NOT NULL UNIQUE,
  expireDate NUMERIC NOT NULL);

CREATE  TABLE IF NOT EXISTS User (
  id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL UNIQUE  ,
  login TEXT NOT NULL ,
  pass TEXT NOT NULL ,
  creationDate NUMERIC NOT NULL ,
  authToken TEXT NULL REFERENCES TemporaryAuthTokens(authToken)
  );
4

2 回答 2

12

你有一个比赛条件;您的最后一个查询(其回调关闭连接)可能会在之前的一个查询完成之前完成,不用说,这会影响之前的查询。您需要重新编写代码,以便最后一个查询完成,而不是最后一个查询开始,关闭连接(例如,为查询数量设置一个计数器,并让每个查询在完成时递减它。递减它的那个为零关闭连接)。

您可能还想查看serialize可用于数据库对象的方法。现在你的初始化查询都是相互独立的,但是如果你开始使用外键约束,如果引用的表还没有创建,你会遇到麻烦,所以你需要强制执行顺序。

于 2012-08-27T09:52:31.090 回答
2

使用承诺。

在这里,我从一个表中获取数据并使用这些数据来创建用于插入另一个表的语句。

serialize()确实运行 1-by-1,但我希望一个查询的响应用于另一个查询。如果我将2nd查询放在回调中,1st那么它会给出SQLITE_MISUSE错误

db.serialize(()=>{

    // QUERY 1 (SELECT) - Get data 
    let promiseGetFoo = new Promise((resolve, reject) => {
      db.all("SELECT * FROM foo", (err, rows) => {
        if (err) {
          console.log(err);
          reject(err);
        } else {
          resolve(rows);
        }
      });
    });

    // QUERY 2 (INSERT) - Use data from QUERY 1
    promiseGetFoo.then((res) => {
      let stmt = (res) => { 
        // code to create INSERT statement 
      }      
      db.run(stmt, (err) => {
        if(err) console.log(err);
        else console.log(">>> Insert DONE");
        closeDb();
      });
    });

});

let closeDb = () => {
    db.close() ;
}
于 2020-05-12T11:43:36.473 回答