DocumentDB - Partitionierung
Wenn Ihre Datenbank über 10 GB hinaus wächst, können Sie einfach skalieren, indem Sie neue Sammlungen erstellen und Ihre Daten dann auf immer mehr Sammlungen verteilen oder partitionieren.
Früher oder später wird eine einzelne Sammlung mit einer Kapazität von 10 GB nicht ausreichen, um Ihre Datenbank zu enthalten. Jetzt klingen 10 GB möglicherweise nicht nach einer sehr großen Zahl, aber denken Sie daran, dass wir JSON-Dokumente speichern, bei denen es sich nur um einfachen Text handelt und Sie viele einfache Textdokumente in 10 GB einfügen können, selbst wenn Sie den Speicheraufwand für die Indizes berücksichtigen.
Speicher ist nicht das einzige Problem, wenn es um Skalierbarkeit geht. Der maximale Durchsatz, der für eine Sammlung verfügbar ist, beträgt zweieinhalbtausend Anforderungseinheiten pro Sekunde, die Sie mit einer S3-Sammlung erhalten. Wenn Sie also einen höheren Durchsatz benötigen, müssen Sie auch durch Partitionierung mit mehreren Sammlungen skalieren. Scale-Out-Partitionierung wird auch genannthorizontal partitioning.
Es gibt viele Ansätze, die zum Partitionieren von Daten mit Azure DocumentDB verwendet werden können. Im Folgenden sind die gängigsten Strategien aufgeführt:
- Spillover-Partitionierung
- Bereichspartitionierung
- Lookup-Partitionierung
- Hash-Partitionierung
Spillover-Partitionierung
Spillover-Partitionierung ist die einfachste Strategie, da kein Partitionsschlüssel vorhanden ist. Es ist oft eine gute Wahl, wenn Sie sich über viele Dinge nicht sicher sind. Möglicherweise wissen Sie nicht, ob Sie jemals über eine einzelne Sammlung hinaus skalieren müssen oder wie viele Sammlungen Sie möglicherweise hinzufügen müssen oder wie schnell Sie sie möglicherweise hinzufügen müssen.
Die Spillover-Partitionierung beginnt mit einer einzelnen Sammlung und es gibt keinen Partitionsschlüssel.
Die Sammlung beginnt zu wachsen und wächst dann weiter und dann weiter, bis Sie sich dem 10-GB-Limit nähern.
Wenn Sie eine Kapazität von 90 Prozent erreichen, werden Sie auf eine neue Sammlung übertragen und diese für neue Dokumente verwenden.
Sobald Ihre Datenbank auf eine größere Anzahl von Sammlungen skaliert ist, möchten Sie wahrscheinlich zu einer Strategie wechseln, die auf einem Partitionsschlüssel basiert.
Wenn Sie dies tun, müssen Sie Ihre Daten neu ausgleichen, indem Sie Dokumente basierend auf der Strategie, zu der Sie migrieren, in verschiedene Sammlungen verschieben.
Bereichspartitionierung
Eine der häufigsten Strategien ist die Bereichspartitionierung. Mit diesem Ansatz bestimmen Sie den Wertebereich, in den der Partitionsschlüssel eines Dokuments fallen kann, und leiten das Dokument an eine Sammlung weiter, die diesem Bereich entspricht.
Bei dieser Strategie werden in der Regel Daten verwendet, bei denen Sie eine Sammlung erstellen, in der Dokumente gespeichert werden, die innerhalb des definierten Datumsbereichs liegen. Wenn Sie Bereiche definieren, die klein genug sind, können Sie sicher sein, dass keine Sammlung jemals die 10-GB-Grenze überschreitet. Beispielsweise kann es ein Szenario geben, in dem eine einzelne Sammlung Dokumente für einen ganzen Monat angemessen verarbeiten kann.
Es kann auch vorkommen, dass die meisten Benutzer nach aktuellen Daten fragen, bei denen es sich um Daten für diesen Monat oder möglicherweise für den letzten Monat handelt. Benutzer suchen jedoch selten nach viel älteren Daten. Sie beginnen also im Juni mit einer S3-Kollektion, die die teuerste Kollektion ist, die Sie kaufen können, und den besten Durchsatz liefert, den Sie erzielen können.
Im Juli kaufen Sie eine weitere S3-Sammlung, um die Juli-Daten zu speichern, und Sie skalieren die Juni-Daten auf eine kostengünstigere S2-Sammlung. Im August erhalten Sie dann eine weitere S3-Sammlung und skalieren den Juli auf einen S2 und den Juni bis auf einen S1. Monat für Monat halten Sie immer die aktuellen Daten für einen hohen Durchsatz verfügbar, und ältere Daten werden für einen niedrigeren Durchsatz verfügbar gehalten.
Solange die Abfrage einen Partitionsschlüssel bereitstellt, wird nur die Sammlung abgefragt, die abgefragt werden muss, und nicht alle Sammlungen in der Datenbank, wie dies bei der Spillover-Partitionierung der Fall ist.
Lookup-Partitionierung
Mit der Lookup-Partitionierung können Sie eine Partitionszuordnung definieren, die Dokumente basierend auf ihrem Partitionsschlüssel an bestimmte Sammlungen weiterleitet. Sie können beispielsweise nach Regionen partitionieren.
Speichern Sie alle US-Dokumente in einer Sammlung, alle europäischen Dokumente in einer anderen Sammlung und alle Dokumente aus einer anderen Region in einer dritten Sammlung.
Verwenden Sie diese Partitionszuordnung, und ein Lookup-Partitionsauflöser kann anhand des Partitionsschlüssels, der die in jedem Dokument enthaltene Regionseigenschaft ist, herausfinden, in welcher Sammlung ein Dokument erstellt und welche Sammlungen abgefragt werden sollen.
Hash-Partitionierung
Bei der Hash-Partitionierung werden Partitionen basierend auf dem Wert einer Hash-Funktion zugewiesen, sodass Sie Anforderungen und Daten gleichmäßig auf mehrere Partitionen verteilen können.
Dies wird häufig zum Partitionieren von Daten verwendet, die von einer großen Anzahl unterschiedlicher Clients erstellt oder verbraucht wurden, und ist nützlich zum Speichern von Benutzerprofilen, Katalogelementen usw.
Schauen wir uns ein einfaches Beispiel für die Bereichspartitionierung mit dem vom .NET SDK bereitgestellten RangePartitionResolver an.
Step 1- Erstellen Sie einen neuen DocumentClient, und wir erstellen zwei Sammlungen in der Aufgabe "CreateCollections". Eines enthält Dokumente für Benutzer mit Benutzer-IDs, die mit A bis M beginnen, und das andere für Benutzer-IDs N bis Z.
private static async Task CreateCollections(DocumentClient client) {
await client.CreateDocumentCollectionAsync(“dbs/myfirstdb”, new DocumentCollection {
Id = “CollectionAM” });
await client.CreateDocumentCollectionAsync(“dbs/myfirstdb”, new DocumentCollection {
Id = “CollectionNZ” });
}
Step 2 - Registrieren Sie den Entfernungsauflöser für die Datenbank.
Step 3- Erstellen Sie einen neuen RangePartitionResolver <string>, der der Datentyp unseres Partitionsschlüssels ist. Der Konstruktor verwendet zwei Parameter, den Eigenschaftsnamen des Partitionsschlüssels und ein Wörterbuch, das die Shard-Map oder Partitions-Map ist. Dies ist nur eine Liste der Bereiche und entsprechenden Sammlungen, die wir für den Resolver vordefinieren.
private static void RegisterRangeResolver(DocumentClient client) {
//Note: \uffff is the largest UTF8 value, so M\ufff includes all strings that start with M.
var resolver = new RangePartitionResolver<string>(
"userId", new Dictionary<Range<string>, string>() {
{ new Range<string>("A", "M\uffff"), "dbs/myfirstdb/colls/CollectionAM" },
{ new Range<string>("N", "Z\uffff"), "dbs/myfirstdb/colls/CollectionNZ" },
});
client.PartitionResolvers["dbs/myfirstdb"] = resolver;
}
Hier muss der größtmögliche UTF-8-Wert codiert werden. Andernfalls würde der erste Bereich mit keinem Ms außer dem einen einzelnen M und ebenfalls mit Z im zweiten Bereich übereinstimmen. Sie können sich diesen codierten Wert hier also als Platzhalter für den Abgleich mit dem Partitionsschlüssel vorstellen.
Step 4- Registrieren Sie den Resolver nach dem Erstellen für die Datenbank beim aktuellen DocumentClient. Weisen Sie es dazu einfach der Dictionary-Eigenschaft von PartitionResolver zu.
Wir erstellen und fragen Dokumente für die Datenbank ab, nicht wie gewohnt für eine Sammlung. Der Resolver verwendet diese Zuordnung, um Anforderungen an die entsprechenden Sammlungen weiterzuleiten.
Lassen Sie uns nun einige Dokumente erstellen. Zuerst erstellen wir eine für userId Kirk und dann eine für Spock.
private static async Task CreateDocumentsAcrossPartitions(DocumentClient client) {
Console.WriteLine();
Console.WriteLine("**** Create Documents Across Partitions ****");
var kirkDocument = await client.CreateDocumentAsync("dbs/myfirstdb", new { userId =
"Kirk", title = "Captain" });
Console.WriteLine("Document 1: {0}", kirkDocument.Resource.SelfLink);
var spockDocument = await client.CreateDocumentAsync("dbs/myfirstdb", new { userId =
"Spock", title = "Science Officer" });
Console.WriteLine("Document 2: {0}", spockDocument.Resource.SelfLink);
}
Der erste Parameter hier ist eine Selbstverknüpfung mit der Datenbank, keine bestimmte Sammlung. Dies ist ohne einen Partitionsauflöser nicht möglich, aber mit einem funktioniert es einfach nahtlos.
Beide Dokumente wurden in der Datenbank myfirstdb gespeichert, aber wir wissen, dass Kirk in der Sammlung für A bis M und Spock in der Sammlung für N bis Z gespeichert werden, wenn unser RangePartitionResolver ordnungsgemäß funktioniert.
Rufen Sie diese von der CreateDocumentClient-Task aus auf, wie im folgenden Code gezeigt.
private static async Task CreateDocumentClient() {
// Create a new instance of the DocumentClient
using (var client = new DocumentClient(new Uri(EndpointUrl), AuthorizationKey)) {
await CreateCollections(client);
RegisterRangeResolver(client);
await CreateDocumentsAcrossPartitions(client);
}
}
Wenn der obige Code ausgeführt wird, erhalten Sie die folgende Ausgabe.
**** Create Documents Across Partitions ****
Document 1: dbs/Ic8LAA==/colls/Ic8LAO2DxAA=/docs/Ic8LAO2DxAABAAAAAAAAAA==/
Document 2: dbs/Ic8LAA==/colls/Ic8LAP12QAE=/docs/Ic8LAP12QAEBAAAAAAAAAA==/
Wie zu sehen ist, haben die Selbstverknüpfungen der beiden Dokumente unterschiedliche Ressourcen-IDs, da sie in zwei separaten Sammlungen vorhanden sind.