MEAN開發棧中使用MongoDB的時候,與之配對的ORM最好的選擇就是Mongoose了。本文就和大家一起探討一下如何使用Mongoose來實現MongoDB的增刪改查。
為了能使文中的例子更加生動,我們會實現一個對於用戶的增刪改查的RESTful API。
Mongoose簡介
mongoose是一個nodejs下,專門基於no-sql數據庫mongodb的ORM框架。我們可以使用mongoose輕松實現對於mongodb的操作。要是用mongoose首先要在項目中添加這個框架:
$ npm install mongoose --save
注意,這里假設你已經安裝了MongoDB。如果還沒有,那么請參考這里下載,參考這里安裝,並創建第一個數據庫。
在項目中引用mongoose:
var mongoose = require('mongoose');
連接到已有數據庫:
var url = 'mongodb://localhost/yourappdatabase';
mongoose.createConnection(url);
連接到數據庫后就要處理里面的數據了。
定義一個Model
在處理增刪改查以前,我們先看看mongoose的Model。這些Model就代表了MongoDB庫里面的Document(MongoDB術語,相當於Sql數據庫的表),以后的增刪改查都要通過這個Model實現。
mongoose的Schema
是用來定義document的屬性的。Schema
中也可以定義Methods
。
首先定義一個model:
var mongoose = require('mongoose');
var Schema = mongoose.Schema;
var userSchema = new Schema({
name: String,
username: { type: String, required: true, unique: true },
password: { type: String, required: true },
admin: Boolean,
location: String,
meta: {
age: Number,
website: String
},
created_at: Date,
updated_at: Date
});
var User = mongoose.model('User', userSchema);
module.exports = User;
首先使用Schema
定義一個userSchema
,作為之后model里包含的字段和字段的類型使用。如果schema里包含了其他的對象,那么就把這個對象定義在meta屬性內。
在mongoose的Schema里給字段指定類型時可用的類型有:
- String
- Number
- Date
- Buffer
- Boolean
- Mixed
- ObjectId
- Array
之后用mongoose.model
類創建Model。我們還可以在這里做更多,比如定義一個給用戶密碼加密的方法(稍后講到)。
自定義方法
var mongoose = require('mongoose');
var Schema = mongoose.Schema;
var userSchema = new Schema({
name: String,
username: { type: String, required: true, unique: true },
password: { type: String, required: true },
admin: Boolean,
location: String,
meta: {
age: Number,
website: String
},
created_at: Date,
updated_at: Date
});
// 在Schema里添加自定義方法
userSchema.methods.capitalizeName = function () {
this.name = this.name.toUpperCase();
return this.name;
};
var User = mongoose.model('User', userSchema);
module.exports = User;
使用舉例
下面就看看我們自定義的model和方法如何使用:
app.post('/new', function (req, res) {
var newGuy = new User({
name: req.body.name,
username: req.body.username,
password: req.body.password // 千萬別用這種密碼
});
newGuy.capitalizeName(function (err, name) {
if (err) {
res.send({error: err});
return;
}
console.log(`name is ${name}`);
});
newGuy.save(function (err) {
if (err) {
res.send({error: err});
} else {
res.send({message: 'done', user: newGuy});
}
這是在一個Http POST請求里獲取用戶的name
, username
, password
三個值。之后使用User
來創建一個新的用戶newGuy
,調用方法newGuy.capitalizeName
,注意這個犯法里有回調。最后使用save
方法保存用戶數據。由於在定義User
這個model的時候指定username
是唯一的,所以多次插入操作只會保存一條數據用戶名相同的數據。
在保存以前調用的方法
我們在model里也定義了created_at
等關於時間的屬性,這樣就知道什么時候doc被創建。我們使用Schema
的pre
方法來保證保存以前可以調用執行一段特定的代碼。
// 保存以前執行的代碼
userSchema.pre('save', function (next) {
var currentDate = new Date();
this.updated_at = currentDate;
if (!this.created_at) {
this.created_at = currentDate;
}
next();
});
現在每次save都會執行這段代碼,給model的created_at
和updated_at
設定時間。這也是一個hash密碼的地方。畢竟直接保存明文密碼太不應該了。
Create
用前面定義好的model:User
然后new一個新的變量。調用內置的save
方法就會創建一個新的user。
// create a new user
var newUser = User({
name: 'Peter Quill',
username: 'starlord55',
password: 'password',
admin: true
});
// save the user
newUser.save(function(err) {
if (err) throw err;
console.log('User created!');
});
讀取
讀取就是根據一定的條件把數據庫里的user讀出來。由這個條件可以決定我們是讀一個,幾個還是全部user。
讀取全部
app.get('/all', function (req, res) {
User.find({}, function (err, users) {
if (err) {
res.send({message: 'error'});
return;
}
res.send({message: 'done', data: users});
});
});
請求結果:
{
"message": "done",
"data": [
{
"_id": "576bfe34137a575cf8c854cc",
"name": "sue",
"age": 26,
"status": "A"
},
{
"_id": "576df2960f0a5baabce7bc16",
"name": "JACK",
"username": "uncle_charlie",
"password": "123456",
"__v": 0
}
]
}
查找一個
app.get('/some/:name', function (req, res) {
var name = req.params.name;
User.find({name: name.toUpperCase()}, function (err, user) {
if (err) {
res.send({message: 'error'});
return;
}
res.send({message: 'done', data: user});
});
});
使用User
的name
屬性的唯一約束查找一個user。因為用戶的name在數據庫中是大寫的,要匹配到用戶需要把傳過來的參數全部大寫之后使用。
用ID查找
app.get('/some/:name/:id', function (req, res) {
var name = req.params.name;
var userId = req.params.id;
if (name === 'none') {
User.findById(userId, function (err, user) {
if (err) {
res.send({message: 'error'});
return;
}
res.send({message: 'done', data: user});
});
} else {
// 略
}
});
查詢特定user
除了下面給出的方法以外,你可以可以使用MongoDB本身的查詢方法。
app.get('/find', function (req, res) {
var lastDay = new Date();
lastDay.setDate(lastDay.getDate() - 1);
User.find({admin: false}).where('created_at').gt(lastDay).exec(function (err, users) {
if (err) {
res.send({message: 'error'});
return;
}
res.send({message: 'done', data: users});
});
});
找到昨天添加的非admin用戶。
更新
我們接下來找幾個用戶出來,修改他們的某些屬性並再次存入數據庫。
app.post('/update/:name', function (req, res) {
var name = req.params.name.toUpperCase();
User.find({ name: name.toUpperCase() }, function (err, user) {
if (err) {
res.send({ message: 'error' });
return;
}
var aUser = user[0];
// 更改屬性
// 如果是admin的全部撤銷,不是admin則指定為admin
aUser.admin = req.body.admin;
// 定位為Beijing
aUser.location = req.body.location;
// save
aUser.save(function (err, data) {
if (err) {
res.send({ message: 'error' });
return;
}
console.log(`done ${data}`);
res.send({message: 'done', data: aUser});
});
});
});
查找並更新
User.findOneAndUpdate({name: name}, {username: 'mr_charlie'}, function (err, user) {
if (err) {
res.send({ message: 'error' });
return;
}
res.send({message: 'done', data: user});
});
還可以使用一個類似的方法findByIdAndUpdate
,各位可以在這里查看詳細的介紹。
刪除
獲取一個用戶,然后刪除。
app.get('/delete/:name', function (req, res) {
var un = req.params.name.toUpperCase();
User.find({name: un}, function (err, user) {
if (err) {
res.send({ message: 'error'});
return;
}
if (user.length <= 0) {
res.send({ message: 'error', 'data': 'no such user'});
return;
}
user[0].remove(function(err, data) {
if (err) {
res.send({ message: 'error' });
return;
}
console.log(`data to remove ${data}`);
res.send({messge: 'done', data: data});
});
});
});
用戶名會在get的路徑中傳入:http://localhost:4100/delete/sue
。之后使用這個用戶名找到用戶名稱為SUE的全部用戶(這時是個數組),如果這個用戶數組大於0,則刪除數組中的第一個model。最后,把刪除的用戶數據通過RESTful API發送出去。
類似的方法還有:findOneAndRemove
、findByIdAndRemove
。
// find the user with id 4
User.findOneAndRemove({ username: 'starlord55' }, function(err) {
if (err) throw err;
// we have deleted the user
console.log('User deleted!');
});
// find the user with id 4
User.findByIdAndRemove(4, function(err) {
if (err) throw err;
// we have deleted the user
console.log('User deleted!');
});
處理回調
前部分的內容都是關於如何使用mongoose處理數據的常用的增刪改查的方法。在其中涉及到了很多的回調。比如查找之后更新這一部分,首先要查找,在回調中得知成功之后調用更新,更新回調顯示成功之后把數據發回客戶端。在以上步驟中任意一步出錯則發送錯誤數據。
User.find({ name: name.toUpperCase() }, function (err, user) {
// 1. 查找以后的回調
if (err) {
res.send({ message: 'error' });
return;
}
// 修改數據
// save
aUser.save(function (err, data) {
// 更新的回調
if (err) {
res.send({ message: 'error' });
return;
}
// 更新成功
console.log(`done ${data}`);
res.send({ message: 'done', data: aUser });
});
});
查找並更新的兩個回調。
下面就來講解如何消除回調。這樣的神器叫做bluebird。bluebird使用了一種叫做Promise
的工具,可以通過then
方法依次的分解回調的代碼。
安裝bluebird。
npm install bluebird
在引入的mongoose里設置這個bluebird。
app.post('/asyncupdate/:name', function (req, res) {
var name = req.params.name.toUpperCase();
User.find({ name: name.toUpperCase() }).exec().then(function (users) {
if (users.length <= 0) {
throw new Error('no such user');
}
var aUser = users[0];
// 更改屬性
// 如果是admin的全部撤銷,不是admin則指定為admin
aUser.admin = req.body.admin;
// 定位為Beijing
aUser.location = req.body.location;
return aUser.save();
}).then(function (user) {
console.log('updated user: ' + user.name);
res.send({message: 'done', data: user});
}).catch(function (err) {
console.log('err ' + err);
res.send({ message: 'error' });
});
});
和上面的使用回調的查找-更新比較一下,使用Promise以后代碼不僅容易讀,而且以后也會更加容易維護。當然,Promise能做的絕對不僅是這樣寫。有興趣的同學可以到bluebird官網查看他們的文檔。
最后
Mongoose是一個MongoDB下非常好用的ORM庫,而且簡單易學。是開發的好幫手。另外還有bluebird加成。處理mongodb的時候就更加的得心應手了。