DynamoDB - Küresel İkincil Endeksler

Farklı özniteliklere sahip çeşitli sorgu türleri gerektiren uygulamalar, bu ayrıntılı sorguları gerçekleştirirken tek veya birden çok genel ikincil dizin kullanabilir.

For example - Kullanıcıları, oturum açma durumlarını ve oturum açma sürelerini takip eden bir sistem. Önceki örneğin büyümesi, verileri üzerindeki sorguları yavaşlatır.

Global ikincil dizinler, bir tablodan bir dizi öznitelik düzenleyerek sorguları hızlandırır. Verileri sıralamak için birincil anahtarları kullanırlar ve hiçbir anahtar tablo öznitelikleri veya tablo ile aynı anahtar şeması gerektirmezler.

Tüm genel ikincil dizinler, sıralama anahtarı seçeneğiyle birlikte bir bölüm anahtarı içermelidir. Dizin anahtarı şeması tablodan farklı olabilir ve dizin anahtarı öznitelikleri herhangi bir üst düzey dize, sayı veya ikili tablo özniteliklerini kullanabilir.

Bir projeksiyonda, diğer tablo niteliklerini kullanabilirsiniz, ancak sorgular üst tablolardan alınmaz.

Öznitelik Tahminleri

Öngörüler, tablodan ikincil dizine kopyalanan bir öznitelik kümesinden oluşur. Bir projeksiyon her zaman tablo bölüm anahtarı ve sıralama anahtarı ile gerçekleşir. Sorgularda, projeksiyonlar DynamoDB'nin projeksiyonun herhangi bir özelliğine erişmesine izin verir; esasen kendi tabloları olarak var olurlar.

İkincil bir dizin oluşturmada, projeksiyon için öznitelikler belirtmelisiniz. DynamoDB, bu görevi gerçekleştirmek için üç yol sunar -

  • KEYS_ONLY- Tüm dizin öğeleri, tablo bölümü ve sıralama anahtarı değerleri ile dizin anahtarı değerlerinden oluşur. Bu, en küçük dizini oluşturur.

  • INCLUDE - KEYS_ONLY özniteliklerini ve anahtar olmayan belirtilmiş öznitelikleri içerir.

  • ALL - Mümkün olan en büyük dizini oluşturarak tüm kaynak tablo özelliklerini içerir.

Verimlilik ve depolama maliyeti ile ilgili olan nitelikleri küresel bir ikincil dizine yansıtmadaki değiş tokuşlara dikkat edin.

Aşağıdaki noktaları düşünün -

  • Düşük gecikmeyle yalnızca birkaç özelliğe erişmeniz gerekiyorsa, yalnızca ihtiyacınız olanları yansıtın. Bu, depolama ve yazma maliyetlerini azaltır.

  • Bir uygulama belirli anahtar olmayan özelliklere sık sık erişiyorsa, bunları yansıtın çünkü tarama tüketimine kıyasla depolama maliyetleri azalır.

  • Sık erişilen büyük öznitelik kümelerini yansıtabilirsiniz, ancak bu yüksek bir depolama maliyeti taşır.

  • Seyrek tablo sorguları ve sık yazma / güncellemeler için KEYS_ONLY kullanın. Bu, boyutu kontrol eder, ancak yine de sorgularda iyi performans sunar.

Küresel İkincil Dizin Sorguları ve Taramaları

Bir dizindeki tek veya birden çok öğeye erişmek için sorguları kullanabilirsiniz. Dizin ve tablo adını, istenen öznitelikleri ve koşulları belirtmelisiniz; sonuçları artan veya azalan sırada döndürme seçeneği ile.

Tüm dizin verilerini almak için taramaları da kullanabilirsiniz. Tablo ve dizin adı gerektirir. Belirli verileri almak için bir filtre ifadesi kullanırsınız.

Tablo ve Dizin Verileri Senkronizasyonu

