關(guān)于MongoDB唯一索引(Unique)的那些事

寫在前面

關(guān)于什么是索引以及唯一索引這里就不做說明了,不清楚的可以自行谷歌或者百度搬俊。是什么引起我寫這篇文章呢紊扬,這來自于之前項目中的一個問題。

我們用的是MongoDB數(shù)據(jù)存儲用戶信息唉擂,用戶表中曾經(jīng)用戶注冊是通過手機號注冊的餐屎,所以很理所當(dāng)然的給手機號加上了唯一索引(Unique),這是沒有什么毛病玩祟。后期腹缩,我們需求改了。你也可以想到變成了既可以手機號注冊又可以郵箱注冊空扎,這個時候由于手機號加了Unique索引藏鹊,事實上這時候是會出現(xiàn)問題的。


func init() {
    phoneIndex := mgo.Index{
        Key:    []string{"phone"},
        Unique: true,
    }

    col := db.Collection(&User{})
    col.EnsureIndex(phoneIndex)
}

當(dāng)然這問題其實也容易想到转锈,當(dāng)用戶通過郵箱注冊此時手機號填空的時候盘寡,第一次沒什么問題,下個用戶再以這種方式注冊的時候便會提示建立在phone上的索引值重復(fù)撮慨,很正常嘛竿痰,因為插入了兩個空值,注意這里是空字符串砌溺,而不是null影涉。

于是我們嘗試修改,由于MongoDB是文檔型靈活的數(shù)據(jù)庫规伐,少插多插一兩個字段不受影響蟹倾,所以我們嘗試修改User實體Phone字段的入口,當(dāng)phone是空字符串的時候,不讓插入此字段党觅。于是笆包,我們便在phone字段中加入了omitempty標(biāo)簽(我們微服務(wù)用Go語言寫的)。下面展示User一部分內(nèi)容:

type User struct {
    Email         string `bson:"email"`
    Salt          string `bson:"salt"`
    Phone         string `bson:"phone,omitempty"`
    IDCard        string `bson:"idcard"`
    RealName      string `bson:"realname"`
    AuthStatus    int    `bson:"auth_status"`
}

可以看到phone字段后加了omitempty標(biāo)簽岔留,表示當(dāng)該字段為空的時候不插入。這還是會出現(xiàn)問題检柬,那么既然還是會出問題為什么會想到這么解決呢献联?這源于對Mysql的使用經(jīng)驗,習(xí)慣性的以為MongoDB和Mysql那樣何址,對null的值會不做其索引里逆。也就是說,在Mysql中用爪,若在多條記錄中Phone值為Null是被允許的原押。

上面那種做法,還是會報錯偎血,提示插入了重復(fù)的值诸衔,只不過這時不是空字符串盯漂,而是null。所以有時候就不要把Mysql那套拿來了笨农,Mysql是可以的就缆,但Mongo不行。mongo還是會對該條記錄索引谒亦,即使該字段為被插入竭宰。

我喜歡看官方文檔,下面給出MongoDB官方文檔說明:

If a document does not have a value for the indexed field in a unique index, the index will store a null value for this document. Because of the unique constraint, MongoDB will only permit one document that lacks the indexed field. If there is more than one document without a value for the indexed field or is missing the indexed field, the index build will fail with a duplicate key error.

其實已經(jīng)說得很清楚了份招,稍微會點英語應(yīng)該都能看懂切揭,下面還是給出翻譯版:

如果文檔沒有唯一索引中索引字段的值,則索引將為此文檔存儲null值锁摔。由于唯一約束廓旬,MongoDB只允許一個缺少索引字段的文檔。如果有多個文檔沒有索引字段的值或缺少索引字段鄙漏,則索引構(gòu)建將失敗并出現(xiàn)重復(fù)鍵錯誤嗤谚。

也就是說這個字段哪怕在文檔中沒有,那么該字段將會存null值怔蚌,該字段上也不能同時出現(xiàn)兩個null值巩步,這就是為什么上面那種做法還是行不通的原因,其實上面那種做法也打破了數(shù)據(jù)結(jié)構(gòu)桦踊,雖然手機號未填椅野,但數(shù)據(jù)庫中也不應(yīng)該缺少這個字段,盡管是非關(guān)系數(shù)據(jù)庫籍胯,畢竟還得考慮下業(yè)務(wù)設(shè)計竟闪。

解決方式

