使用aggregate在MongoDB中查找重覆的數據記錄

来源:http://www.cnblogs.com/jaxu/archive/2016/01/19/5143707.html
-Advertisement-
Play Games

我們知道,MongoDB屬於文檔型資料庫,其存儲的文檔類型都是JSON對象。正是由於這一特性,我們在Node.js中會經常使用MongoDB進行數據的存取。但由於Node.js是非同步執行的,這就導致我們無法保證每一次的資料庫save操作都是原子型的。也就是說,如果客戶端連續兩次發起同一事件將數據.....


  我們知道,MongoDB屬於文檔型資料庫,其存儲的文檔類型都是JSON對象。正是由於這一特性,我們在Node.js中會經常使用MongoDB進行數據的存取。但由於Node.js是非同步執行的,這就導致我們無法保證每一次的資料庫save操作都是原子型的。也就是說,如果客戶端連續兩次發起同一事件將數據存入資料庫,很可能會導致數據被重覆保存。高併發的情況下,哪怕是你在代碼中已經做了非常嚴格的校驗,例如插入數據前判斷要保存的數據是否已經存在,但仍然有可能會出現數據被重覆保存的風險。因為在非同步執行中,你沒有辦法保證哪個線程先執行,哪個線程後執行,客戶端發起的所有請求並非按我們想象的都是順序執行的。一個較好的解決辦法是在Mongo資料庫的所有表中創建唯一索引。事實上,MongoDB預設會為所有表創建一個_id欄位的唯一索引(可以取消)。如果你想在Node.js中通過mongoose.schema來自動創建索引,可以參考下麵的代碼:

var mongoose = require('mongoose');
var Schema = mongoose.Schema;

var customerSchema = new mongoose.Schema({
    cname: String,
    cellPhone, String,
    sender: String,
    tag: String,
    behaviour: Number,
    createTime: {
        type: Date,
        default: Date.now
    },
    current:{
        type: Boolean,
        default: true
    }
}, {
    versionKey: false
});

customerSchema.index({cname:1,cellPhone:1,sender:1,tag:1,behaviour:1}, {unique: true});

module.exports = mongoose.model('customer', customerSchema);

  上面的model中我們定義了表customer的結構,並通過index()方法在欄位cname,cellPhone,sender,tag,behaviour上創建了唯一索引,這樣當包含這些欄位的重覆數據被插入時,資料庫會拋出異常。借用mongoose,如果資料庫表之前已經被創建並且程式正在運行中,當我們修改model並添加索引,然後重新啟動app,只要有對該model的訪問,mongoose會自動進行檢測並創建索引。當然,如果數據出現重覆,則索引創建會失敗。此時我們可以通過在創建索引時添加dropDups選項,讓資料庫自動將重覆的數據刪除,如:

customerSchema.index({cname:1,cellPhone:1,sender:1,tag:1,behaviour:1}, {unique: true, dropDups: true});

  不過據MongoDB的官方說明,自3.0以後的版本不再使用該選項,而且也並沒有提供替代的解決辦法。貌似官方不再提供創建索引時自動刪除重覆記錄的功能。那如何才能快速有效地找出重覆的記錄並且刪除呢?首先我們要找出這些記錄,然後通過remove()方法進行刪除。下麵的查詢語句可以找出給定欄位有重覆數據的記錄:

db.collection.aggregate([
  { $group: { 
    _id: { firstField: "$firstField", secondField: "$secondField" }, 
    uniqueIds: { $addToSet: "$_id" },
    count: { $sum: 1 } 
  }}, 
  { $match: { 
    count: { $gt: 1 } 
  }}
])

  替換_id屬性的值以指定你想要進行判斷的欄位。相應地,在Node.js中代碼如下:

var deferred = Q.defer();
var group = { firstField: "$firstField", secondField: "$secondField"};

model.aggregate().group({
    _id: group,
    uniqueIds: {$addToSet: '$_id'},
    count: {$sum: 1}
}).match({ count: {$gt: 1}}).exec(deferred.makeNodeResolver());

return deferred.promise;

  上述代碼使用了Q來替換函數執行中的回調。在Node.js的非同步編程中,使用Q來處理回調是個不錯的選擇。

  下麵是返回的結果:

