Mongoose輕松搞定MongoDB,不要回調!


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被創建。我們使用Schemapre方法來保證保存以前可以調用執行一段特定的代碼。

// 保存以前執行的代碼
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_atupdated_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});
    });
});

使用Username屬性的唯一約束查找一個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發送出去。

類似的方法還有:findOneAndRemovefindByIdAndRemove

// 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的時候就更加的得心應手了。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM