Node中通过使用FFI调用外部库文件

Node中通过使用FFI调用外部库文件

因为NPM上的Sqlite3版本过于低了(3.1.8),有很多新的特性和性能提升没有及时更新,所以决定使用最新版本发布的dll/so文件作为方案。
代码如下:

var fs = require('fs')
  , ref = require('ref')
  , ffi = require('ffi')
  , ArrayType = require('ref-array')
var dbName = process.argv[2] || 'test.sqlite3'
var sqlite3 = 'void'
  , sqlite3Ptr = ref.refType(sqlite3)
  , sqlite3PtrPtr = ref.refType(sqlite3Ptr)
  , sqlite3_exec_callback = 'pointer'
  , stringPtr = ref.refType('string')
  , string = ref.types.CString
  , stringArray = ArrayType(string)

var SQLite3 = ffi.Library('./wxsqlite3-sqlite3-aes128_x64/aes128/dll/release/sqlite3_x64', {
  'sqlite3_libversion': [ 'string', [ ] ],
  'sqlite3_open': [ 'int', [ 'string', sqlite3PtrPtr ] ],
  'sqlite3_close': [ 'int', [ sqlite3Ptr ] ],
  'sqlite3_changes': [ 'int', [ sqlite3Ptr ]],
  'sqlite3_exec': [ 'int', [ sqlite3Ptr, 'string', sqlite3_exec_callback, 'void *', stringPtr ] ],
  'sqlite3_key': ['int', [sqlite3Ptr, 'string', 'int']],
  'sqlite3_rekey': ['int', [sqlite3Ptr, 'string', 'int']]
})

var db = ref.alloc(sqlite3PtrPtr)
SQLite3.sqlite3_open(dbName, db)
db = db.deref()
var x = process.argv[3];
if(x === 'create'){
    // SQLite3.sqlite3_exec(db, 'CREATE TABLE tb_test (name VARCHAR, password VARCHAR, gender int, email VARCHAR, address VARCHAR);', null, null, null)
    // SQLite3.sqlite3_exec(db, 'INSERT INTO tb_test VALUES(\'randy3\', \'123456\', 0, \'typhoeus@163.com\', \'Changchun Jilin\');', null, null, null)
    SQLite3.sqlite3_exec(db, 'UPDATE tb_test SET name = \'Lily\' WHERE name = \'randy2\';', null, null, null)

    // SQLite3.sqlite3_rekey(db, '456', 3)

} else {
    
    // SQLite3.sqlite3_key(db, '456', 3);
    // SQLite3.sqlite3_rekey(db, null, 0);
    var rowCount = 0

    var callback = ffi.Callback('int', ['void *', 'int', stringPtr, stringPtr], function (tmp, cols, argv, colv) {
        var obj = {}
        var colName = stringArray.untilZeros(colv).toJSON();
        var colData = stringArray.untilZeros(argv).toJSON();
        for (let i = 0; i < cols; i++) {
            obj[colName[i]] = colData[i]
        }
        console.log(obj);
        // console.log('Row: %j', obj)
        rowCount++
        return 0
    })

    var b = new Buffer('test')
    SQLite3.sqlite3_exec.async(db, 'SELECT * FROM tb_test;', callback, b, null, function (err, ret) {
        if (err) throw err
        console.log('Total Rows: %j', rowCount)
        //console.log('Changes: %j', SQLite3.sqlite3_changes(db))
        //console.log('Closing...')
        //SQLite3.sqlite3_close(db)
        //fs.unlinkSync(dbName)
        fin = true
    })
}
//SQLite3.sqlite3_close(db)


/*
module.exports = {
  fooSync : mylibrary.foo,
  foo: mylibrary.foo.async,
  barSync : mylibrary.bar,
  bar: mylibrary.bar.async
};
*/

ref这个很坑,完全看不懂好吧,不过官方有例子,正好又是跟我一样连的Sqlite3,省去无数麻烦,但有个缺点,只能获取第一列的数据,好在又有ref-array,不过也很难理解,最终凭经验猜到方法用toJSON(),成功搞定,获取到了完整结果集。


相关阅读:


拓展阅读:

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 222,729评论 6 517
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 95,226评论 3 399
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 169,461评论 0 362
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 60,135评论 1 300
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 69,130评论 6 398
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,736评论 1 312
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 41,179评论 3 422
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 40,124评论 0 277
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,657评论 1 320
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,723评论 3 342
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,872评论 1 353
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,533评论 5 351
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 42,213评论 3 336
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,700评论 0 25
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,819评论 1 274
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 49,304评论 3 379
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,876评论 2 361

推荐阅读更多精彩内容