DynamoDB, üst tablolarıyla dizinlerde otomatik olarak senkronizasyon gerçekleştirir. Öğeler üzerinde yapılan her değişiklik işlemi eşzamansız güncellemelere neden olur, ancak uygulamalar dizinlere doğrudan yazmaz.

DynamoDB bakımının endeksler üzerindeki etkisini anlamanız gerekir. Bir dizin oluştururken, anahtar özniteliklerini ve veri türlerini belirtirsiniz, yani bir yazma işleminde bu veri türlerinin anahtar şema veri türleriyle eşleşmesi gerekir.

Öğe oluşturma veya silme işleminde, dizinler sonuçta tutarlı bir şekilde güncellenir, ancak verilerde yapılan güncellemeler saniyenin çok altında bir sürede yayılır (bazı türlerde sistem arızası olmadıkça). Başvurulardaki bu gecikmeyi hesaba katmalısınız.

Throughput Considerations in Global Secondary Indexes- Birden çok küresel ikincil dizin, verimi etkiler. Dizin oluşturma, tablodan ayrı var olan kapasite birimi belirtimlerini gerektirir ve bu da işlemlerin tablo birimleri yerine dizin kapasite birimlerini tüketmesine neden olur.

Bu, bir sorgu veya yazma, sağlanan aktarım hızını aşarsa azaltmaya neden olabilir. Kullanarak işleme hızı ayarlarını görüntüleyinDescribeTable.

Read Capacity- Küresel ikincil dizinler nihai tutarlılık sağlar. Sorgularda DynamoDB, tablolar için kullanılanla aynı provizyon hesaplamalarını, kalem boyutu yerine dizin girişi boyutunu kullanma farkıyla gerçekleştirir. Bir sorgu döndürme sınırı, döndürülen her öğe için öznitelik adı boyutunu ve değerlerini içeren 1 MB olarak kalır.

Yazma Kapasitesi

Yazma işlemleri gerçekleştiğinde, etkilenen dizin yazma birimlerini tüketir. Yazma çıktı maliyetleri, tablo yazma işlemlerinde tüketilen yazma kapasitesi birimleri ve dizin güncellemelerinde tüketilen birimlerin toplamıdır. Başarılı bir yazma işlemi, yeterli kapasite gerektirir veya kısıtlamaya neden olur.

Yazma maliyetleri de, bazıları aşağıdaki gibi belirli faktörlere bağlı kalır -

  • Dizine alınmış öznitelikleri tanımlayan yeni öğeler veya tanımlanmamış dizine alınmış öznitelikleri tanımlayan öğe güncellemeleri, öğeyi dizine eklemek için tek bir yazma işlemi kullanır.

  • Dizine alınmış anahtar özellik değerini değiştiren güncellemeler, bir öğeyi silmek ve yeni bir tane yazmak için iki yazma kullanır.

  • Dizine alınmış bir özniteliğin silinmesini tetikleyen bir tablo yazımı, dizindeki eski öğe projeksiyonunu silmek için tek bir yazma kullanır.

  • Bir güncelleme işleminden önce ve sonra dizinde bulunmayan öğeler yazma kullanmaz.

  • Dizine alınmış anahtar öznitelik değerini değil, dizin anahtarı şemasında yalnızca öngörülen öznitelik değerini değiştiren güncellemeler, öngörülen özniteliklerin değerlerini dizine güncellemek için bir yazma kullanır.

Tüm bu faktörler, öğe boyutunun 1KB'den küçük veya buna eşit olduğunu varsayar.

Global İkincil Endeks Depolama

Bir materyal yazıldığında, DynamoDB, özniteliklerin var olması gereken herhangi bir dizine doğru öznitelik kümesini otomatik olarak kopyalar. Bu, hesabınızı tablo öğesi depolaması ve öznitelik depolaması için ücretlendirerek etkiler. Kullanılan alan, bu miktarların toplamından elde edilir -

  • Tablo birincil anahtarının bayt boyutu
  • Dizin anahtarı özelliğinin bayt boyutu
  • Öngörülen özniteliklerin bayt boyutu
  • Dizin öğesi başına 100 bayt ek yük

