《MongoDB入门教程》第30篇 唯一索引
创始人
2025-06-01 17:18:19
0

本文将会介绍 MongoDB 唯一索引,它可以用于确保文档字段值的唯一性。

唯一索引

很多时候我们需要确保文档中某个字段值的唯一性,例如 email 或者 username。唯一索引(unique index)可以帮助我们实现这种业务规则。实际上,MongoDB 使用唯一索引确保主键 _id 的唯一性。

创建唯一索引的方法和普通索引相同,只需要额外指定 {unique: true} 选项:

db.collection.createIndex({ field: 1}, {unique: true});

索引示例

首先,创建一个新的集合 users,插入一些文档:

db.users.insertMany([{ name: "张三", dob: "1990-01-01", email: "zhangsan@test.com"},{ name: "李四", dob: "1992-06-30", email: "lisi@test.com"}
]);

然后,我们基于 email 字段创建一个唯一索引:

db.users.createIndex({email:1},{unique:true});

接下来我们尝试插入一个已经存在的文档:

db.users.insertOne({ name: "张叁", dob: "1995-03-12", email: "zhangsan@test.com"}
);

此时,MongoDB 将会返回以下错误:

MongoServerError: E11000 duplicate key error collection: book.users index: email_1 dup key: { email: 'zhangsan@test.com' }

下面我们演示一下集合已经存在重复数据时如何创建唯一索引。首先删除并重建集合 users:

db.users.drop()db.users.insertMany([{ name: "张三", dob: "1990-01-01", email: "zhangsan@test.com"},{ name: "张叁", dob: "1995-03-12", email: "zhangsan@test.com"},{ name: "李四", dob: "1992-06-30", email: "lisi@test.com"}
]);

然后,基于 email 字段创建一个唯一索引:

db.users.createIndex({email: 1},{unique:true})MongoServerError: Index build failed: 95f78956-d5d0-4882-bfe0-2d856df18c61: Collection book.users ( 6da472db-2884-4608-98b6-95a003b4f29c ) :: caused by :: E11000 duplicate key error collection: mflix.users index: email_1 dup key: { email: zhangsan@test.com }

以上错误的原因在于 email 中存在重复的记录。

通常,我们会在插入数据之前创建唯一索引,可以从头开始确保数据的唯一性。如果基于已有数据创建唯一索引,可能会由于重复数据导致索引创建失败。为此,我们需要删除重复的数据之后再创建索引。

例如,我们可以首先删除重复的 user:

db.users.deleteOne({ name: "张叁", dob: "1995-03-12", email: "zhangsan@test.com"});{ acknowledged: true, deletedCount: 1 }

然后基于 email 字段创建唯一索引:

db.users.createIndex({email: 1},{unique:true})email_1

复合唯一索引

基于多个字段创建的唯一索引就是复合唯一索引(unique compound index)。复合唯一索引可以确保多个字段值的组合唯一。例如,基于字段 field1 和 field2 创建复合唯一索引,以下数据具有唯一性:

field1field2组合
11(1,1)
12(1,2)
21(2,1)
22(2,2)

以下数据存在重复值:

field1field2组合
11(1,1)
11(1,1) 重复
21(2,1)
21(2,1) 重复

接下来我们看一个示例,首先创建一个集合 locations。

db.locations.insertOne({address: "北京市丰台区莲花池东路118号北京西站",latitude: 39.894793,longitude: 116.321592
})

然后,基于 latitude 和 longitude 创建一个复合唯一索引:

db.locations.createIndex({latitude: 1,longitude: 1
},{ unique: true });'latitude_1_longitude_1'

插入一个经度相同的地点:

db.locations.insertOne({address: "北京市海淀区复兴路甲9号中华世纪坛",lat: 39.910577,long: 116.321592
})

操作执行成功,因为复合索引 latitude_1_longitude_1 校验的是经度和纬度的组合值。

最后,插入一个经纬度已经存在的地点:

db.locations.insertOne({address: "北京市丰台区莲花池东路118号北京西站",latitude: 39.894793,longitude: 116.321592
})

此时,MongoDB 返回重复键错误:

MongoServerError: E11000 duplicate key error collection: book.locations index: latitude_1_longitude_1 dup key: { lat: 39.894793, long: 116.321592 }

相关内容

热门资讯

净利跌超80%、销售费用砍超7... 本报(chinatimes.net.cn)记者于娜 见习记者 赵文娟 北京报道 近日,葵花药业发布的...
最新通胀数据“达标”,欧洲央行... 转自:中证金牛座 北京时间7月17日下午,欧洲统计局公布欧元区6月CPI终值数据:欧元区6月CPI同...
瑞典编程初创公司Lovable... 瑞典AI编程初创公司Lovable日前完成2亿美元(约合 143.6亿人民币)的A轮融资后,成为欧洲...
原创 银... 近些年,国内居民存款热情越来越高。数据显示,今年上半年,住户存款增加10.77万亿元,平均每个月新增...
国内商品期市早盘收盘涨多跌少 ... 据Choice数据,7月18日,国内商品期市早盘收盘主力合约涨多跌少,截至11:30,焦煤涨超2%,...
商务部:因时因势出台有针对性措... 商务部部长王文涛7月18日在国新办举行的“高质量完成‘十四五’规划”系列主题新闻发布会上表示,展望“...
美企涌向链博会,从中可以读出三... 来源:国是直通车 第三届中国国际供应链促进博览会现场。(贸促会供图) 中新社记者 尹倩芸 此间举行...
上交所:推动科创板“1+6”政... 证券时报记者 张淑贤 上交所近期先后在上海、杭州、南京、合肥等长三角区域重点城市联合地方政府相关部门...
经济学家:AI投资崩盘隐忧,泡... 7 月 19 日消息,科技媒体 Tom's Hardware 昨日(7 月 18 日)发布博文,报道...
开展产业链上下游整合 长鸿高科... 7月18日晚间,长鸿高科发布发行股份、可转债及支付现金购买资产并募集配套资金暨关联交易预案。同时,公...
国金基金管理有限公司旗下全部基... 本公司董事会及董事保证基金季度报告所载资料不存在虚假记载、误导性陈述或重大遗漏,并对其内容的真实性、...
宁波银行中标结果:浙江博宏工程... 证券之星消息,根据天眼查APP信息整理,7月18日公布的《浙江博宏工程管理咨询有限公司关于浙江钱海市...
深度 | 内窥镜医疗器械行业分... 1. 全球内窥镜市场概览 1.1 市场规模与增长趋势 全球内窥镜市场近年来呈现稳健的增长态势,并预计...
苹果全球前200家供应商超八成... 7月16日-7月20日,第三届中国国际供应链促进博览会在北京举办。今年,苹果公司携手三家中国供应商⸺...
金评天下|稳定币掀起蝴蝶效应 ... 金融投资报评论员 刘柯 美国国会众议院17日经表决通过三项有关稳定币等加密数字货币的法案。其中,《...
高盛预计黄金明年可达四千美元?... 最近几年,黄金的价格可谓是水涨船高,好不容易最近一段时间黄金价格出现了回调,就在这样的情况下,世界第...
原创 没... 据央视新闻报道,特朗普宣称若俄乌50天内未达成和平协议,美国将对俄罗斯实施100%关税。此消息瞬间搅...
男子用“AI换脸”登录23人账... 近日,南京市玄武区人民检察院办理了一起“AI换脸”诈骗案,嫌疑人符某利用非法获取的195万多条公民个...
工信部:实施新一轮钢铁、有色金... 21世纪经济报道记者周潇枭 北京报道7月18日,国新办举行新闻发布会,邀请工业和信息化部总工程师谢少...