Unique Indexes
項目一覧
一意のインデックス により、インデックス フィールドに重複する値が保存されなくなります。つまり、インデックス フィールドが一意であることを強制します。 MongoDB はデフォルトでは、コレクションの作成中に _idフィールドに一意のインデックスが作成されます。
注意
新しい内部形式
MongoDB 4.2 以降、featureCompatibilityVersion(fCV)が 4.2(またはそれ以降)の場合は、その前のバージョンとは互換性のない新しい内部形式がユニークインデックスに使用されます。新しい形式は、既存のユニークインデックスと新しく作成/再構築されたユニークインデックスの両方に適用されます。
MongoDB Atlas でホストされる配置用に、ユニークインデックスを UI に作成して管理できます。
ユニークインデックスの作成
ユニークインデックスを作成するには、unique
オプションを true
に設定した状態で、db.collection.createIndex()
メソッドを使用します。
db.collection.createIndex( <key and index type specification>, { unique: true } )
単一フィールドのユニークインデックス
たとえば、 members
コレクションのuser_id
フィールドに一意のインデックスを作成するには、 mongosh
で次の操作を使用します。
db.members.createIndex( { "user_id": 1 }, { unique: true } )
ユニーク複合インデックス
複合インデックスにもユニーク制約を適用できます。複合インデックスにこの制約を適用すると、MongoDB はインデックス キー値の組み合わせに一意であることを強制します。
たとえば、 members
コレクションのgroupNumber
、 lastname
、およびfirstname
フィールドに一意のインデックスを作成するには、 mongosh
で次の操作を使用します。
db.members.createIndex( { groupNumber: 1, lastname: 1, firstname: 1 }, { unique: true } )
作成されたインデックスでは、groupNumber
、lastname
、firstname
の値の組み合わせは必然的に一意になります。
別の例として、次のドキュメントを含むコレクションを考えてみましょう。
{ _id: 1, a: [ { loc: "A", qty: 5 }, { qty: 10 } ] }
次のとおり、ユニークな複合マルチキー インデックスを a.loc
と a.qty
に作成します。
db.collection.createIndex( { "a.loc": 1, "a.qty": 1 }, { unique: true } )
ユニークインデックスを使用すると、a.loc
と a.qty
の値の 組み合わせが必然的に一意になるため、次のドキュメントをコレクションに挿入できます。
db.collection.insertMany( [ { _id: 2, a: [ { loc: "A" }, { qty: 5 } ] }, { _id: 3, a: [ { loc: "A", qty: 10 } ] } ] )
動作
制限事項
MongoDBは、コレクションにユニーク制約に違反するデータが既に含まれている場合、指定されたインデックス フィールドにユニークインデックスを作成できません。
ハッシュされたインデックスにユニーク制約を指定することはできません。
レプリカセットとシャーディングされたクラスターでのユニークインデックスの構築
レプリカセットとシャーディングされたクラスターでローリング手順を使用してユニークインデックスを作成するには、その手順の実行中にコレクションへの書込みをすべて停止する必要があります。書込みを停止できない場合は、ローリング手順は使用せず、代わりに、次のいずれかの方法でコレクションにユニークインデックスを構築します。
レプリカセットのプライマリで
db.collection.createIndex()
を発行する。シャーディングされたクラスター向けに
mongos
に対してdb.collection.createIndex()
を発行する。
個別ドキュメント間でのユニーク制約
ユニーク制約は、コレクション内のドキュメントに個別に適用されます。具体的には、個別のドキュメントでインデックス キーの値が同じになることはありません。
この制約は個別のドキュメントに適用されるため、ユニークマルチキー インデックスの場合、ドキュメントに含まれる配列要素によって、そのドキュメントのインデックス キー値と別のドキュメントのインデックス キー値が重複していない限り、インデックス キー値の反復につながることがあります。この場合、反復されるインデックスエントリはインデックスに 1 度のみ挿入されます。
たとえば、次のドキュメントを含むコレクションで考えてみましょう。
{ _id: 1, a: [ { loc: "A", qty: 5 }, { qty: 10 } ] } { _id: 2, a: [ { loc: "A" }, { qty: 5 } ] } { _id: 3, a: [ { loc: "A", qty: 10 } ] }
次のとおり、ユニーク複合マルチキー インデックスを a.loc
と a.qty
に作成します。
db.collection.createIndex( { "a.loc": 1, "a.qty": 1 }, { unique: true } )
ユニークインデックスを使用すると、インデックスキー値が { "a.loc": "B", "a.qty": null }
であるドキュメントが他にない場合は、次のドキュメントをコレクションに挿入できます。
db.collection.insertOne( { _id: 4, a: [ { loc: "B" }, { loc: "B" } ] } )
ユニークな単一フィールド インデックス内でのドキュメント フィールドの欠落
ドキュメントの単一フィールドのユニークインデックスのインデックス フィールドに null
または欠損値がある場合、インデックスはそのドキュメントに null
値を保存します。単一フィールドのユニークインデックスはユニーク制約により、インデックスエントリに null
値を含むドキュメントを 1 つしか含めることができません。インデックスエントリに null
値があるドキュメントが複数ある場合、インデックス構築は重複キーエラーで失敗します。
x
にユニークな単一フィールド インデックスがあるコレクションを例にしましょう。
db.collection.createIndex( { "x": 1 }, { unique: true } )
ユニークインデックスを使用しているため、x
フィールドのないドキュメントがコレクションに既にない場合は、x
フィールドがなくてもドキュメントを挿入できます。
db.collection.insertOne( { y: 1 } )
ただし、x
フィールドのないドキュメントがコレクションに既に含まれている場合は、x
フィールドがないとドキュメントを挿入できません。
db.collection.insertOne( { z: 1 } )
x
フィールドの値に課されているユニーク制約に違反しているため、ドキュメントの挿入操作は失敗します。
WriteResult({ "nInserted" : 0, "writeError" : { "code" : 11000, "errmsg" : "E11000 duplicate key error index: test.collection.$a.b_1 dup key: { : null }" } })
ユニーク複合インデックス内でのドキュメント フィールドの欠落
ドキュメントの複合ユニークインデックスの 1 つ以上のインデックス フィールドにnull
または欠落値がある場合、インデックスは、ドキュメントのインデックスエントリ内の各 null
または欠落フィールドに null 値を保存します。複合ユニークインデックスではユニーク制約により、複合ユニークインデックスでインデックスエントリ内のインデックス フィールドに null
値を含めることができるドキュメントは 1 つに限定されます。インデックス フィールドに null
値があるインデックスエントリが複数ある場合、インデックス構築は重複キーエラーで失敗します。MongoDB は、各インデックスエントリが一意である限り、複合ユニークインデックスに欠落フィールドがあるドキュメントを複数許容します。
たとえば、コレクション students
の name
、age
、grade
の各フィールドに一ユニーク複合インデックスがあるとします。
db.students.createIndex( { "name": 1, "age": -1, "grade": 1 }, { unique: true } )
コレクションに既存の同一ドキュメントがない場合、ユニーク複合インデックスでは、grade
フィールドがすべて欠落している次のドキュメントを挿入することができます。
db.students.insertMany( { "name": "Meredith", "age": 12 }, { "name": "Olivia", "age": 11 }, { "name": "Benjamin" } )
ただし、インデックスキー(name
、age
、grade
の値)がコレクション内の別のドキュメントと同じであるドキュメントは挿入できません。
db.students.insertOne( { name: "Meredith", age: 12 } )
name
、age
、grade
の各フィールドの値に課されている一意という制約に違反しているため、ドキュメントの挿入操作は失敗します。
WriteResult({ "nInserted" : 0, "writeError" : { "code" : 11000, "errmsg" : "E11000 duplicate key error collection: test.students index: name_1_age_-1_grade_1 dup key: { name: "Meredith", age: 12, grade: null } } } )
また、一意であっても既存のインデックスエントリとインデックスキーを共有するドキュメントは挿入できません。
db.students.insertOne( { name: "Olivia", "age": 11, "favorite color": "red"} )
name
、age
、grade
の各フィールドの値に課されている一意という制約に違反しているため、ドキュメントの挿入操作は失敗します。
WriteResult({ "nInserted" : 0, "writeError" : { "code" : 11000, "errmsg" : "E11000 duplicate key error collection: test.students index: name_1_age_-1_grade_1 dup key: { name: "Olivia", age: 11, grade: null } } } )
ユニーク部分インデックス
バージョン 3.2 で追加。
部分インデックスは、指定されたフィルター式を満たすコレクション内のドキュメントのみをインデックスします。partialFilterExpression
とユニーク制約の両方を指定した場合、ユニーク制約はフィルター式を満たすドキュメントにのみ適用されます。
部分インデックスでユニーク制約を指定しても、ドキュメントがフィルタリング条件を満たさない場合は、当該ユニーク制約を満たさないドキュメントも挿入されます。例については、「ユニーク制約を持つ部分インデックス」を参照してください。
シャーディングされたクラスターとユニークインデックス
ハッシュされたインデックスにユニーク制約を指定することはできません。
範囲指定のシャーディングされたコレクションでは、次のインデックスのみが一意になります。
シャードキーのインデックス
デフォルトの
_id
インデックス、ただし、 インデックスは、 フィールドがシャードキーでない場合にのみ、シャードごとに一意の制約を強制します。_id
_id
重要
シャーディングされたクラスターでは、 _id
フィールドもシャードキーである場合にのみ、クラスター全体の_id
フィールドに一意の制約が強制されます。
_id
フィールドがシャードキーのプレフィックスのみの場合、一意制約はドキュメントを保存するシャードにのみ適用されます。つまり、異なるシャードで発生する場合には、2 つ以上のドキュメントが同じ _id
値を持つことができます。
例、2 つのシャード A と B にまたがるシャードキー{x:
1}
を持つシャーディングされたシャーディングされたコレクション。_id
キーはシャードキーではないため、コレクションにはシャード A に _id
値 1
を持つドキュメントがある可能性があります。 、およびシャード B に _id
値 1
を持つ別のドキュメントが含まれている。
_id
フィールドがシャードキーではない場合、MongoDB では、アプリケーションがシャード全体で_id
値の一意性を強制することを想定しています。
ユニークインデックスの制約とは、次のことを意味します。
シャーディングされたコレクションされたコレクションの場合、コレクションに複数の一意なインデックスがある場合、シャードキーはすべての一意なインデックスのプレフィックスではない限り、コレクションをシャードできません。
すでに シャーディングされたシャーディングされたコレクションの場合、シャードキーがプレフィックスとして含まれていない限り、他のフィールドに一意なインデックスを作成することはできません。
ユニークインデックスは、インデックスフィールドがないドキュメントの null 値を保存します。つまり、インデックスフィールドがない場合は、
null
インデックスキー値の別のインスタンスとして扱われます。詳細については、「 一意の単一フィールドインデックスにドキュメントフィールドがない場合 」を参照してください。
シャードキーではないフィールドの一意性を維持するには、「任意のフィールドに対する一意の制約 」を参照してください。
ユニークスパース/非スパースインデックス
MongoDB 5.0 以降、キー パターンが同じユニークスパースおよびユニーク非スパースインデックスを単一コレクションに混在させることができます。
ユニークおよびスパースインデックスの作成
次の例では、キー パターンが同じで sparse
オプションが異なる複数のインデックスを作成します。
db.scoreHistory.createIndex( { score : 1 }, { name: "unique_index", unique: true } ) db.scoreHistory.createIndex( { score : 1 }, { name: "unique_sparse_index", unique: true, sparse: true } )
基本インデックスとスパースインデックスの作成
スパース オプションの有無にかかわらず、同じキー パターンで基本インデックスを作成することもできます。
db.scoreHistory.createIndex( { score : 1 }, { name: "sparse_index", sparse: true } ) db.scoreHistory.createIndex( { score : 1 }, { name: "basic_index" } )
キー パターンが重複する基本インデックスとユニークインデックス
MongoDB 5.0 以降、基本インデックスとユニークインデックスは同じキー パターンで共存できます。
重複するキー パターンが認められるため、インデックスを作成済みのフィールドにユニークインデックスを追加できます。
この例では、次のことが行われます。
キー パターン { score : 1 }
を使用して基本インデックスを作成し、3 つのドキュメントを挿入します。
db.scoreHistory.createIndex( { score : 1 }, { name: "basic_index" } ) db.scoreHistory.insert( { score : 1 } ) db.scoreHistory.insert( { score : 2 } ) db.scoreHistory.insert( { score : 3 } )
同じキーパターン { score : 1 }
を持つユニークインデックスを作成します。
db.scoreHistory.createIndex( { score : 1 }, { name: "unique_index", unique: true } )
score
が重複するドキュメントの挿入を試みても、ユニークインデックスであるため失敗します。
db.scoreHistory.insert( { score : 3 } )