DynamoDB-로컬 보조 인덱스
일부 응용 프로그램은 기본 키로 만 쿼리를 수행하지만 일부 상황에서는 대체 정렬 키의 이점을 얻습니다. 단일 또는 다중 local secondary index를 생성하여 애플리케이션이 선택할 수 있도록합니다.
수백만 개의 항목을 결합하는 것과 같은 복잡한 데이터 액세스 요구 사항으로 인해보다 효율적인 쿼리 / 스캔을 수행해야합니다. 로컬 보조 인덱스는 파티션 키 값에 대한 대체 정렬 키를 제공합니다. 또한 전체 또는 일부 테이블 속성의 사본을 보유합니다. 테이블 파티션 키로 데이터를 구성하지만 다른 정렬 키를 사용합니다.
local secondary index를 사용하면 전체 테이블 스캔이 필요 없으며 정렬 키를 사용하여 간단하고 빠른 쿼리가 가능합니다.
모든 local secondary index는 특정 조건을 충족해야합니다.
- 동일한 파티션 키와 소스 테이블 파티션 키.
- 스칼라 속성이 하나 뿐인 정렬 키입니다.
- 키가 아닌 속성으로 작동하는 소스 테이블 정렬 키의 프로젝션.
모든 local secondary index는 상위 테이블의 파티션 및 정렬 키를 자동으로 보유합니다. 쿼리에서 이것은 프로젝션 된 속성의 효율적인 수집과 프로젝션되지 않은 속성의 검색을 의미합니다.
local secondary index의 스토리지 제한은 모든 테이블 항목과 파티션 키 값을 공유하는 인덱스 항목을 포함하는 파티션 키 값당 10GB로 유지됩니다.
속성 투영
일부 작업은 복잡성으로 인해 과도한 읽기 / 가져 오기가 필요합니다. 이러한 작업은 상당한 처리량을 소비 할 수 있습니다. 프로젝션을 사용하면 이러한 속성을 분리하여 값 비싼 가져 오기를 피하고 풍부한 쿼리를 수행 할 수 있습니다. 프로젝션은 보조 인덱스에 복사 된 속성으로 구성됩니다.
보조 인덱스를 만들 때 프로젝션 된 속성을 지정합니다. DynamoDB에서 제공하는 세 가지 옵션을 생각해보십시오.KEYS_ONLY, INCLUDE, and ALL.
예측에서 특정 속성을 선택할 때 관련 비용 절충을 고려하십시오.
필요한 속성의 작은 세트 만 프로젝션하면 스토리지 비용을 크게 줄일 수 있습니다.
자주 액세스하는 키가 아닌 속성을 프로젝션하는 경우 스캔 비용을 스토리지 비용으로 상쇄합니다.
키가 아닌 속성의 대부분 또는 전부를 투영하는 경우 유연성이 극대화되고 처리량이 감소합니다 (검색 없음). 그러나 스토리지 비용은 증가합니다.
빈번한 쓰기 / 업데이트 및 빈번한 쿼리에 대해 KEYS_ONLY를 프로젝션하면 크기가 최소화되지만 쿼리 준비는 유지됩니다.
로컬 보조 인덱스 생성
사용 LocalSecondaryIndex단일 또는 다중 local secondary index를 만들기위한 CreateTable의 매개 변수. 정렬 키에 키가 아닌 속성을 하나 지정해야합니다. 테이블 생성시 로컬 보조 인덱스를 생성합니다. 삭제시 이러한 인덱스를 삭제합니다.
local secondary index가있는 테이블은 파티션 키 값당 크기 제한 인 10GB를 준수해야하지만 항목을 얼마든지 저장할 수 있습니다.
로컬 보조 인덱스 쿼리 및 스캔
로컬 보조 인덱스에 대한 쿼리 작업은 인덱스의 여러 항목이 정렬 키 값을 공유 할 때 파티션 키 값이 일치하는 모든 항목을 반환합니다. 일치하는 항목은 특정 순서로 반환되지 않습니다. local secondary index에 대한 쿼리는 최신 값을 제공하는 강력한 일관된 읽기와 함께 최종 또는 강력한 일관성을 사용합니다.
스캔 작업은 모든 local secondary index 데이터를 반환합니다. 스캔시 테이블 및 인덱스 이름을 제공해야하며 필터 표현식을 사용하여 데이터를 삭제할 수 있습니다.
항목 작성
local secondary index를 생성 할 때 정렬 키 속성과 해당 데이터 유형을 지정합니다. 항목을 쓸 때 항목이 인덱스 키의 속성을 정의하는 경우 해당 유형은 키 스키마의 데이터 유형과 일치해야합니다.
DynamoDB는 테이블 항목 및 local secondary index 항목에 일대일 관계 요구 사항을 부과하지 않습니다. local secondary index가 여러 개인 테이블은 적은 비용보다 쓰기 비용이 높습니다.
로컬 보조 인덱스의 처리량 고려 사항
쿼리의 읽기 용량 소비는 데이터 액세스의 특성에 따라 다릅니다. 쿼리는 최종 일관성 또는 강력한 일관성을 사용합니다. 하나의 단위를 사용하는 강력한 일관성 읽기가 최종 일관성 읽기의 절반 단위에 비해 비교됩니다.
결과 제한에는 최대 1MB 크기가 포함됩니다. 결과 크기는 일치하는 인덱스 항목 크기의 합계에서 가장 가까운 4KB로 반올림되고 일치하는 테이블 항목 크기도 가장 가까운 4KB로 반올림됩니다.
쓰기 용량 소비는 프로비저닝 된 단위 내에 남아 있습니다. 테이블 작성에서 사용 된 단위와 인덱스 업데이트에서 사용 된 단위의 합계를 찾아 총 프로비저닝 된 비용을 계산합니다.
비용에 영향을 미치는 주요 요소를 고려할 수도 있습니다. 그 중 일부는 다음과 같습니다.
인덱싱 된 속성을 정의하는 항목을 작성하거나 정의되지 않은 인덱싱 된 속성을 정의하기 위해 항목을 업데이트하면 단일 쓰기 작업이 발생합니다.
테이블 업데이트가 인덱싱 된 키 속성 값을 변경하면 두 번의 쓰기가 발생하여 삭제 한 다음 항목을 추가합니다.
쓰기로 인해 인덱싱 된 속성이 삭제되면 이전 항목 프로젝션을 제거하기 위해 하나의 쓰기가 발생합니다.
업데이트 전후에 인덱스 내에 항목이 없으면 쓰기가 발생하지 않습니다.
로컬 보조 인덱스 스토리지
테이블 항목 쓰기시 DynamoDB는 올바른 속성 세트를 필요한 로컬 보조 인덱스에 자동으로 복사합니다. 이것은 귀하의 계정에 청구됩니다. 사용 된 공간은 테이블 기본 키 바이트 크기, 인덱스 키 속성 바이트 크기, 현재 투영 된 속성 바이트 크기 및 각 인덱스 항목에 대한 오버 헤드 100 바이트의 합계에서 발생합니다.
예상 스토리지는 평균 인덱스 항목 크기를 추정하고 테이블 항목 수량을 곱하여 얻습니다.
Java를 사용하여 로컬 보조 인덱스 작업
먼저 DynamoDB 클래스 인스턴스를 생성하여 local secondary index를 생성합니다. 그런 다음 필요한 요청 정보로 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());
describe 메소드를 사용하여 local secondary index에 대한 정보를 검색합니다. DynamoDB 클래스 인스턴스를 생성하고 Table 클래스 인스턴스를 생성 한 다음 테이블을 describe 메서드에 전달하기 만하면됩니다.
예
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 클래스 인스턴스, Table 클래스 인스턴스, Index 클래스 인스턴스, 쿼리 객체를 생성하고 쿼리 메서드를 활용하기 만하면됩니다.
예
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− 다음 예제는 이전에 생성 된 데이터 소스를 가정 할 수 있습니다. 실행을 시도하기 전에 지원 라이브러리를 확보하고 필요한 데이터 소스 (필수 특성이있는 테이블 또는 기타 참조 소스)를 작성하십시오.
다음 예제는 또한 Eclipse IDE, AWS 자격 증명 파일 및 Eclipse AWS Java 프로젝트 내에서 AWS Toolkit을 사용합니다.
예
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());
}
}
}
}