Node.js中的ORM
ORM2是一款基于Node.js实现的ORM框架,名字相当的霸气,算是同类框架中非常出色的一款,具体介绍请猛击:https://github.com/dresende/node-orm2
刚接触Node.js + MySQL,在引入项目之初,受Asp.Net经验的影响,产生了许多不小的麻烦。下面是我定义的一个BaseProvider,作为所有DB Provider的父类,提供了一些公共的方法和属性。
function BaseProvider() {
this.table_name = {};
this.properties = {};
this.opts = {};
this.getProviderModel = function (callback) {
define(this.table_name, this.properties, this.opts, function (error, model) {
callback(error, model);
});
};
};
var define = function (name, properties, opts, callback) {
ORM.connect(pomelo.app.get("mysql"), function (error, db) {
if (error) {
logger.error(error);
callback(error);
} else {
db.settings.set("connection.pool", true);
var model = db.define(name, properties, opts);
callback(null, model);
}
});
};
getProviderModel 相当于获得一个Model,具有CRUD的功能,简化我们写SQL语句的繁琐过程,下面是一个具体子类的实现。
function Dungeon() {
this.table_name = "dungeon_userdungeons";
this.properties = {
dungeon_userdungeon_id: Number,
user_id: Number,
dungeon_id: String,
latest_scene_id: String,
status: Number,
create_datetime_utc: Date
};
this.opts = {id: 'dungeon_userdungeon_id'};
this.getUserCompleteDungeonIds = function (user_id, callback) {
this.getProviderModel(function (error, model) {
model.aggregate(["dungeon_id"], {user_id: user_id, status: Dungeon.Status.Complete}).max("dungeon_id").groupBy("dungeon_id").get(function (error, userdungeons) {
var dungeon_ids = [];
userdungeons.forEach(function (e) {
dungeon_ids.push(e.dungeon_id);
})
callback(error, dungeon_ids);
});
});
};
};
Dungeon.prototype = new BaseProvider();
在阅读ORM2的源码后,我发现了一个严重的问题,该方法用于连接,返回一个名为db的ORM对象,这个对象的职责是维护Nodejs到具体DB的连接信息、各种配置、连接池、缓存、Model等,而非ADO.Net中单纯的Connection,下面是ORM. Connect的部分代码。
var connect = function () {
try {
var Driver = require("./Drivers/DML/" + proto).Driver;
var settings = new Settings.Container(exports.settings.get('*'));
var debug = extractOption(opts, "debug");
var pool = extractOption(opts, "pool");
var driver = new Driver(opts, null, {
debug: (debug !== null ? Boolean(debug) : settings.get("connection.debug")),
pool: (pool !== null ? Boolean(pool) : settings.get("connection.pool")),
settings: settings
});
db = new ORM(proto, driver, settings);
driver.connect(function (err) {
if (typeof cb === "function") {
if (err) {
return cb(err);
} else {
return cb(null, db);
}
}
db.emit("connect", err, !err ? db : null);
});
} catch (ex) {
if (ex.code === "MODULE_NOT_FOUND" || ex.message.indexOf('find module')) {
return ORM_Error(ErrorCodes.generateError(ErrorCodes.NO_SUPPORT, "CONNECTION_PROTOCOL_NOT_SUPPORTED"), cb);
}
return ORM_Error(ex, cb);
}
return db;
};
而我常用define方法,就是在上面的那个db中的models数组中push一个对象而已。
ORM.prototype.define = function (name, properties, opts) {
var i;
properties = properties || {};
opts = opts || {};
for (i = 0; i < this.plugins.length; i++) {
if (typeof this.plugins[i].beforeDefine === "function") {
this.plugins[i].beforeDefine(name, properties, opts);
}
}
this.models[name] = new Model({
db: this,
settings: this.settings,
driver_name: this.driver_name,
driver: this.driver,
table: opts.table || opts.collection || ((this.settings.get("model.namePrefix") || "") + name),
properties: properties,
extension: opts.extension || false,
indexes: opts.indexes || [],
cache: opts.hasOwnProperty("cache") ? opts.cache : this.settings.get("instance.cache"),
id: opts.id || this.settings.get("properties.primary_key"),
autoSave: opts.hasOwnProperty("autoSave") ? opts.autoSave : this.settings.get("instance.autoSave"),
autoFetch: opts.hasOwnProperty("autoFetch") ? opts.autoFetch : this.settings.get("instance.autoFetch"),
autoFetchLimit: opts.autoFetchLimit || this.settings.get("instance.autoFetchLimit"),
cascadeRemove: opts.hasOwnProperty("cascadeRemove") ? opts.cascadeRemove : this.settings.get("instance.cascadeRemove"),
hooks: opts.hooks || {},
methods: opts.methods || {},
validations: opts.validations || {}
});
for (i = 0; i < this.plugins.length; i++) {
if (typeof this.plugins[i].define === "function") {
this.plugins[i].define(this.models[name], this);
}
}
return this.models[name];
};
ORM2在connect时就把mysql的连接池自己维护起来了,存在一个变量中,所以Provider的每一次操作都会ORM.connect,返回的db作为一个局部变量用完就丢弃,维护的连接池也没有保存,造成连接数太多,从而产生many connection error,下图是mysql统计的数据,丢失的连接数达1000多。

每次调用都是新的连接池。

