关于dynamodb中query之于分页的做法
1、首先介绍一下我的数据字段。
表名是homeworks,hash键设置为teacher_id,range键设置为id(一个由时间戳和随机数组成的长字串)
2、应用场景是查询某一个老师布置的作业,query返回等操作被我包进handler.query中。我在这个函数中进行了报错的处理,但是正常的返回值就是dynamodb的返回值。
3、首先想到的是使用query来进行查询,参数如下,其中teacher_id = 1,limit = 5:
var reqParams = {
'TableName' : 'homeworks',
'IndexName' : 'teacher_id',
'Limit': para.limit,
'ProjectionExpression': "id, teacher_id",
'KeyConditionExpression': 'teacher_id = :v1',
'ExpressionAttributeValues': {
":v1": { "N": para.teacher_id },
},
}
if(para.offset != 0)
{
reqParams['ExclusiveStartKey'] = {
"teacher_id": { "N": para.teacher_id },
"id": { 'N': para.offset }
}
}
返回如下:
{"Items":[
{"id":{"N":"1494917803"},"teacher_id":{"N":"1"}},
{"id":{"N":"1494917858"},"teacher_id":{"N":"1"}},
{"id":{"N":"1494917915"},"teacher_id":{"N":"1"}},
{"id":{"N":"1494922586"},"teacher_id":{"N":"1"}},
{"id":{"N":"1494923540"},"teacher_id":{"N":"1"}}],
"Count":5,"ScannedCount":5,
"LastEvaluatedKey":{"id":{"N":"1494923540"},"teacher_id":{"N":"1"}}
}
4、然而当我想要加入一个名为delstatus = 0的条件时,将条件改为了:
'KeyConditionExpression': 'teacher_id = :v1 AND delstatus = :v3',
'ExpressionAttributeValues': {
":v1": { "N": para.teacher_id },
":v3": { "N": "0" }
},
发生了如下错误:
ValidationException: Query condition missed key schema element: id
我对此的理解是,需要查询除了两个主键以外的数值的时候,两个主键必须先被指定。
不过单独查询hash键也是可以的。
5、然后我尝试更换了一个teacher_id,获得数据如下:
{"Items":[
{"id":{"N":"14955108888001"},"teacher_id":{"N":"2"}},
{"id":{"N":"149551050797611"},"teacher_id":{"N":"2"}},
{"id":{"N":"149551232742291"},"teacher_id":{"N":"2"}},
{"id":{"N":"149551232910731"},"teacher_id":{"N":"2"}},
{"id":{"N":"149551233151301"},"teacher_id":{"N":"2"}}],
"Count":5,"ScannedCount":5
,"LastEvaluatedKey":{"id":{"N":"149551233151301"},"teacher_id":{"N":"2"}}
}
6、当我分别将offset替换为LastEvaluatedKey的返回值时,(3)和(5)也是一切正常的。
7、因为一定要指定主键值,所以我决定加入一个id > offset的条件:
'KeyConditionExpression': 'teacher_id = :v1 AND id > :v2 AND delstatus = :v3',
'ExpressionAttributeValues': {
":v1": { "N": para.teacher_id },
":v2": { "N": para.offset },
":v3": { "N": "0" }
},
执行后收到报错:
Conditions can be of length 1 or 2 only
这时候我才发现这个参数的名字是KeyConditionExpression,也就是只能查询key的,也就是说其他的value要用别的参数查询。
8、查询文档,看到有这样两个参数:
"FilterExpression": "string",
"ExpressionAttributeValues": {
"string" : {
"B": blob,
"BOOL": boolean,
"BS": [ blob ],
"L": [
"AttributeValue"
],
"M": {
"string" : "AttributeValue"
},
"N": "string",
"NS": [ "string" ],
"NULL": boolean,
"S": "string",
"SS": [ "string" ]
}
},
于是将参数更改为:
'KeyConditionExpression': 'teacher_id = :v1',
"FilterExpression": "delstatus = :v3",
'ExpressionAttributeValues': {
":v1": { "N": para.teacher_id },
":v3": { "N": '0' }
},
但所获得的却是:
{"Items":[],
"Count":0,"ScannedCount":5,
"LastEvaluatedKey":{"id":{"N":"1494923540"},"teacher_id":{"N":"1"}}
}
对于这个我产生了一些疑惑,明明没有查询到数据为什么还有LastEvaluatedKey。
后来我稍微想明白了一点,就是这个filter和limit的意思。
是先limit出来5条,然后再在这5条中进行filter,因为原先limit出来的5条里没有teacher_id=1的,所以自然是空的。
我试着将teacher_id改成了2,items里就有内容了。
但是这个参数依旧不是我想要的效果。
9、我怀疑过参数的排序,但是即使将limit参数放到最后也没有影响,可见是没有关系的。
网友评论