Ortalama öğe boyutunu tahmin ederek ve genel ikincil dizin anahtarı öznitelikleri ile tablo öğelerinin miktarıyla çarparak depolama ihtiyaçlarını tahmin edebilirsiniz.

DynamoDB, dizin bölümü veya sıralama anahtarı olarak tanımlanan tanımsız bir özniteliğe sahip bir tablo öğesi için öğe verilerini yazmaz.

Küresel İkincil Endeksi Crud

Kullanarak global ikincil dizinlere sahip bir tablo oluşturun CreateTable ile eşleştirilmiş operasyon GlobalSecondaryIndexesparametre. Dizin bölüm anahtarı olarak işlev görecek bir öznitelik belirtmeli veya dizin sıralama anahtarı için başka bir tane kullanmalısınız. Tüm dizin anahtarı öznitelikleri dize, sayı veya ikili skaler olmalıdır. Ayrıca aşağıdakilerden oluşan aktarım hızı ayarlarını da sağlamalısınız:ReadCapacityUnits ve WriteCapacityUnits.

Kullanım UpdateTable GlobalSecondaryIndexes parametresini bir kez daha kullanarak mevcut tablolara global ikincil dizinler eklemek için.

Bu işlemde aşağıdaki girişleri sağlamalısınız -

  • Dizin adı
  • Anahtar şema
  • Öngörülen öznitelikler
  • Çıktı ayarları

Genel bir ikincil dizin ekleyerek, öğe hacmi, öngörülen öznitelikler hacmi, yazma kapasitesi ve yazma etkinliği nedeniyle büyük tablolarda önemli bir zaman alabilir. KullanımCloudWatch süreci izlemek için metrikler.

Kullanım DescribeTableküresel bir ikincil dizin için durum bilgilerini almak için. Dörtten birini döndürürIndexStatus GlobalSecondaryIndexes için -

  • CREATING - Dizinin oluşturulma aşamasını ve kullanılamadığını gösterir.

  • ACTIVE - Dizinin kullanıma hazır olduğunu gösterir.

  • UPDATING - Verim ayarlarının güncelleme durumunu gösterir.

  • DELETING - Dizinin silinme durumunu ve kullanım için kalıcı olarak kullanılamadığını gösterir.

Yükleme / doldurma aşamasında genel ikincil dizin tarafından sağlanan aktarım hızı ayarlarını güncelleyin (DynamoDB bir dizine öznitelikleri yazma ve eklenen / silinen / güncellenen öğeleri izleme). KullanımUpdateTable bu işlemi gerçekleştirmek için.

Dolgu aşamasında diğer endeksleri ekleyemeyeceğinizi / silemeyeceğinizi unutmamalısınız.

Genel ikincil dizinleri silmek için UpdateTable'ı kullanın. İşlem başına yalnızca bir dizinin silinmesine izin verir, ancak aynı anda birden çok işlemi beş adede kadar çalıştırabilirsiniz. Silme işlemi, üst tablonun okuma / yazma etkinliklerini etkilemez, ancak işlem tamamlanana kadar diğer dizinleri ekleyemez / silemezsiniz.

Global İkincil Dizinlerle Çalışmak için Java'yı Kullanma

CreateTable aracılığıyla dizini olan bir tablo oluşturun. Basitçe bir DynamoDB sınıfı örneği oluşturun,CreateTableRequest bilgi isteği için sınıf örneği ve istek nesnesini CreateTable yöntemine iletin.

Aşağıdaki program kısa bir örnektir -

DynamoDB dynamoDB = new DynamoDB(new AmazonDynamoDBClient ( 
   new ProfileCredentialsProvider()));
   
// Attributes 
ArrayList<AttributeDefinition> attributeDefinitions = new 
   ArrayList<AttributeDefinition>();  
attributeDefinitions.add(new AttributeDefinition() 
   .withAttributeName("City") 
   .withAttributeType("S"));
   
attributeDefinitions.add(new AttributeDefinition() 
   .withAttributeName("Date") 
   .withAttributeType("S"));
   