我在系统启动时创建ORM的DB,存到BaseProvider的原型中。
ORM.connect(pomelo.app.get("mysql"), function (error, db) {
BaseProvider.prototype.db = db;
console.log("init connect~");
});
Define的时候直接访问缓存的db变量。
var define = function (name, properties, opts, callback) {
var model = BaseProvider.prototype.db.define(name, properties, opts);
callback(null, model);
};
我并发执行6个操作,连接池会创建6个连接满足执行需求。

因为先前已经创建了6个连接,第二次执行的时候,则直接返回。

测试了一下午,连接池终于正常了。

Node.js中的ORM的更多相关文章
- 在node.js中,使用基于ORM架构的Sequelize,操作mysql数据库之增删改查
Sequelize是一个基于promise的关系型数据库ORM框架,这个库完全采用JavaScript开发并且能够用在Node.JS环境中,易于使用,支持多SQL方言(dialect),.它当前支持M ...
- [转]在node.js中,使用基于ORM架构的Sequelize,操作mysql数据库之增删改查
本文转自:https://www.cnblogs.com/kongxianghai/p/5582661.html Sequelize是一个基于promise的关系型数据库ORM框架,这个库完全采用Ja ...
- 如何在Node.js中合并两个复杂对象
通常情况下,在Node.js中我们可以通过underscore的extend或者lodash的merge来合并两个对象,但是对于像下面这种复杂的对象,要如何来应对呢? 例如我有以下两个object: ...
- Node.js中的Session,不要觉得简单哦。
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,博客地址为http://www.cnblogs.com/jasonnode/ .学习网站上有对应 ...
- Node.js 中MongoDB的基本接口操作
Node.js 中MongoDB的基本接口操作 连接数据库 安装mongodb模块 导入mongodb模块 调用connect方法 文档的增删改查操作 插入文档 方法: db.collection(& ...
- 在node.js中使用COOKIE
node.js中如何向客户端发送COOKIE呢?有如下两个方案: 一.使用response.writeHead,代码示例: //设置过期时间为一分钟 var today = new Date(); v ...
- 初步揭秘node.js中的事件
当你学习node.js的时候,Events是一个非常重要的需要理解的事情.非常多的Node对象触发事件,你能在文档API中找到很多例子.但是关于如何写自己的事件和监听,你可能还不太清楚.如果你不了解, ...
- Node.js权威指南 (10) - Node.js中的错误处理与断言处理
10.1 使用domain模块处理错误 / 272 10.1.1 domain模块概述 / 272 10.1.2 创建并使用Domain对象 / 274 10.1.3 隐式绑定与显式绑定 / 276 ...
- Node.js中的URL
Node.js中的URL 什么是URL URL是Uniform Location Resource的缩写,翻译为"统一资源定位符",也就是描述资源位置的固定表示方法.被URL描述的 ...
随机推荐
- Discuz使用tools修复数据文件后,访问URL多出/source/plugin/tools,导致文章栏目无法访问
今天我的婚嫁亲子网数据库出了点错误,于是就用dz官方的tool工具修复了以下,然后就发生了这个错误.. 本来频道页面的地址是:http://www.ifen8.com/article/ 结果自动跳转成 ...
- Android支付接入(三):电信爱游戏支付
原地址:http://blog.csdn.net/simdanfeg/article/details/9011977 注意事项: 1.电信要求必须先启动电信的闪屏界面 2.非网络游戏不允许有Inter ...
- linux Ubuntu安装后没有引导 解决方案
用EasyBCD添加ubuntu grub2引导,适用于12.04 及之前版本的ubuntu安装好easybcd后运行,之后看图
- Java将整个文件夹里的文本中的字符串替换成另外一个字符串(可用于项目复制,变成另一个项目)
import org.junit.Test; import java.io.*; /** * User: HYY * Date: 13-8-18 * Time: 下午8:11 * To change ...
- [mock]12月27日
一开始介绍项目,最后的反馈是,还是说得不清楚,需要再准备准备. 然后两道题,第一题是有个数组,有2*n个数字,从1~n.比如n=3的数组,{1,2,2,3,1,3}.然后两两相同的数字删除,每次删除得 ...
- Java Web开发 之小张老师总结GET和POST区别
get和post区别1.传输方式不同,get在request-line中传输(即在URL中传输).post在request-line及 request-body中传输(可认为隐藏传输)2.get传输长 ...
- c/c++中一些高级函数的使用
setvbuf 函数名: setvbuf 功 能: 把缓冲区与流相关 用 法: int setvbuf(FILE *stream, char *buf, int type, unsigned size ...
- WPF中通过代码定义模板
WPF中可以再XAML中定义模板,也可以通过C#代码定义模板,通过代码可能更清楚的看清其逻辑,而且代码的好处就是可以随时动态的去操作,而在XAML中定义的一般都是静态的. //控件呈现的显示内容1(这 ...
- 【原创】oracle的tpc-c测试及方法
大家好,很高兴来到博客园分享自己的所见所得.希望和大家多多交流,共同进步. 本文重点在于简介使用BenchmarkSQL对oracle进行tpcc的测试步骤,只是一个简单入门的过程. 开源测试工具:B ...
- 25.allegro中模块复用[原创]
一,Module reuse 1,打开原理图 ------------------- --------------------- ctrl+i过滤器 直选part ctrl+e 查看属性 查看: 是否 ...