2

sqlite3在我的网络应用程序中将 npm 包用于模拟餐厅(用于学习目的)。在我当前的实现中,当我的本地服务器启动时,我正在创建一个menuItems表,如下所示:

var db = new sqlite3.Database(':memory:');
db.serialize(function() {
  db.run('CREATE TABLE menuItems ('
          + 'itemName VARCHAR(255),'
          + 'itemDescription VARCHAR(255),'
          + 'unitPrice REAL'
          + ');')
  .run("INSERT INTO menuItems (itemName, itemDescription, unitPrice) VALUES"
          + " ('Fish Filet', 'Yummy fish in a sandwich.', 9.95)")
});

但是,我希望我的menuItems表格不在js文件中进行硬编码,并且希望实际项目与该文件分开js。有几个我能想到的解决方案:

  1. 使用当前实现和硬编码命令将单个项目插入表中。

  2. 使用 SQLite3 导入csv文件并将其插入表中。*

  3. 从文件中读入项目csv并通过循环插入它们。

  4. 在服务器启动之前建立数据库并将其传递给sqlite3.Database()构造函数。

*我想选择选项 2。但是,由于这是出于学习目的,我愿意接受任何建议,包括但不限于使用不同的数据库管理包或使用不同类型的文件(也许只是一个txt文件什么的)。

我知道有一些方法csv可以使用 SQLite3导入文件。我正在尝试做同样的事情,但来自sqlite3npm 包。当我尝试通过链接页面中的相同命令(这似乎是特定于 sqlite3 的命令)导入文件时,

db.run('.import "C:/Users/path/to/csv/file.csv"'
      + 'INTO TABLE menuItems'
      + 'FIELDS TERMINATED BY ","'
      + 'ENCLOSED BY "\'"'
      + 'LINES TERMINATED BY \'\\n\''
      + 'IGNORE 1 ROWS'
      + ');');

我收到错误

events.js:183
  throw er; // Unhandled 'error' event
  ^

Error: SQLITE_ERROR: near ".": syntax error

尝试使用正常的 SQL 语法

    .run('BULK INSERT menuItems'
          + 'FROM "C:/Users/path/to/csv/file.csv" '
          + 'WITH '
          + '{ '
          + 'FIRSTROW = 2,'
          + 'FIELDTERMINATOR = ","'
          + 'ROWTERMINATOR = "\n",'
          + 'ERRORFILE = "C:/Users/path/to/csv/error_file.csv" '
          + 'TABLOCK'
          + '}')

我收到错误

events.js:183
      throw er; // Unhandled 'error' event
      ^

Error: SQLITE_ERROR: near "BULK": syntax error

我的语法中有什么不正确的吗?有没有更好/工作/更有效的方法来做到这一点?选项 3 似乎可行,但我还没有尝试过。

4

3 回答 3

1

我最终选择了选项 3。也就是说,我使用fs包读取 CSV 文件,进行一些解析,在对象中返回结果Promise,并在then所述Promise对象中插入数据库。

于 2018-08-10T16:24:05.223 回答
0

尝试

db.run('.import "C:/Users/path/to/csv/file.csv"'
      + 'INTO TABLE menuItems'
      + 'FIELDS TERMINATED BY ","'
      + 'ENCLOSED BY "\'"'
      + 'LINES TERMINATED BY \'\\n\''
      + 'IGNORE 1 ROWS'
      + ')');

以开头的命令.结尾不需要分号

于 2019-06-28T20:14:20.820 回答
0

有一个小例子可以将你的 csv 读入 sqlite3 db。

const sql3 = require('better-sqlite3');
const   db = new sql3( 'memory.db' );
const  csv = require('csv-parser');
const   fs = require('fs');

// create table
db.exec( 'CREATE TABLE IF NOT EXISTS menuItems ( itemName TEXT, itemDescription TEXT, unitPrice REAL );' );
//db.exec( 'DROP TABLE menuItems;' );

const insrow = db.prepare( 'insert into menuItems ( itemName, itemDescription, unitPrice ) VALUES (?, ?, ?)' );


fs.createReadStream('C:/Users/path/to/csv/file.csv')
  .pipe(csv({"separator":";"}))
  .on('data', (row) => {
    
    insrow.run( row.itemName, row.itemDescription, row.unitPrice );
    console.log(row);
  })
  .on('end', () => {
    console.log('CSV file successfully processed');
    db.close();
  });
      

例子中的数据库是memory.db,csv格式不是逗号分隔的,而是分号分隔的,如果需要的话可以改变分隔符。

这种情况下的 csv 必须以itemName;itemDescription;unitPrice标题开头,并且一行看起来像Fish Filet;Yummy fish in a sandwich.;9.95等等。

于 2021-02-03T21:51:56.080 回答