attributeDefinitions.add(new AttributeDefinition() 
   .withAttributeName("Wind") 
   .withAttributeType("N"));
   
// Key schema of the table 
ArrayList<KeySchemaElement> tableKeySchema = new ArrayList<KeySchemaElement>(); 
tableKeySchema.add(new KeySchemaElement()
   .withAttributeName("City") 
   .withKeyType(KeyType.HASH));              //Partition key
   
tableKeySchema.add(new KeySchemaElement() 
   .withAttributeName("Date") 
   .withKeyType(KeyType.RANGE));             //Sort key
   
// Wind index 
GlobalSecondaryIndex windIndex = new GlobalSecondaryIndex() 
   .withIndexName("WindIndex") 
   .withProvisionedThroughput(new ProvisionedThroughput() 
   .withReadCapacityUnits((long) 10) 
   .withWriteCapacityUnits((long) 1)) 
   .withProjection(new Projection().withProjectionType(ProjectionType.ALL));
   
ArrayList<KeySchemaElement> indexKeySchema = new ArrayList<KeySchemaElement>(); 
indexKeySchema.add(new KeySchemaElement() 
   .withAttributeName("Date") 
   .withKeyType(KeyType.HASH));              //Partition key
   
indexKeySchema.add(new KeySchemaElement() 
   .withAttributeName("Wind") 
   .withKeyType(KeyType.RANGE));             //Sort key
   
windIndex.setKeySchema(indexKeySchema);  
CreateTableRequest createTableRequest = new CreateTableRequest() 
   .withTableName("ClimateInfo") 
   .withProvisionedThroughput(new ProvisionedThroughput() 
   .withReadCapacityUnits((long) 5) 
   .withWriteCapacityUnits((long) 1))
   .withAttributeDefinitions(attributeDefinitions) 
   .withKeySchema(tableKeySchema) 
   .withGlobalSecondaryIndexes(windIndex); 
Table table = dynamoDB.createTable(createTableRequest); 
System.out.println(table.getDescription());

İndeks bilgilerini şu şekilde alın: DescribeTable. İlk olarak, bir DynamoDB sınıfı örneği oluşturun. Ardından bir dizini hedeflemek için bir Tablo sınıfı örneği oluşturun. Son olarak, tabloyu açıklama yöntemine geçirin.

İşte kısa bir örnek -

DynamoDB dynamoDB = new DynamoDB(new AmazonDynamoDBClient ( 
   new ProfileCredentialsProvider()));
   
Table table = dynamoDB.getTable("ClimateInfo"); 
TableDescription tableDesc = table.describe();  
Iterator<GlobalSecondaryIndexDescription> gsiIter = 
   tableDesc.getGlobalSecondaryIndexes().iterator(); 

while (gsiIter.hasNext()) { 
   GlobalSecondaryIndexDescription gsiDesc = gsiIter.next(); 
   System.out.println("Index data " + gsiDesc.getIndexName() + ":");  
   Iterator<KeySchemaElement> kse7Iter = gsiDesc.getKeySchema().iterator(); 
   
   while (kseIter.hasNext()) { 
      KeySchemaElement kse = kseIter.next(); 
      System.out.printf("\t%s: %s\n", kse.getAttributeName(), kse.getKeyType()); 
   }
   Projection projection = gsiDesc.getProjection(); 
   System.out.println("\tProjection type: " + projection.getProjectionType()); 
   
   if (projection.getProjectionType().toString().equals("INCLUDE")) { 
      System.out.println("\t\tNon-key projected attributes: " 
         + projection.getNonKeyAttributes()); 
   } 
}

Tablo sorgusunda olduğu gibi bir dizin sorgusu gerçekleştirmek için Sorguyu kullanın. Basitçe bir DynamoDB sınıfı örneği, hedef dizin için bir Tablo sınıfı örneği, belirli dizin için bir Index sınıfı örneği oluşturun ve dizini ve sorgu nesnesini sorgu yöntemine iletin.

