1

首先,我想用 DynamoDB 实现什么:

-> (我试图在登录系统的扫描过滤器中包含用户名和密码。因此,如果对于电子邮件/用户和密码组合,返回对象的scan方法AWSDynamoDBObjectMapper,登录成功,否则不会。)

怎么了?

为此,我一直在尝试阅读Github上的文档和示例代码,但似乎没有一个正确使用它的示例。从文档中我只发现它scanFilter是 NSDictionary 类型的。因此,我尝试包含多个用于过滤的参数 ( UserName/UserEmail and Password),但它无法扫描结果。

 scanExpression.scanFilter = @{@"Password": passwordCondition, UserNameOrUserEmail : usernameOrEmailCondition};

但是,如果我只传递一个参数,它会起作用并返回一个对象,或者说是数据库的一行。

scanExpression.scanFilter = @{@"Password": passwordCondition};

完整代码供参考(当我提到正在做 Android 的同事时,Android 的类似代码可以正常工作。):

// Check if email or username and decide DynamoDB Attribute accordingly.
NSString *UserNameOrUserEmail;   
if ([usernameOrEmail rangeOfString:@"@"].location == NSNotFound) 
    UserNameOrUserEmail = @"UserName";
else
    UserNameOrUserEmail = @"UserEmail";

AWSDynamoDBScanExpression *scanExpression = [AWSDynamoDBScanExpression new];

AWSDynamoDBCondition *passwordCondition = [AWSDynamoDBCondition new];
AWSDynamoDBAttributeValue *passwordAttribute = [AWSDynamoDBAttributeValue new];
passwordAttribute.S = password;
passwordCondition.attributeValueList = @[passwordAttribute];
passwordCondition.comparisonOperator = AWSDynamoDBComparisonOperatorEQ;

AWSDynamoDBCondition *usernameOrEmailCondition = [AWSDynamoDBCondition new];
AWSDynamoDBAttributeValue *usernameOrEmailAttribute = [AWSDynamoDBAttributeValue new];
usernameOrEmailAttribute.S = @"sammy@hhd.com";//usernameOrEmail;
usernameOrEmailCondition.attributeValueList = @[usernameOrEmailAttribute];
usernameOrEmailCondition.comparisonOperator = AWSDynamoDBComparisonOperatorEQ;

//Apply filter condtions: http://stackoverflow.com/a/27856299/818506
scanExpression.scanFilter = @{@"Password": passwordCondition, UserNameOrUserEmail : usernameOrEmailCondition}; //@{@"UserEmail": usernameOrEmailCondition};//

 //^^^^^^^^^^^^^^^^^^^^^^^^^^^Is this correct?^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^    

//Perform a scan:
[[dynamoDBObjectMapper scan:[User class] expression:scanExpression]continueWithBlock: ^id (AWSTask *task){

    if (task.error) {
        NSLog(@"The request failed. Error: [%@]", task.error);
    }
    if (task.exception) {
        NSLog(@"The request failed. Exception: [%@]", task.exception);
    }
    if (task.result) {
        AWSDynamoDBPaginatedOutput *paginatedOutput = task.result;
        for (User *user in paginatedOutput.items) {
            isLoginSuccessful = YES;
         }
        completionHandler(isLoginSuccessful);
    }
    return nil;
}];

任何建议/想法/评论将不胜感激。

谢谢,罗汉。

4

2 回答 2

1

您是否考虑过使用AWSDynamoDBScanInput和 a NSMutableDictionary?我在 Swift 中有以下示例,应该很容易转换为 Objective-C:

    // create password condition
    let passwordCondition = AWSDynamoDBCondition()
    let v1    = AWSDynamoDBAttributeValue(); v1.S = password
    passwordCondition.comparisonOperator = AWSDynamoDBComparisonOperator.EQ
    passwordCondition.attributeValueList = [ v1 ]

    // create username condition        
    let usernameCondition = AWSDynamoDBCondition()
    let v2    = AWSDynamoDBAttributeValue(); v2.S = "sammy@hhd.com"
    usernameCondition.comparisonOperator = AWSDynamoDBComparisonOperator.EQ
    usernameCondition.attributeValueList = [ v2 ]

    // add these conditions to a NSMutableDictionary
    var scanFilterDict : NSMutableDictionary! = NSMutableDictionary()
    scanFilterDict.addEntriesFromDictionary(["password": passwordCondition])
    scanFilterDict.addEntriesFromDictionary(["UserNameOrUserEmail": usernameCondition])

    // use the mutable dictionary as scan filter
    scanInput.scanFilter = scanFilterDict as Dictionary<NSObject, AnyObject>
    scanInput.exclusiveStartKey = self.lastEvaluatedKey
    scanInput.limit = 20
    scanInput.tableName = "yourDynamoDBTable"

    // apply AND operator on both conditions
    scanInput.conditionalOperator = AWSDynamoDBConditionalOperator.And

    AWSDynamoDB.defaultDynamoDB().scan(input)
于 2015-09-10T07:40:54.443 回答
0

我在处理类似问题时遇到了这个问题。这是我解决它的方法:

    let objectMapper = AWSDynamoDBObjectMapper.default()
    let scanExpression = AWSDynamoDBScanExpression()

    scanExpression.filterExpression = "#UserNameOrUserEmail = :UserNameOrUserEmail AND #password = :follows"
    scanExpression.expressionAttributeNames = ["#UserNameOrUserEmail": "UserNameOrUserEmail" , "#password": "password" ,]
    scanExpression.expressionAttributeValues = [":UserNameOrUserEmail": "sammy@hhd.com", ":password": password ,]

    objectMapper.scan(User.self, expression: scanExpression) { (response: AWSDynamoDBPaginatedOutput?, error: Error?) in
        // handle data or error here...
    }

只是它可以帮助别人。

于 2017-08-23T15:40:59.980 回答