/* 1 */
{
    "result" : [ 
        {
            "_id" : {
                "cellPhone" : "15827571111",
                "actId" : ObjectId("5694565fa50fea7705f01789")
            },
            "uniqueIds" : [ 
                ObjectId("569b5d03b3d206f709f97685"), 
                ObjectId("569b5d01b3d206f709f97684")
            ],
            "count" : 2.0000000000000000
        }, 
        {
            "_id" : {
                "cellPhone" : "18171282222",
                "actId" : ObjectId("566b0d8dc02f61ae18e68e48")
            },
            "uniqueIds" : [ 
                ObjectId("566d16e6cf86d12d1abcee8b"), 
                ObjectId("566d16e6cf86d12d1abcee8a")
            ],
            "count" : 2.0000000000000000
        }
    ],
    "ok" : 1.0000000000000000
}

  從結果中可以看到,一共有兩組數據相同的記錄,所以返回的result數組的長度為2。uniqueIds屬性為一個數組,其中存放了重覆記錄的_id欄位的值,通過該值我們可以使用remove()方法來查找並刪除對應的數據。

 

補充:Mongoose支持findOneAndUpdate(在MongoDB中對應的方法叫findAndModify),選項upsert=true表示當要要更新的數據不存在時會自動創建。該選項預設值為false。示例代碼如下:

var query = {'username':req.user.username};
req.newData.username = req.user.username;
MyModel.findOneAndUpdate(query, req.newData, {upsert:true}, function(err, doc){
    if (err) return res.send(500, { error: err });
    return res.send("succesfully saved");
});

  通過該方法我們可以將數據的唯一性校驗交給MongoDB來完成。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • mustache.js是一個簡單強大的Javascript模板引擎,使用它可以簡化在js代碼中的html編寫,壓縮後只有9KB,非常值得在項目中使用。本文總結它的使用方法和一些使用心得,內容不算很高深,純粹是入門內容,看看即可。不過要是你還沒有用過此類的javascript引擎庫,那麼本文還是值得你...
  • 官網地址:http://harvesthq.github.io/chosen/Chosen (v1.4.2)Chosen has a number of options and attributes that allow you to have full control of your select...
  • 最近閑著沒事,研究了一下Node.js技術。發現Node.js真的很強大,它配合socket.io竟然可以實現Web的socket通訊技術。所以我決定來做個Web版的鬥地主來好好玩玩這個WebSocket技術,呵呵!!!先來部署環境1.安裝Node.js根據自己的操作系統,去Node.js官網下載安...
  • http含義:http 200:-文件被正常的訪問http 302:臨時重定向HTTP錯誤列表HTTP 400 - 請求無效HTTP 401.1 - 未授權:登錄失敗HTTP 401.2 - 未授權:伺服器配置問題導致登錄失敗HTTP 401.3 - ACL 禁止訪問資源HTTP 401.4 - 未...
  • 如果JS代碼中設置元素的另一個CSS屬性font-family。這個屬性的獲取方式與color屬性略有不同,因為 font和family之間的連字元與JS中減法操作符相同,JS會把它解釋為減號。如果你像下邊這樣訪問名為 font-family 的屬性,會收到一條出錯信息:Element.style....
  • 寫了這麼久的CSS,但大部分前端er都沒有按照良好的CSS書寫規範來寫CSS代碼,這樣會影響代碼的閱讀體驗,這裡總結一個CSS書寫規範、CSS書寫順序供大家參考,這些是參考了國外一些文章以及我的個人經驗總結出來,我想對寫CSS的前端用戶來說是值得學習的。 CSS書寫順序 1.位置屬性(positi....
  • 在這時碰到了一個圓角邊框的問題,以前的代碼是在每個頁面寫一個固定結構的div,使用背景圖片來實現圓角邊框。代碼結構大致如下:.top_border{background:url(topborder.png);}.left_border{background:url(leftborder.png);}...
  • 好吧,因為很重要的事情,幾天沒寫筆記了。關於對象:||可以用來填充預設值,如:myApp.name || "無"&&可以用來避免錯誤,myApp.NameObj有某種情況不存在,那麼可以用myApp.NameObj &&myApp.NameObj.Name避免腳本錯誤所有通過對象字面量創建的對象都連...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...