Daha iyi anlamak için aşağıdaki koda bir göz atın -

DynamoDB dynamoDB = new DynamoDB(new AmazonDynamoDBClient ( 
   new ProfileCredentialsProvider()));
   
Table table = dynamoDB.getTable("ClimateInfo"); 
Index index = table.getIndex("WindIndex");  
QuerySpec spec = new QuerySpec() 
   .withKeyConditionExpression("#d = :v_date and Wind = :v_wind") 
   .withNameMap(new NameMap() 
   .with("#d", "Date"))
   .withValueMap(new ValueMap() 
   .withString(":v_date","2016-05-15") 
   .withNumber(":v_wind",0));
   
ItemCollection<QueryOutcome> items = index.query(spec);
Iterator<Item> iter = items.iterator();

while (iter.hasNext()) {
   System.out.println(iter.next().toJSONPretty()); 
}

Aşağıdaki program daha iyi anlamak için daha büyük bir örnektir -

Note- Aşağıdaki program önceden oluşturulmuş bir veri kaynağını varsayabilir. Yürütme girişiminde bulunmadan önce, destekleyici kitaplıklar edinin ve gerekli veri kaynaklarını oluşturun (gerekli özelliklere sahip tablolar veya diğer başvurulan kaynaklar).

Bu örnekte ayrıca Eclipse IDE, bir AWS kimlik bilgileri dosyası ve bir Eclipse AWS Java Projesi içindeki AWS Toolkit kullanılmaktadır.

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.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.GlobalSecondaryIndex;
import com.amazonaws.services.dynamodbv2.model.KeySchemaElement;
import com.amazonaws.services.dynamodbv2.model.KeyType;
import com.amazonaws.services.dynamodbv2.model.Projection;
import com.amazonaws.services.dynamodbv2.model.ProvisionedThroughput;

