<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
前些天遇到一個需求,不復雜,用 SQL 表現的話,大約如此:
SELECT * FROM db1 LEFT JOIN db2 ON db1.a = db2.b WHERE db1.userId='$me' AND db2.status=1
沒想到搜了半天,我廠的程式碼倉庫裡沒有這種用法,各種教學也多半隻針對合併查詢(即只篩選 db1
,沒有 db2
的條件)。所以最後只好讀檔案+程式碼嘗試,終於找到答案,記錄一下。
$lookup
ObjectId
,就很簡單:// 假設下面兩個表 db1 和 db2 export const Db1Schema = new mongoose.Schema( { userId: { type: String, index: true }, couponId: { type: ObjectId, ref: Db2Schema }, }, { versionKey: false, timestamps: true } ); export const Db2Schema = new mongoose.Schema( { status: { type: Boolean, default: 0 }, }, { versionKey: false, timestamps: true } ); // 那麼只要 db1Model.aggregate([ { $lookup: { from: 'db2', // 目標表 localField: 'couponId', // 本地欄位 foreignField: '_id', // 對應的目標欄位 as: 'source', }, { $match: [ /* 各種條件 */ ], }, ]);
但是我們沒有用 ObjectId
,而是用 string
作為外來鍵,所以無法直接用上面的聯查。必須在 pipeline
裡手動轉換、聯合。此時,當前表(db1
)的欄位不能直接使用,要配合 let
,然後加上 $$
字首;連表(db2
)直接加 $
字首即可。
最終程式碼如下:
mongo.ts
// 每次必有的條件,當前表的欄位用 `$$`,連表的欄位用 `$` const filter = [{ $eq: ['$$userId', userId] }, { $eq: ['$isDeleted', false] }]; if (status === Expired) { dateOp = '$lte'; } else if (status === Normal) { dateOp = '$gte'; filter.push({ $in: ['$$status', [Normal, Shared]] }); } else { dateOp = '$gte'; filter.push({ $eq: ['$$status', status] }); } const results = await myModel.aggregate([ { $lookup: { from: 'coupons', // 當前表欄位必須 `let` 之後才能用 let: { couponId: '$couponId', userId: '$userId', status: '$status' }, // 在 pipeline 裡完成篩選 pipeline: [ { $match: { $expr: { // `$toString` 是內建方法,可以把 `ObjectId` 轉換成 `string` $and: [{ $eq: [{ $toString: '$_id' }, '$$couponId'] }, ...filter, { [dateOp]: ['$endAt', new Date()] }], }, }, }, // 只要某些欄位,在這裡篩選 { $project: couponFields, }, ], as: 'source', }, }, { // 這種篩選相當 LEFT JOIN,所以需要去掉沒有連表內容的結果 $match: { source: { $ne: [] }, }, }, { // 為了一次查表出結果,要轉換一下輸出格式 $facet: { results: [{ $skip: size * (page - 1) }, { $limit: size }], count: [ { $count: 'count', }, ], }, }, ]);
同事告訴我,這樣做的效率不一定高。我覺得,考慮到實際場景,他說的可能沒錯,不過,早晚要邁出這樣的一步。而且,未來我們也應該慢慢把外來鍵改成 ObjectId
型別。
總結
到此這篇關於MongoDB中實現多表聯查的文章就介紹到這了,更多相關MongoDB多表聯查內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45