最近应老大要求,对orm2进行再一步封装,所以记录下封装和使用心得(文中数据库:mysql)。

数据库连接

var orm = require("orm");

orm.connect("mysql://username:password@host/database", function (err, db) {
}

实际情况: 由于公司产品比较多,每个产品就会有一个数据库,所以我们做底层的必须连接各个产品的数据库。

所以解决方案:

1.配置文件中将所有数据库都配置好,如下:

server: {
mysql: {
product: {
host: '',
user: '',
password: '',
database: '',
port: 3306,
multipleStatements: true,
insecureAuth: true
},
ss: {
host: '',
user: '',
password: '',
database: '',
port: 3306,
multipleStatements: true,
insecureAuth: true
}
}
}

2.定义全局DB操作对象

global.dbs = {};

3.通过bluebird,underscore等包,类似循环连接数据库,并保存链接,下面为部分代码

var   _ = require('underscore'),
promise = require('bluebird'); /**
* 初始化数据库连接
* @param conn 数据库连接字符串
* @param dbServer 数据库服务器
* @param callback
*/
var toDB = function (conn, dbServer, callback) {
orm.connect(conn, function (err, db) {
//连接错误,则抛出异常
if (err) return comm.nextTick(err, callback);
//将数据库连接存入全局数据库连接对象中
dbs[dbServer] = db;
comm.nextTick(null, dbs, callback);
})
} //toDB函数promise化
var toDBPrms = promise.promisify(toDB); //?pool=true为开启连接池
var connections = _.map(config.server.mysql, function (connection, key) {
return toDBPrms('mysql://' + connection.user + ':' + connection.password + '@' + connection.host + '/' + connection.database + '?pool=true', key);
}); promise
.all(connections)
.then(function () {
console.log('数据库连接成功');
})
.catch(function (err) {
log.error({'数据库连接错误:': JSON.stringify(err)});
})

Model定义

官方的Model加载定义是在连接数据库的时候,如下:

orm.connect("mysql://username:password@host/database", function (err, db) {
if (err) throw err; var Person = db.define("person", {
name : String,
surname : String,
age : Number, // FLOAT
male : Boolean,
continent : [ "Europe", "America", "Asia", "Africa", "Australia", "Antartica" ], // ENUM type
photo : Buffer, // BLOB/BINARY
data : Object // JSON encoded
} .......

可是这样太冗余,对于大项目表又多的情况明显不合适,所以它还有一种加载定义Model方式,即Model作为单独的一个文件。

//定义表结构Model
module.exports = function (db, cb) {
//define方法的第一个参数为表名
db.define('xxx', {
innerid: { type: 'serial', key: true } , //主键
name: String,
identifying: String,
purpose_code: Number,
org_id: String,
position_x: Number,
position_y: Number,
isenabled: Number,
isdeleted: Number
});
return cb();
}

加载

 //db为数据库连接,相当于connection;  load第一个参数为Model的路径
db.load('XXX', function (err) {
//加载model错误,记录异常
if (err) {
return callback(err,xxx);
}
callback(null,xxx);
})

数据插入

Model.Create:

Person.create([
{
name: "John",
surname: "Doe",
age: 25,
male: true
},
{
name: "Liza",
surname: "Kollan",
age: 19,
male: false
}
], function (err, items) {
// err - description of the error or null
// items - array of inserted items
});

下面是我对其进行封装

/**
* 数据添加
* @param dbServer db服务器
* @param modelName model名字
* @param entity 插入数据(JSON格式)
* @param callback 返回结果
*/
daoComm.prototype.create = function (dbServer, modelName, entity, callback) {
//数据库连接,dbs里面存了各个产品的数据库,所以要获取对应的数据库连接
var db = dbs[dbServer];
//连接不存在,则抛错
if (!db) return comm.nextTick(comm.response(config.status.FAILURE, 'DB Connection IS NOT Exist'), callback);
//model没加载,则自动加载
if (!db.models[modelName]) {
//加载model
loadModel(db, dbServer, modelName, function (err) {
//如有错误,则返回
if (err) return comm.nextTick(err, callback);
//执行插入数据操作
add(db.models[modelName], modelName, entity, callback);
});
} else {
//执行插入数据操作
add(db.models[modelName], modelName, entity, callback);
}
}
/**
* 实际插入数据操作
* @param model model对象
* @param modelName 表名
* @param entity 插入数据
* @param callback 返回值
*/
function add(model, modelName, entity, callback) {
model.create(entity, function (err, resItems) {
if (err) return comm.nextTick(comm.response(config.status.FAILURE, modelName + ' Data Insert ERROR : ' + err), callback);
return comm.nextTick(null, comm.response(config.status.OK, resItems), callback);
})
}

数据查找

1.Model.get(第一个参数为id的值):

Person.get(4, function(err, person) {
console.log( person.fullName() );
})

2.Model.find([ conditions ] [, options ] [, limit ] [, order ] [, cb ])

Person.find({ name: "John", surname: "Doe" }, 3, function (err, people) {
// finds people with name='John' AND surname='Doe' and returns the first 3
});

3.Model.find链式写法

Person.find({ surname: "Doe" }).limit(3).offset(2).only("name", "surname").run(function (err, people) {
// finds people with surname='Doe', skips first 2 and limits to 3 elements,
// returning only 'name' and 'surname' properties
});

这里数据查找的方法比较多。要再次封装的时候头很大,封装链式太麻烦;封装第二个find方法,参数又不固定,如何只取当中几列啥的文档也没写清楚,原来limit,offset,only这些都能放在options里面,所以最后还是看底层源码解决的...

/**
* 数据查询
* @param dbServer db服务器
* @param modelName model名字
* @param conditions 查询条件(json)
* @param options 查询选项(json) {"offset":2,"limit":3,"only":["innerid","name"]} offset跳过条数 limit查询条数 only:取的列
* @param order 排列(object) 例:["name","Z"],按name倒序
* 备注 传入的conditions,options,order都必须json.parse一下
* @param callback 返回结果
*/
daoComm.prototype.find = function (dbServer, modelName, conditions, options, order, callback) {
//数据库连接
var db = dbs[dbServer];
//连接不存在,则抛错
if (!db) return comm.nextTick(comm.response(config.status.FAILURE, 'DB Connection IS NOT Exist'), callback);
//参数没有时初始化
if (!conditions) conditions = {};
if (!options) options = {};
if (!order) order = [];
//判断参数类型
if (typeof conditions != "object" || typeof options != "object" || typeof order != "object") {
return comm.nextTick(comm.response(config.status.FAILURE, 'TypeOf Param Error'), callback);
}
//model没加载,则自动加载
if (!db.models[modelName]) {
//加载model
loadModel(db, dbServer, modelName, function (err) {
//如有错误,则返回
if (err) return comm.nextTick(err, callback);
//执行查询数据操作
query(db.models[modelName], modelName, conditions, options, order, callback);
});
} else {
//执行查询数据操作
query(db.models[modelName], modelName, conditions, options, order, callback);
}
}
/**
* 实际查询数据操作
* @param model model对象
* @param modelName 表名
* @param conditions 查询条件
* @param options 查询选项
* @param order 排序
* @param callback
*/
function query(model, modelName, conditions, options, order, callback) {
model.find(conditions, options, order, function (err, data) {
if (err) return comm.nextTick(comm.response(config.status.FAILURE, modelName + ' Data Query ERROR : ' + err), callback);
return comm.nextTick(null, comm.response(config.status.OK, data), callback);
})
}

可能这种封装方式也不好,希望各位大大能给点建议看看还有木有更好的~~

数据删除

简单来说就是先查后删Model.find().remove();

Person.find({ surname: "Doe" }).remove(function (err) {
// Does gone..
});

数据更新

稍微有点麻烦,先查后遍历修改再保存 Model.find().each().save()

Person.find({ surname: "Doe" }).each(function (person) {
person.surname = "Dean";
}).save(function (err) {
// done!
});

取COUNT

Model.find().count()

Person.find({ surname: "Doe" }).count(function (err, people) {
// people = number of people with surname="Doe"
});

判断是否存在

Model.Exists([ conditions, ] cb)

Person.exists({ surname: "Doe" }, function (err, exists) {
console.log("We %s Does in our db", exists ? "have" : "don't have");
});

函数使用

Model.aggregate

Person.aggregate({ surname: "Doe" }).min("age").max("age").get(function (err, min, max) {
console.log("The youngest Doe guy has %d years, while the oldest is %d", min, max);
}); //The same as "select avg(weight), age from person where country='someCountry' group by age;"
Person.aggregate(["age"], { country: "someCountry" }).avg("weight").groupBy("age").get(function (err, stats) {
// stats is an Array, each item should have 'age' and 'avg_weight'
});

官方给出这几个函数:min、max、avg、sum、count

groupBy只能在Model.aggregate中使用

多表查询

hasOne()、hasMany(),很遗憾,限制太多(具体哪些,自己去踩坑吧)俺不搞!

多表查询替代方法 && 直接执行SQL

db.driver.execQuery

db.driver.execQuery("SELECT id, email FROM user", function (err, data) { ... })

// 上面是直接执行SQL,下面是类似参数化。   列用??表示, 列值用?表示。   建议使用下面这种
db.driver.execQuery(
"SELECT user.??, user.?? FROM user WHERE user.?? LIKE ? AND user.?? > ?",
['id', 'name', 'name', 'john', 'id', 55],
function (err, data) { ... }
)

Sequelize也接触过一点点,不过感觉没ORM2上手那么容易,而且感觉ORM2的坑比较少 哈哈。

由于比较菜,各位在看本人自己封装的代码时吐槽时嘴下留情。 希望也能给本人多些建议,tks~

node-orm2的更多相关文章

  1. [转]为什么我要用 Node.js? 案例逐一介绍

    原文地址:http://blog.jobbole.com/53736/ 介绍 JavaScript 高涨的人气带来了很多变化,以至于如今使用其进行网络开发的形式也变得截然不同了.就如同在浏览器中一样, ...

  2. 【转】为什么我要用 Node.js? 案例逐一介绍

    原文转自:http://blog.jobbole.com/53736/ 介绍 JavaScript 高涨的人气带来了很多变化,以至于如今使用其进行网络开发的形式也变得截然不同了.就如同在浏览器中一样, ...

  3. Node.js(转) -- 临时来说还看不懂!

    转自:http://blog.jobbole.com/53736/ 本文由 伯乐在线 - Lellansin 翻译.未经许可,禁止转载!英文出处:toptal.欢迎加入翻译组. 介绍 JavaScri ...

  4. 为什么我要用 Node.js? 案例逐一介绍

    介绍 JavaScript 高涨的人气带来了很多变化,以至于如今使用其进行网络开发的形式也变得截然不同了.就如同在浏览器中一样,现在我们也可以在服务器上运行 JavaScript ,从前端跨越到后端, ...

  5. 为什么应使用 Node.js

    为什么应使用 Node.js JavaScript 高涨的人气带来了很多变化,以至于如今使用其进行网络开发的形式也变得截然不同了.就如同在浏览器中一样,现在我们也可以在服务器上运行 JavaScrip ...

  6. Node.js具体解析

    介绍 JavaScript 高涨的人气带来了非常多变化.以至于现在使用其进行网络开发的形式也变得截然不同了.就如同在浏览器中一样,现在我们也能够在server上执行 JavaScript ,从前端跨越 ...

  7. babeljs源码

    babel.min.js!function(e,t){"object"==typeof exports&&"object"==typeof mo ...

  8. node中通过orm2链接mysql的一个坑

    代码是orm上的例子,出现如下错误: ORMError: Connection protocol not supported - have you installed the database dri ...

  9. Node.js中的ORM

    ORM2是一款基于Node.js实现的ORM框架,名字相当的霸气,算是同类框架中非常出色的一款,具体介绍请猛击:https://github.com/dresende/node-orm2 刚接触Nod ...

  10. koa中间件系统原理及koa+orm2实践。

    koa是由 Express 原班人马打造的新的web框架.套用其官方的说法:Koa 应用是一个包含一系列中间件 generator 函数的对象. 这些中间件函数基于 request 请求以一个类似于栈 ...

随机推荐

  1. linux删除N天前的文件

    列出一天之前的文件的属性用下面的命令:find . ! -mtime -1 -print |xargs ls -lfind . -mtime +1 要删除文件的话用下面的命令:find . ! -mt ...

  2. AFNetWorking出现code=-1016错误解决办法

    报错类似: 2015-12-09 15:58:03.062 Carloans[14328:2300485] Error Domain=com.alamofire.error.serialization ...

  3. ios sqlite 简单使用

    // // ViewController.m // sqlitedemo // // Created by lam_TT on 15-4-11. // Copyright (c) 2015年 lam_ ...

  4. 【bzoj3123】[Sdoi2013]森林 倍增LCA+主席树+启发式合并

    题目描述 输入 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M,T,分别表示节点数.初始边数.操作数.第三行包含N个非负 ...

  5. 最小的图灵完备语言——BrainFuck

    最小的图灵完备语言--BrainFuck 图灵完备性(Turing completness) 在可计算性理论(computability theory)中,图灵等价指的是:对于两个计算机A和B,如果A ...

  6. BZOJ 4516 [Sdoi2016]生成魔咒 ——后缀自动机

    本质不同的字串,考虑SA的做法,比较弱,貌似不会. 好吧,只好用SAM了,由于后缀自动机的状态最简的性质, 所有不同的字串就是∑l[i]-l[fa[i]], 然后后缀自动机是可以在线的,然后维护一下就 ...

  7. 【bzoj1710】[Usaco2007 Open]Cheappal 廉价回文

    [bzoj1710][Usaco2007 Open]Cheappal 廉价回文 Description 为了跟踪所有的牛,农夫JOHN在农场上装了一套自动系统. 他给了每一个头牛一个电子牌号 当牛走过 ...

  8. cf21D Traveling Graph

    You are given undirected weighted graph. Find the length of the shortest cycle which starts from the ...

  9. javascript 日期处理类库 moment.js

  10. Laravel 静态资源管理

    <link rel="stylesheet" href="{{ asset('bootstrap/css/bootstrap.min.css') }}" ...