public class GlobalSecondaryIndexSample {  
   static DynamoDB dynamoDB = new DynamoDB(new AmazonDynamoDBClient ( 
      new ProfileCredentialsProvider()));  
   public static String tableName = "Bugs";   
   public static void main(String[] args) throws Exception {  
      createTable(); 
      queryIndex("CreationDateIndex"); 
      queryIndex("NameIndex"); 
      queryIndex("DueDateIndex"); 
   }
   public static void createTable() {  
      // Attributes 
      ArrayList<AttributeDefinition> attributeDefinitions = new 
         ArrayList<AttributeDefinition>();  
      attributeDefinitions.add(new AttributeDefinition()
         .withAttributeName("BugID") 
         .withAttributeType("S")); 
         
      attributeDefinitions.add(new AttributeDefinition() 
         .withAttributeName("Name")
         .withAttributeType("S"));
         
      attributeDefinitions.add(new AttributeDefinition() 
         .withAttributeName("CreationDate")
         .withAttributeType("S"));
         
      attributeDefinitions.add(new AttributeDefinition() 
         .withAttributeName("DueDate") 
         .withAttributeType("S"));
         
      // Table Key schema
      ArrayList<KeySchemaElement> tableKeySchema = new ArrayList<KeySchemaElement>(); 
      tableKeySchema.add (new KeySchemaElement() 
         .withAttributeName("BugID") 
         .withKeyType(KeyType.HASH));              //Partition key 
      
      tableKeySchema.add (new KeySchemaElement() 
         .withAttributeName("Name") 
         .withKeyType(KeyType.RANGE));             //Sort key
         
      // Indexes' initial provisioned throughput
      ProvisionedThroughput ptIndex = new ProvisionedThroughput()
         .withReadCapacityUnits(1L)
         .withWriteCapacityUnits(1L);
         
      // CreationDateIndex 
      GlobalSecondaryIndex creationDateIndex = new GlobalSecondaryIndex() 
         .withIndexName("CreationDateIndex") 
         .withProvisionedThroughput(ptIndex) 
         .withKeySchema(new KeySchemaElement() 
         .withAttributeName("CreationDate") 
         .withKeyType(KeyType.HASH),               //Partition key 
         new KeySchemaElement()
         .withAttributeName("BugID") 
         .withKeyType(KeyType.RANGE))              //Sort key 
         .withProjection(new Projection() 
         .withProjectionType("INCLUDE") 
         .withNonKeyAttributes("Description", "Status"));
         
      // NameIndex 
      GlobalSecondaryIndex nameIndex = new GlobalSecondaryIndex() 
         .withIndexName("NameIndex") 
         .withProvisionedThroughput(ptIndex) 
         .withKeySchema(new KeySchemaElement()  
         .withAttributeName("Name")  
         .withKeyType(KeyType.HASH),                  //Partition key 
         new KeySchemaElement()  
         .withAttributeName("BugID")  
         .withKeyType(KeyType.RANGE))                 //Sort key 
         .withProjection(new Projection() 
         .withProjectionType("KEYS_ONLY"));
         
      // DueDateIndex 
      GlobalSecondaryIndex dueDateIndex = new GlobalSecondaryIndex() 
         .withIndexName("DueDateIndex") 
         .withProvisionedThroughput(ptIndex) 
         .withKeySchema(new KeySchemaElement() 
         .withAttributeName("DueDate") 
         .withKeyType(KeyType.HASH))               //Partition key 
         .withProjection(new Projection() 
         .withProjectionType("ALL"));
         
      CreateTableRequest createTableRequest = new CreateTableRequest() 
         .withTableName(tableName) 
         .withProvisionedThroughput( new ProvisionedThroughput() 
         .withReadCapacityUnits( (long) 1) 
         .withWriteCapacityUnits( (long) 1)) 
         .withAttributeDefinitions(attributeDefinitions)
         .withKeySchema(tableKeySchema)
         .withGlobalSecondaryIndexes(creationDateIndex, nameIndex, dueDateIndex);  
         System.out.println("Creating " + tableName + "..."); 
         dynamoDB.createTable(createTableRequest);  
      
      // Pause for active table state 
      System.out.println("Waiting for ACTIVE state of " + tableName); 
      try { 
         Table table = dynamoDB.getTable(tableName); 
         table.waitForActive(); 
      } catch (InterruptedException e) { 
         e.printStackTrace(); 
      } 
   }
   public static void queryIndex(String indexName) { 
      Table table = dynamoDB.getTable(tableName);  
      System.out.println 
      ("\n*****************************************************\n"); 
      System.out.print("Querying index " + indexName + "...");  
      Index index = table.getIndex(indexName);  
      ItemCollection<QueryOutcome> items = null; 
      QuerySpec querySpec = new QuerySpec();  
      
      if (indexName == "CreationDateIndex") { 
         System.out.println("Issues filed on 2016-05-22"); 
         querySpec.withKeyConditionExpression("CreationDate = :v_date and begins_with
            (BugID, :v_bug)") 
            .withValueMap(new ValueMap() 
            .withString(":v_date","2016-05-22")
            .withString(":v_bug","A-")); 
         items = index.query(querySpec); 
      } else if (indexName == "NameIndex") { 
         System.out.println("Compile error"); 
         querySpec.withKeyConditionExpression("Name = :v_name and begins_with
            (BugID, :v_bug)") 
            .withValueMap(new ValueMap() 
            .withString(":v_name","Compile error") 
            .withString(":v_bug","A-")); 
         items = index.query(querySpec); 
      } else if (indexName == "DueDateIndex") { 
         System.out.println("Items due on 2016-10-15"); 
         querySpec.withKeyConditionExpression("DueDate = :v_date") 
         .withValueMap(new ValueMap() 
         .withString(":v_date","2016-10-15")); 
         items = index.query(querySpec); 
      } else { 
         System.out.println("\nInvalid index name"); 
         return; 
      }  
      Iterator<Item> iterator = items.iterator(); 
      System.out.println("Query: getting result..."); 
      
      while (iterator.hasNext()) { 
         System.out.println(iterator.next().toJSONPretty()); 
      } 
   } 
}