在 NodeJS 中利用 bookshelf.js 进行事务管理

作者:link

术语事务指的是构成单一逻辑工作单元的操作的集合。比如:将钱从一个账户转到另一个账户就是一个事务,该事务包括分别针对每个账户的两个更新。

英文中transaction又是交易的意思,我想应该是因为事务(transaction)管理的场景首先是出现在利用银行账户进行交易(transaction)的过程中,所以计算机科学家们把数据库的这一特性称为事务(transaction)。

事务有以下几个属性:

  • 原子性(atomicity):事务的所有操作在数据库中要么全部正确反映出来,要么完全不反映。
  • 一致性(consistency):隔离执行事务时(换言之,在没有其他事务并发执行的情况下)保持数据库的一致性。
  • 隔离性(isolation): 尽管多个事务可能并发执行,但系统保证,对于任何一对事务T<sub>i</sub>和T<sub>j</sub>,在T<sub>i</sub>看来,T<sub>j</sub>或者在T<sub>i</sub>开始之前已经执行完成,或者在T<sub>i</sub>完成之后开始执行。因此,每一个事务都感觉不到系统中有其他事务在并发执行。
  • 持久性(durability):一旦事务执行成功,它对数据库的改变必须是永久的——即一个可能出现的系统故障不应该导致数据库忽略成功完成的事务。

这些性质通常成为ACID特性,这一缩写来自四条性质的首字母。

在NodeJS中我们可以借助ORM框架来方便地实现事务操作,这里用bookshelf.js来举例说明。

bookshelf.js是一个基于knex.js的Node.js ORM框架,支持PostgreSQL,MySQL和SQLite3

简单来说,bookself是一个优秀的代码库,它易于阅读、理解、可扩展。它不强制你使用任何特定的校验scheme,而是提供灵活有效的关系或嵌套关系加载策略,一级类支持事务。它是一个精益的对象关系映射器(lean Object Relation Mapper),允许你使用原始的knex接口,因为当你需要自定义查询时,它有时并不能完全满足老一套的惯例。

bookshelf遵从和backbone.js一样棒的Models和Collections思想,使用相同的模式、命名惯例和哲学构建轻量、易于操控的ORM。如果你已经了解backbone,你就知道如何使用bookshelf.

bookshelf使用bluebird管理异步操作。

下面来演示如何使用bookshelf进行事务管理

model/db_config.json

{
  client: 'mysql',
  connection: {
    host     : '127.0.0.1',
	port     : 3306,
    user     : 'your_database_user',
    password : 'your_database_password',
    database : 'myapp_test',
    charset  : 'utf8'
  }
}

model/base.js

'use strict'

/**
 * init bookshelf connection
 * @type {[type]}
 */
let Bookshelf = null,
    dbConfig = require('./db_config');
module.exports = function() {
	"use strict";

	if (Bookshelf) {
		return Bookshelf;
	}

	var knex = require('knex')(dbConfig);

	Bookshelf = require('bookshelf')(knex);

	/**
     * This solves the circular module dependency problem created by Bookshelf models
     * in a previous commit #38d98bb4c33e91b636a3538bd546ebe7f5077328
     **/
	Bookshelf.plugin('registry');

	return Bookshelf;
};

model/users.js

'use strict'

const bookshelf = require('./base')();

// 一般情况下后台或者DBA的同学会帮我们把数据库和表建好,我们直接操作就好。所以我们只需要利用已有的表结构初始化一个ORM的实例来进行操作。
var Users = bookshelf.Model.extend({
    tableName: 'users'
});

module.exports = Users;

model/room.js

'use strict'

const bookshelf = require('./base')();

// 一般情况下后台或者DBA的同学会帮我们把数据库和表建好,我们直接操作就好。所以我们只需要利用已有的表结构初始化一个ORM的实例来进行操作。
var Room = bookshelf.Model.extend({
    tableName: 'room'
});

module.exports = Room;

controllers/transaction.js

'use strict'

const Bookshelf = require('../model/base')();
const Room = require('../model/room');
const Users = require('../model/users');

Bookshelf.transaction(function (t) {
    return Users.forge(data) // 需要存入users表的数据
    .save(null, { transacting: t })
    .then(function (users){
        return Room.forge({ userId: user.id }) // 将新增的user同时存入room表中
        .save(null, { transacting: t })
        .then(function (){
            return users; // 返回users表的model
        });
    })
}).then((users) => {
	// doSomething
}).catch((error) => {
	//handle error
})

关于bookshelf.js的更多信息可以参考他们的官网:http://bookshelfjs.org/

原文链接:http://ivweb.io/topic/56628e8bd91952db73b41f4f

原文链接:

编辑于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏海说

基于dbunit进行mybatis DAO层Excel单元测试

DAO层测试难点 可重复性,每次运行单元测试,得到的数据是重复的 独立性,测试数据与实际数据相互独立 数据库中脏数据预处理 不能给数据库中数据带来变化 DAO层...

2600
来自专栏牛肉圆粉不加葱

Spark Task 的执行流程③ - 执行 task

创建、分发 Task一文中我们提到 TaskRunner(继承于 Runnable) 对象最终会被提交到 Executor 的线程池中去执行,本文就将对该执行过...

521
来自专栏Spark生态圈

[spark streaming] ReceiverTracker 数据产生与存储

在Spark Streaming里,总体负责任务的动态调度是JobScheduler,而JobScheduler有两个很重要的成员:JobGenerator 和...

741
来自专栏阿杜的世界

Spring Boot with Mysql

Spring Boot大大简化了持久化任务,几乎不需要写SQL语句,之前我写过一篇关于Mongodb的——RESTful:Spring Boot with Mo...

752
来自专栏JackieZheng

漫谈可视化Prefuse(一)---从SQL Server数据库读取数据

  上篇《可视化工具solo show-----Prefuse自带例子GraphView讲解》主要介绍了整个Prefuse工具集具有的一些特征、框架的运行流程,...

1776
来自专栏linux驱动个人学习

vivi虚拟摄像头驱动程序

一、vivi虚拟摄像头驱动 基于V4L2(video for linux 2)摄像头驱动程序,我们减去不需要的ioctl_fops的函数,只增加ioctl函数增...

3014
来自专栏函数式编程语言及工具

Scalaz(12)- Monad:再述述flatMap,顺便了解MonadPlus

  在前面的几篇讨论里我们初步对FP有了些少了解:FP嘛,不就是F[A]吗?也是,FP就是在F[]壳子(context)内对程序的状态进行更改,也就是在F壳子(...

1767
来自专栏潇涧技术专栏

Lint Tool Analysis (2)

本系列的几篇源码分析文档意义不大,如果你正好也在研究lint源码,或者你想知道前面自定义lint规则中提出的那几个问题,抑或你只是想大致了解下lint的源码都有...

511
来自专栏架构师之旅

Spring框架知识总结-Spring事务配置的五种方式

前段时间对Spring的事务配置做了比较深入的研究,在此之间对Spring的事务配置虽说也配置过,但是一直没有一个清楚的认识。通过这次的学习发觉Spring的事...

1649
来自专栏Coding01

学几个 Laravel Eloquent 方法和技巧

我第一次寻找所谓的 Laravel 框架的时候,我的其中一个目标就是要找:利用最简单的操作数据库的方法。后来目标就停在了 Eloquent ORM 上。

551

扫码关注云+社区