डायनॉम्बीडी - स्थानीय माध्यमिक सूचकांक
कुछ एप्लिकेशन केवल प्राथमिक कुंजी के साथ क्वेरी करते हैं, लेकिन कुछ स्थितियों को वैकल्पिक सॉर्ट कुंजी से लाभ होता है। एक या कई स्थानीय माध्यमिक अनुक्रमित बनाकर अपने आवेदन को एक विकल्प दें।
जटिल डेटा एक्सेस आवश्यकताएं, जैसे कि लाखों आइटमों का मुकाबला करना, अधिक कुशल प्रश्न / स्कैन करना आवश्यक है। स्थानीय माध्यमिक सूचकांक एक विभाजन कुंजी मूल्य के लिए एक वैकल्पिक सॉर्ट कुंजी प्रदान करते हैं। वे सभी या कुछ तालिका विशेषताओं की प्रतियां भी रखते हैं। वे तालिका विभाजन कुंजी द्वारा डेटा व्यवस्थित करते हैं, लेकिन एक अलग प्रकार की कुंजी का उपयोग करते हैं।
एक स्थानीय माध्यमिक सूचकांक का उपयोग पूरे टेबल स्कैन की आवश्यकता को हटाता है, और सॉर्ट कुंजी का उपयोग करके एक सरल और त्वरित क्वेरी की अनुमति देता है।
सभी स्थानीय माध्यमिक सूचकांक को कुछ शर्तों को पूरा करना होगा -
- आइडेंटिकल पार्टीशन की और सोर्स टेबल पार्टीशन की।
- केवल एक स्केलर विशेषता की एक सॉर्ट कुंजी।
- एक गैर-प्रमुख विशेषता के रूप में स्रोत तालिका सॉर्ट कुंजी अभिनय की प्रोजेक्शन।
सभी स्थानीय द्वितीयक अनुक्रमणिका स्वचालित रूप से विभाजन और पैरेंट टेबल से कुंजियाँ छाँटते हैं। प्रश्नों में, इसका मतलब है कि अनुमानित विशेषताओं का कुशल जमावड़ा, और उन विशेषताओं की पुनर्प्राप्ति भी जो अनुमानित नहीं हैं।
स्थानीय सेकेंडरी इंडेक्स के लिए स्टोरेज लिमिट 10GB प्रति पार्टीशन की वैल्यू पर रहती है, जिसमें सभी टेबल आइटम और एक डिवीजन की वैल्यू शेयर करने वाले इंडेक्स आइटम शामिल हैं।
एक विशेषता पेश करना
कुछ ऑपरेशनों में जटिलता के कारण अतिरिक्त रीड / भ्रूण की आवश्यकता होती है। ये ऑपरेशन पर्याप्त थ्रूपुट का उपभोग कर सकते हैं। प्रोजेक्शन आपको इन विशेषताओं को अलग करके महंगे प्रश्नों से बचने और समृद्ध क्वेरी करने की अनुमति देता है। याद रखें कि अनुमानों में एक द्वितीयक सूचकांक में कॉपी की गई विशेषताएँ शामिल हैं।
द्वितीयक सूचकांक बनाते समय, आप अनुमानित विशेषताओं को निर्दिष्ट करते हैं। डायनेमोडीबी द्वारा दिए गए तीन विकल्पों को याद करें:KEYS_ONLY, INCLUDE, and ALL।
जब प्रक्षेपण में कुछ विशेषताओं के लिए चुनते हैं, तो संबंधित लागत ट्रेडऑफ़ पर विचार करें -
यदि आप केवल आवश्यक विशेषताओं का एक छोटा सा सेट प्रोजेक्ट करते हैं, तो आप नाटकीय रूप से भंडारण लागत को कम करते हैं।
यदि आप अक्सर गैर-प्रमुख विशेषताओं को एक्सेस करते हैं, तो आप स्टोरेज लागत के साथ स्कैन लागत को ऑफसेट करते हैं।
यदि आप अधिकांश या सभी गैर-प्रमुख विशेषताओं को प्रोजेक्ट करते हैं, तो यह लचीलेपन को अधिकतम करता है और थ्रूपुट (कोई पुनर्प्राप्ति) को कम करता है; हालाँकि, भंडारण लागत में वृद्धि होती है।
यदि आप बार-बार लिखने / अपडेट करने और संक्षिप्त प्रश्नों के लिए KEYS_ONLY प्रोजेक्ट करते हैं, तो यह आकार को कम करता है, लेकिन क्वेरी तैयारी को बनाए रखता है।
स्थानीय माध्यमिक सूचकांक निर्माण
उपयोग LocalSecondaryIndexसिंगल या मल्टीपल लोकल सेकेंडरी इंडेक्स बनाने के लिए क्रिएटटेबल का पैरामीटर। आपको सॉर्ट कुंजी के लिए एक गैर-कुंजी विशेषता निर्दिष्ट करनी होगी। टेबल निर्माण पर, आप स्थानीय माध्यमिक सूचकांक बनाते हैं। डिलीट करने पर आप इन इंडेक्स को डिलीट कर देते हैं।
एक स्थानीय माध्यमिक सूचकांक के साथ तालिकाओं को विभाजन कुंजी मूल्य के अनुसार 10GB आकार की सीमा का पालन करना चाहिए, लेकिन किसी भी आइटम को संग्रहीत कर सकते हैं।
स्थानीय माध्यमिक सूचकांक प्रश्न और स्कैन
स्थानीय माध्यमिक अनुक्रमितों पर एक क्वेरी ऑपरेशन सभी आइटमों को एक मिलान विभाजन कुंजी मूल्य के साथ लौटाता है जब सूचकांक में कई आइटम प्रमुख मूल्यों को साझा करते हैं। मैचिंग आइटम एक निश्चित क्रम में वापस नहीं आते हैं। स्थानीय माध्यमिक अनुक्रमणिका के लिए क्वेरीज़ या तो लगातार या मजबूत स्थिरता का उपयोग करते हैं, नवीनतम मूल्यों को वितरित करने के लिए दृढ़ता से लगातार पढ़ते हैं।
एक स्कैन ऑपरेशन सभी स्थानीय माध्यमिक सूचकांक डेटा लौटाता है। स्कैन के लिए आपको एक तालिका और सूचकांक नाम प्रदान करने की आवश्यकता होती है, और डेटा को छोड़ने के लिए एक फ़िल्टर अभिव्यक्ति के उपयोग की अनुमति देता है।
आइटम लेखन
एक स्थानीय माध्यमिक सूचकांक के निर्माण पर, आप एक सॉर्ट कुंजी विशेषता और उसके डेटा प्रकार को निर्दिष्ट करते हैं। जब आप एक आइटम लिखते हैं, तो उसका प्रकार कुंजी स्कीमा के डेटा प्रकार से मेल खाना चाहिए यदि आइटम किसी इंडेक्स कुंजी की विशेषता को परिभाषित करता है।
डायनॉम्बीडी टेबल आइटम और स्थानीय माध्यमिक सूचकांक वस्तुओं पर कोई एक-से-एक संबंध आवश्यकताओं को नहीं लगाता है। कई स्थानीय माध्यमिक अनुक्रमित तालिकाएँ कम लिखने वालों की तुलना में अधिक लेखन लागत लेती हैं।
स्थानीय माध्यमिक सूचकांक में थ्रूपुट विचार
एक क्वेरी की क्षमता खपत डेटा एक्सेस की प्रकृति पर निर्भर करती है। अंततः अंतिम रीडर्स में आधे यूनिट की तुलना में एक यूनिट का उपयोग करके दृढ़ता से लगातार रीड्स के साथ क्वेरीज़ या तो सुसंगत या मजबूत स्थिरता का उपयोग करती हैं।
परिणाम सीमाओं में अधिकतम 1MB आकार शामिल है। परिणाम आकार निकटतम 4KB तक गोल अनुक्रमणिका आइटम आकार के योग से आते हैं, और मिलान तालिका आइटम आकार भी निकटतम 4KB तक गोल है।
लिखित क्षमता की खपत प्रावधानित इकाइयों के भीतर ही रहती है। तालिका लेखन में खपत इकाइयों का योग और सूचकांकों को अद्यतन करने में खपत इकाइयों की कुल मात्रा की गणना करें।
आप लागत को प्रभावित करने वाले प्रमुख कारकों पर भी विचार कर सकते हैं, जिनमें से कुछ हो सकते हैं -
जब आप अनुक्रमित विशेषता को परिभाषित करने वाले आइटम को लिखते हैं या किसी अपरिभाषित अनुक्रमित विशेषता को परिभाषित करने के लिए आइटम को अपडेट करते हैं, तो एकल लेखन ऑपरेशन होता है।
जब एक तालिका अद्यतन एक अनुक्रमित कुंजी विशेषता मान को बदलता है, तो दो राइट्स डिलीट होते हैं और फिर - एक आइटम जोड़ते हैं।
जब कोई लेखन अनुक्रमित विशेषता को हटाने का कारण बनता है, तो एक लेखन पुराने आइटम प्रक्षेपण को हटाने के लिए होता है।
जब कोई आइटम इंडेक्स में अपडेट के पहले या बाद में मौजूद नहीं होता है, तो कोई लिखता नहीं है।
स्थानीय माध्यमिक सूचकांक भंडारण
एक टेबल आइटम लिखने पर, DynamoDB स्वचालित रूप से आवश्यक स्थानीय माध्यमिक अनुक्रमित करने के लिए सेट सही विशेषता की प्रतिलिपि बनाता है। इससे आपका अकाउंट चार्ज हो जाता है। अंतरिक्ष ने तालिका प्राथमिक कुंजी बाइट आकार, सूचकांक कुंजी विशेषता बाइट आकार, किसी भी वर्तमान अनुमानित विशेषता बाइट आकार और प्रत्येक सूचकांक आइटम के लिए ओवरहेड में 100 बाइट्स के योग से परिणाम का उपयोग किया।
अनुमान संग्रहण औसत इंडेक्स आइटम आकार का आकलन करके और तालिका आइटम मात्रा से गुणा करके प्राप्त किया जाता है।
जावा का उपयोग स्थानीय माध्यमिक सूचकांक के साथ काम करने के लिए
पहली बार एक डायनॉम्बीडी श्रेणी उदाहरण बनाकर एक स्थानीय माध्यमिक सूचकांक बनाएं। फिर, आवश्यक अनुरोध जानकारी के साथ CreateTableRequest वर्ग का उदाहरण बनाएँ। अंत में, CreateTable मेथड का उपयोग करें।
उदाहरण
DynamoDB dynamoDB = new DynamoDB(new AmazonDynamoDBClient(
new ProfileCredentialsProvider()));
String tableName = "Tools";
CreateTableRequest createTableRequest = new
CreateTableRequest().withTableName(tableName);
//Provisioned Throughput
createTableRequest.setProvisionedThroughput (
new ProvisionedThroughput()
.withReadCapacityUnits((long)5)
.withWriteCapacityUnits(( long)5));
//Attributes
ArrayList<AttributeDefinition> attributeDefinitions =
new ArrayList<AttributeDefinition>();
attributeDefinitions.add(new AttributeDefinition()
.withAttributeName("Make")
.withAttributeType("S"));
attributeDefinitions.add(new AttributeDefinition()
.withAttributeName("Model")
.withAttributeType("S"));
attributeDefinitions.add(new AttributeDefinition()
.withAttributeName("Line")
.withAttributeType("S"));
createTableRequest.setAttributeDefinitions(attributeDefinitions);
//Key Schema
ArrayList<KeySchemaElement> tableKeySchema = new
ArrayList<KeySchemaElement>();
tableKeySchema.add(new KeySchemaElement()
.withAttributeName("Make")
.withKeyType(KeyType.HASH)); //Partition key
tableKeySchema.add(new KeySchemaElement()
.withAttributeName("Model")
.withKeyType(KeyType.RANGE)); //Sort key
createTableRequest.setKeySchema(tableKeySchema);
ArrayList<KeySchemaElement> indexKeySchema = new
ArrayList<KeySchemaElement>();
indexKeySchema.add(new KeySchemaElement()
.withAttributeName("Make")
.withKeyType(KeyType.HASH)); //Partition key
indexKeySchema.add(new KeySchemaElement()
.withAttributeName("Line")
.withKeyType(KeyType.RANGE)); //Sort key
Projection projection = new Projection()
.withProjectionType(ProjectionType.INCLUDE);
ArrayList<String> nonKeyAttributes = new ArrayList<String>();
nonKeyAttributes.add("Type");
nonKeyAttributes.add("Year");
projection.setNonKeyAttributes(nonKeyAttributes);
LocalSecondaryIndex localSecondaryIndex = new LocalSecondaryIndex()
.withIndexName("ModelIndex")
.withKeySchema(indexKeySchema)
.withProjection(p rojection);
ArrayList<LocalSecondaryIndex> localSecondaryIndexes = new
ArrayList<LocalSecondaryIndex>();
localSecondaryIndexes.add(localSecondaryIndex);
createTableRequest.setLocalSecondaryIndexes(localSecondaryIndexes);
Table table = dynamoDB.createTable(createTableRequest);
System.out.println(table.getDescription());
वर्णन विधि के साथ एक स्थानीय माध्यमिक सूचकांक के बारे में जानकारी प्राप्त करें। बस एक डायनमोडीबी श्रेणी उदाहरण बनाएं, तालिका वर्ग उदाहरण बनाएं, और तालिका को वर्णन विधि में पास करें।
उदाहरण
DynamoDB dynamoDB = new DynamoDB(new AmazonDynamoDBClient(
new ProfileCredentialsProvider()));
String tableName = "Tools";
Table table = dynamoDB.getTable(tableName);
TableDescription tableDescription = table.describe();
List<LocalSecondaryIndexDescription> localSecondaryIndexes =
tableDescription.getLocalSecondaryIndexes();
Iterator<LocalSecondaryIndexDescription> lsiIter =
localSecondaryIndexes.iterator();
while (lsiIter.hasNext()) {
LocalSecondaryIndexDescription lsiDescription = lsiIter.next();
System.out.println("Index info " + lsiDescription.getIndexName() + ":");
Iterator<KeySchemaElement> kseIter = lsiDescription.getKeySchema().iterator();
while (kseIter.hasNext()) {
KeySchemaElement kse = kseIter.next();
System.out.printf("\t%s: %s\n", kse.getAttributeName(), kse.getKeyType());
}
Projection projection = lsiDescription.getProjection();
System.out.println("\tProjection type: " + projection.getProjectionType());
if (projection.getProjectionType().toString().equals("INCLUDE")) {
System.out.println("\t\tNon-key projected attributes: " +
projection.getNonKeyAttributes());
}
}
तालिका क्वेरी के समान चरणों का उपयोग करके किसी क्वेरी को निष्पादित करें। केवल डायनामोबीडी श्रेणी उदाहरण, तालिका वर्ग उदाहरण, एक इंडेक्स क्लास उदाहरण, एक क्वेरी ऑब्जेक्ट और क्वेरी विधि का उपयोग करें।
उदाहरण
DynamoDB dynamoDB = new DynamoDB(new AmazonDynamoDBClient(
new ProfileCredentialsProvider()));
String tableName = "Tools";
Table table = dynamoDB.getTable(tableName);
Index index = table.getIndex("LineIndex");
QuerySpec spec = new QuerySpec()
.withKeyConditionExpression("Make = :v_make and Line = :v_line")
.withValueMap(new ValueMap()
.withString(":v_make", "Depault")
.withString(":v_line", "SuperSawz"));
ItemCollection<QueryOutcome> items = index.query(spec);
Iterator<Item> itemsIter = items.iterator();
while (itemsIter.hasNext()) {
Item item = itemsIter.next();
System.out.println(item.toJSONPretty());
}
आप निम्न उदाहरण की समीक्षा भी कर सकते हैं।
Note- निम्नलिखित उदाहरण पहले से बनाए गए डेटा स्रोत को मान सकते हैं। निष्पादित करने का प्रयास करने से पहले, सहायक पुस्तकालयों का अधिग्रहण करें और आवश्यक डेटा स्रोत बनाएं (आवश्यक विशेषताओं के साथ टेबल, या अन्य संदर्भित स्रोत)।
निम्न उदाहरण ग्रहण आईडीई, एक एडब्ल्यूएस क्रेडेंशियल फ़ाइल और एक एडब्ल्यूएस जावा प्रोजेक्ट के भीतर एडब्ल्यूएस टूलकिट का भी उपयोग करता है।
उदाहरण
import java.util.ArrayList;
import java.util.Iterator;
import com.amazonaws.auth.profile.ProfileCredentialsProvider;
import com.amazonaws.services.dynamodbv2.AmazonDynamoDBClient;
import com.amazonaws.services.dynamodbv2.document.DynamoDB;
import com.amazonaws.services.dynamodbv2.document.Index;
import com.amazonaws.services.dynamodbv2.document.Item;
import com.amazonaws.services.dynamodbv2.document.ItemCollection;
import com.amazonaws.services.dynamodbv2.document.PutItemOutcome;
import com.amazonaws.services.dynamodbv2.document.QueryOutcome;
import com.amazonaws.services.dynamodbv2.document.Table;
import com.amazonaws.services.dynamodbv2.document.spec.QuerySpec;
import com.amazonaws.services.dynamodbv2.document.utils.ValueMap;
import com.amazonaws.services.dynamodbv2.model.AttributeDefinition;
import com.amazonaws.services.dynamodbv2.model.CreateTableRequest;
import com.amazonaws.services.dynamodbv2.model.KeySchemaElement;
import com.amazonaws.services.dynamodbv2.model.KeyType;
import com.amazonaws.services.dynamodbv2.model.LocalSecondaryIndex;
import com.amazonaws.services.dynamodbv2.model.Projection;
import com.amazonaws.services.dynamodbv2.model.ProjectionType;
import com.amazonaws.services.dynamodbv2.model.ProvisionedThroughput;
import com.amazonaws.services.dynamodbv2.model.ReturnConsumedCapacity;
import com.amazonaws.services.dynamodbv2.model.Select;
public class LocalSecondaryIndexSample {
static DynamoDB dynamoDB = new DynamoDB(new AmazonDynamoDBClient(
new ProfileCredentialsProvider()));
public static String tableName = "ProductOrders";
public static void main(String[] args) throws Exception {
createTable();
query(null);
query("IsOpenIndex");
query("OrderCreationDateIndex");
}
public static void createTable() {
CreateTableRequest createTableRequest = new CreateTableRequest()
.withTableName(tableName)
.withProvisionedThroughput(new ProvisionedThroughput()
.withReadCapacityUnits((long) 1)
.withWriteCapacityUnits((long) 1));
// Table partition and sort keys attributes
ArrayList<AttributeDefinition> attributeDefinitions = new
ArrayList<AttributeDefinition>();
attributeDefinitions.add(new AttributeDefinition()
.withAttributeName("CustomerID")
.withAttributeType("S"));
attributeDefinitions.add(new AttributeDefinition()
.withAttributeName("OrderID")
.withAttributeType("N"));
// Index primary key attributes
attributeDefinitions.add(new AttributeDefinition()
.withAttributeName("OrderDate")
.withAttributeType("N"));
attributeDefinitions.add(new AttributeDefinition()
.withAttributeName("OpenStatus")
.withAttributeType("N"));
createTableRequest.setAttributeDefinitions(attributeDefinitions);
// Table key schema
ArrayList<KeySchemaElement> tableKeySchema = new
ArrayList<KeySchemaElement>();
tableKeySchema.add(new KeySchemaElement()
.withAttributeName("CustomerID")
.withKeyType(KeyType.HASH)); //Partition key
tableKeySchema.add(new KeySchemaElement()
.withAttributeName("OrderID")
.withKeyType(KeyType.RANGE)); //Sort key
createTableRequest.setKeySchema(tableKeySchema);
ArrayList<LocalSecondaryIndex> localSecondaryIndexes = new
ArrayList<LocalSecondaryIndex>();
// OrderDateIndex
LocalSecondaryIndex orderDateIndex = new LocalSecondaryIndex()
.withIndexName("OrderDateIndex");
// OrderDateIndex key schema
ArrayList<KeySchemaElement> indexKeySchema = new
ArrayList<KeySchemaElement>();
indexKeySchema.add(new KeySchemaElement()
.withAttributeName("CustomerID")
.withKeyType(KeyType.HASH)); //Partition key
indexKeySchema.add(new KeySchemaElement()
.withAttributeName("OrderDate")
.withKeyType(KeyType.RANGE)); //Sort key
orderDateIndex.setKeySchema(indexKeySchema);
// OrderCreationDateIndex projection w/attributes list
Projection projection = new Projection()
.withProjectionType(ProjectionType.INCLUDE);
ArrayList<String> nonKeyAttributes = new ArrayList<String>();
nonKeyAttributes.add("ProdCat");
nonKeyAttributes.add("ProdNomenclature");
projection.setNonKeyAttributes(nonKeyAttributes);
orderCreationDateIndex.setProjection(projection);
localSecondaryIndexes.add(orderDateIndex);
// IsOpenIndex
LocalSecondaryIndex isOpenIndex = new LocalSecondaryIndex()
.withIndexName("IsOpenIndex");
// OpenStatusIndex key schema
indexKeySchema = new ArrayList<KeySchemaElement>();
indexKeySchema.add(new KeySchemaElement()
.withAttributeName("CustomerID")
.withKeyType(KeyType.HASH)); //Partition key
indexKeySchema.add(new KeySchemaElement()
.withAttributeName("OpenStatus")
.withKeyType(KeyType.RANGE)); //Sort key
// OpenStatusIndex projection
projection = new Projection() .withProjectionType(ProjectionType.ALL);
OpenStatusIndex.setKeySchema(indexKeySchema);
OpenStatusIndex.setProjection(projection);
localSecondaryIndexes.add(OpenStatusIndex);
// Put definitions in CreateTable request
createTableRequest.setLocalSecondaryIndexes(localSecondaryIndexes);
System.out.println("Spawning table " + tableName + "...");
System.out.println(dynamoDB.createTable(createTableRequest));
// Pause for ACTIVE status
System.out.println("Waiting for ACTIVE table:" + tableName);
try {
Table table = dynamoDB.getTable(tableName);
table.waitForActive();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public static void query(String indexName) {
Table table = dynamoDB.getTable(tableName);
System.out.println("\n*************************************************\n");
System.out.println("Executing query on" + tableName);
QuerySpec querySpec = new QuerySpec()
.withConsistentRead(true)
.withScanIndexForward(true)
.withReturnConsumedCapacity(ReturnConsumedCapacity.TOTAL);
if (indexName == "OpenStatusIndex") {
System.out.println("\nEmploying index: '" + indexName
+ "' open orders for this customer.");
System.out.println(
"Returns only user-specified attribute list\n");
Index index = table.getIndex(indexName);
querySpec.withKeyConditionExpression("CustomerID = :v_custmid and
OpenStatus = :v_openstat")
.withValueMap(new ValueMap()
.withString(":v_custmid", "[email protected]")
.withNumber(":v_openstat", 1));
querySpec.withProjectionExpression(
"OrderDate, ProdCat, ProdNomenclature, OrderStatus");
ItemCollection<QueryOutcome> items = index.query(querySpec);
Iterator<Item> iterator = items.iterator();
System.out.println("Printing query results...");
while (iterator.hasNext()) {
System.out.println(iterator.next().toJSONPretty());
}
} else if (indexName == "OrderDateIndex") {
System.out.println("\nUsing index: '" + indexName
+ "': this customer's orders placed after 05/22/2016.");
System.out.println("Projected attributes are returned\n");
Index index = table.getIndex(indexName);
querySpec.withKeyConditionExpression("CustomerID = :v_custmid and OrderDate
>= :v_ordrdate")
.withValueMap(new ValueMap()
.withString(":v_custmid", "[email protected]")
.withNumber(":v_ordrdate", 20160522));
querySpec.withSelect(Select.ALL_PROJECTED_ATTRIBUTES);
ItemCollection<QueryOutcome> items = index.query(querySpec);
Iterator<Item> iterator = items.iterator();
System.out.println("Printing query results...");
while (iterator.hasNext()) {
System.out.println(iterator.next().toJSONPretty());
}
} else {
System.out.println("\nNo index: All Jane's orders by OrderID:\n");
querySpec.withKeyConditionExpression("CustomerID = :v_custmid")
.withValueMap(new ValueMap()
.withString(":v_custmid", "[email protected]"));
ItemCollection<QueryOutcome> items = table.query(querySpec);
Iterator<Item> iterator = items.iterator();
System.out.println("Printing query results...");
while (iterator.hasNext()) {
System.out.println(iterator.next().toJSONPretty());
}
}
}
}