डायनॉम्बीडी - स्थानीय माध्यमिक सूचकांक

कुछ एप्लिकेशन केवल प्राथमिक कुंजी के साथ क्वेरी करते हैं, लेकिन कुछ स्थितियों को वैकल्पिक सॉर्ट कुंजी से लाभ होता है। एक या कई स्थानीय माध्यमिक अनुक्रमित बनाकर अपने आवेदन को एक विकल्प दें।

जटिल डेटा एक्सेस आवश्यकताएं, जैसे कि लाखों आइटमों का मुकाबला करना, अधिक कुशल प्रश्न / स्कैन करना आवश्यक है। स्थानीय माध्यमिक सूचकांक एक विभाजन कुंजी मूल्य के लिए एक वैकल्पिक सॉर्ट कुंजी प्रदान करते हैं। वे सभी या कुछ तालिका विशेषताओं की प्रतियां भी रखते हैं। वे तालिका विभाजन कुंजी द्वारा डेटा व्यवस्थित करते हैं, लेकिन एक अलग प्रकार की कुंजी का उपयोग करते हैं।

एक स्थानीय माध्यमिक सूचकांक का उपयोग पूरे टेबल स्कैन की आवश्यकता को हटाता है, और सॉर्ट कुंजी का उपयोग करके एक सरल और त्वरित क्वेरी की अनुमति देता है।

सभी स्थानीय माध्यमिक सूचकांक को कुछ शर्तों को पूरा करना होगा -

  • आइडेंटिकल पार्टीशन की और सोर्स टेबल पार्टीशन की।
  • केवल एक स्केलर विशेषता की एक सॉर्ट कुंजी।
  • एक गैर-प्रमुख विशेषता के रूप में स्रोत तालिका सॉर्ट कुंजी अभिनय की प्रोजेक्शन।

सभी स्थानीय द्वितीयक अनुक्रमणिका स्वचालित रूप से विभाजन और पैरेंट टेबल से कुंजियाँ छाँटते हैं। प्रश्नों में, इसका मतलब है कि अनुमानित विशेषताओं का कुशल जमावड़ा, और उन विशेषताओं की पुनर्प्राप्ति भी जो अनुमानित नहीं हैं।

स्थानीय सेकेंडरी इंडेक्स के लिए स्टोरेज लिमिट 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()); 
         } 
      } 
   } 
}