How to get item count from DynamoDB?

前端 未结 12 2065
故里飘歌
故里飘歌 2020-12-04 17:15

I want to know item count with DynamoDB querying.

I can querying for DynamoDB, but I only want to know \'total count of item\'.

For example, \'SELECT COUNT(*

相关标签:
12条回答
  • 2020-12-04 17:44

    This is solution for AWS JavaScript SDK users, it is almost same for other languages.

    Result.data.Count will give you what you are looking for

     apigClient.getitemPost({}, body, {})
    
        .then(function(result){
    
            var dataoutput = result.data.Items[0];
    
            console.log(result.data.Count);
      }).catch( function(result){
    
    });
    
    0 讨论(0)
  • 2020-12-04 17:47

    You could use dynamodb mapper query.

    PaginatedQueryList<YourModel> list = DymamoDBMapper.query(YourModel.class, queryExpression);
    int count = list.size();
    

    it calls loadAllResults() that would lazily load next available result until allResultsLoaded.

    Ref: https://docs.amazonaws.cn/en_us/amazondynamodb/latest/developerguide/DynamoDBMapper.Methods.html#DynamoDBMapper.Methods.query

    0 讨论(0)
  • 2020-12-04 17:48

    You can use the Select parameter and use COUNT in the request. It "returns the number of matching items, rather than the matching items themselves". Important, as brought up by Saumitra R. Bhave in a comment, "If the size of the Query result set is larger than 1 MB, then ScannedCount and Count will represent only a partial count of the total items. You will need to perform multiple Query operations in order to retrieve all of the results".

    I'm Not familiar with PHP but here is how you could use it with Java. And then instead of using Count (which I am guessing is a function in PHP) on the 'Items' you can use the Count value from the response - $result['Count']:

    final String week = "whatever";
    final Integer myPoint = 1337;
    Condition weekCondition = new Condition()
            .withComparisonOperator(ComparisonOperator.EQ)
            .withAttributeValueList(new AttributeValue().withS(week));
    Condition myPointCondition = new Condition()
            .withComparisonOperator(ComparisonOperator.GE)
            .withAttributeValueList(new AttributeValue().withN(myPoint.toString()))
    
    Map<String, Condition> keyConditions = new HashMap<>();
    keyConditions.put("week", weekCondition);
    keyConditions.put("point", myPointCondition);
    
    QueryRequest request = new QueryRequest("game_table");
    request.setIndexName("week-point-index");
    request.setSelect(Select.COUNT);
    request.setKeyConditions(keyConditions);
    
    QueryResult result = dynamoDBClient.query(request);
    Integer count = result.getCount();
    

    If you don't need to emulate the WHERE clause, you can use a DescribeTable request and use the resulting item count to get an estimate.

    The number of items in the specified table. DynamoDB updates this value approximately every six hours. Recent changes might not be reflected in this value.

    Also, an important note from the documentation as noted by Saumitra R. Bhave in the comments on this answer:

    If the size of the Query result set is larger than 1 MB, ScannedCount and Count represent only a partial count of the total items. You need to perform multiple Query operations to retrieve all the results (see Paginating Table Query Results).

    0 讨论(0)
  • 2020-12-04 17:51

    I'm posting this answer for anyone using C# that wants a fully functional, well-tested answer that demonstrates using query instead of scan. In particular, this answer handles more than 1MB size of items to count.

            public async Task<int> GetAvailableCount(string pool_type, string pool_key)
        {
            var queryRequest = new QueryRequest
            {
                TableName = PoolsDb.TableName,
                ConsistentRead = true,
                Select = Select.COUNT,
                KeyConditionExpression = "pool_type_plus_pool_key = :type_plus_key",
                ExpressionAttributeValues = new Dictionary<string, AttributeValue> {
                    {":type_plus_key", new AttributeValue { S =  pool_type + pool_key }}
                },
            };
            var t0 = DateTime.UtcNow;
            var result = await Client.QueryAsync(queryRequest);
            var count = result.Count;
            var iter = 0;
            while ( result.LastEvaluatedKey != null && result.LastEvaluatedKey.Values.Count > 0) 
            {
                iter++;
                var lastkey = result.LastEvaluatedKey.Values.ToList()[0].S;
                _logger.LogDebug($"GetAvailableCount {pool_type}-{pool_key} iteration {iter} instance key {lastkey}");
                queryRequest.ExclusiveStartKey = result.LastEvaluatedKey;
                result = await Client.QueryAsync(queryRequest);
                count += result.Count;
            }
            _logger.LogDebug($"GetAvailableCount {pool_type}-{pool_key} returned {count} after {iter} iterations in {(DateTime.UtcNow - t0).TotalMilliseconds} ms.");
            return count;
        }
    }
    
    0 讨论(0)
  • 2020-12-04 17:57

    If you happen to reach here, and you are working with C#, here is the code:

    var cancellationToken = new CancellationToken();
    
    var request = new ScanRequest("TableName") {Select = Select.COUNT};
    
    var result = context.Client.ScanAsync(request, cancellationToken).Result;
    
    totalCount = result.Count;
    
    0 讨论(0)
  • 2020-12-04 17:58

    I used scan to get total count of the required tableName.Following is a Java code snippet for same

    Long totalItemCount = 0;
    do{
        ScanRequest req = new ScanRequest();
        req.setTableName(tableName);
    
        if(result != null){
            req.setExclusiveStartKey(result.getLastEvaluatedKey());
        }
    
        result = client.scan(req);
    
        totalItemCount += result.getItems().size();
    
    } while(result.getLastEvaluatedKey() != null);
    
    System.out.println("Result size: " + totalItemCount);
    
    0 讨论(0)
提交回复
热议问题