是不是就沒有解決方式了呢?當(dāng)然有杖狼,Mongo提供了Sparse Index炼蛤,被翻譯為稀疏索引。下面是創(chuàng)建稀疏索引的例子:

db.getCollection("test").createIndex( { "phone": 1 }, { sparse: true })

執(zhí)行上面的語句后蝶涩,不會去索引不存在phone字段的文檔理朋。也就是說存在才對其索引,那么此時和Unique索引結(jié)合起來就可以派上用場了绿聘。Unqiue是唯一嗽上,Sparse是存在才索引。所以熄攘,當(dāng)phone或email為空的時候我們可以不將其插入這是可以實現(xiàn)的兽愤。

db.getCollection("test").createIndex( { "phone": 1 }, { sparse: true,unique: true  } )

上面是是mongo shell語法,通常我們一般通過代碼中建立索引,修改如下(當(dāng)然User結(jié)構(gòu)體中Phone字段omitempty標(biāo)簽還是要有的):


func init() {
    phoneIndex := mgo.Index{
        Key:    []string{"phone"},
        Unique: true,
        Sparse: true,
    }

    col := db.Collection(&User{})
    col.EnsureIndex(phoneIndex)
}

但是這又正如我們前面說的那樣浅萧,打破了數(shù)據(jù)原有的數(shù)據(jù)結(jié)構(gòu)逐沙。哎,有得有得惯殊。當(dāng)然我們還可以從業(yè)務(wù)層面去解決酱吝,比如注冊時對其查詢等操作,當(dāng)然會耗一定性能土思,不管你是那空間換時間,還是拿時間換空間總得付出一個忆嗜,別做一個太貪心的人己儒。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市捆毫,隨后出現(xiàn)的幾起案子闪湾,更是在濱河造成了極大的恐慌,老刑警劉巖绩卤,帶你破解...
    沈念sama閱讀 206,378評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件途样,死亡現(xiàn)場離奇詭異,居然都是意外死亡濒憋,警方通過查閱死者的電腦和手機何暇,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來凛驮,“玉大人裆站,你說我怎么就攤上這事∏玻” “怎么了宏胯?”我有些...
    開封第一講書人閱讀 152,702評論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長本姥。 經(jīng)常有香客問我肩袍,道長,這世上最難降的妖魔是什么婚惫? 我笑而不...
    開封第一講書人閱讀 55,259評論 1 279
  • 正文 為了忘掉前任氛赐,我火速辦了婚禮,結(jié)果婚禮上辰妙,老公的妹妹穿的比我還像新娘鹰祸。我一直安慰自己,他們只是感情好密浑,可當(dāng)我...
    茶點故事閱讀 64,263評論 5 371
  • 文/花漫 我一把揭開白布蛙婴。 她就那樣靜靜地躺著,像睡著了一般尔破。 火紅的嫁衣襯著肌膚如雪街图。 梳的紋絲不亂的頭發(fā)上浇衬,一...
    開封第一講書人閱讀 49,036評論 1 285
  • 那天,我揣著相機與錄音餐济,去河邊找鬼耘擂。 笑死,一個胖子當(dāng)著我的面吹牛絮姆,可吹牛的內(nèi)容都是我干的醉冤。 我是一名探鬼主播,決...
    沈念sama閱讀 38,349評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼篙悯,長吁一口氣:“原來是場噩夢啊……” “哼蚁阳!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起鸽照,我...
    開封第一講書人閱讀 36,979評論 0 259
  • 序言:老撾萬榮一對情侶失蹤螺捐,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后矮燎,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體定血,經(jīng)...
    沈念sama閱讀 43,469評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,938評論 2 323
  • 正文 我和宋清朗相戀三年诞外,在試婚紗的時候發(fā)現(xiàn)自己被綠了澜沟。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,059評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡浅乔,死狀恐怖倔喂,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情靖苇,我是刑警寧澤席噩,帶...
    沈念sama閱讀 33,703評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站贤壁,受9級特大地震影響悼枢,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜脾拆,卻給世界環(huán)境...
    茶點故事閱讀 39,257評論 3 307
  • 文/蒙蒙 一馒索、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧名船,春花似錦绰上、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春百揭,著一層夾襖步出監(jiān)牢的瞬間爽哎,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工器一, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留课锌,地道東北人。 一個月前我還...
    沈念sama閱讀 45,501評論 2 354
  • 正文 我出身青樓祈秕,卻偏偏與公主長得像渺贤,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子请毛,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,792評論 2 345