Ios NSDate读取为NSDate和NSCFString

Ios NSDate读取为NSDate和NSCFString,ios,objective-c,parse-platform,ios-simulator,nsdate,Ios,Objective C,Parse Platform,Ios Simulator,Nsdate,我的后端是Parse.com。我让用户使用一组NSDate对象作为完成某些操作的时间戳 我试图读取该数组中的最后一个对象,以测试它是否在一天的前3个小时内完成 我遇到的奇怪问题如下。在我的物理设备上,我加载假定的NSDate对象数组,并将对象和对象类记录到控制台。我得到以下信息: OBJECT: 2016-01-19T01:09:18.239Z CLASS: __NSCFString OBJECT: 2016-01-20 05:50:34 +0000 CLASS: __NSDate -[__

我的后端是Parse.com。我让用户使用一组
NSDate
对象作为完成某些操作的时间戳

我试图读取该数组中的最后一个对象,以测试它是否在一天的前3个小时内完成

我遇到的奇怪问题如下。在我的物理设备上,我加载假定的
NSDate
对象数组,并将对象和对象类记录到控制台。我得到以下信息:

OBJECT: 2016-01-19T01:09:18.239Z
CLASS: __NSCFString
OBJECT: 2016-01-20 05:50:34 +0000
CLASS: __NSDate
-[__NSDate length]: unrecognized selector sent to instance 0x7ffcf3f9c480
因为它不是一个
NSDate
对象,所以我使用格式化程序将其设置为一个对象并进行计算。到目前为止还没有撞车

当我在模拟器上运行相同的代码时,我得到以下结果:

OBJECT: 2016-01-19T01:09:18.239Z
CLASS: __NSCFString
OBJECT: 2016-01-20 05:50:34 +0000
CLASS: __NSDate
-[__NSDate length]: unrecognized selector sent to instance 0x7ffcf3f9c480
因此,应用程序崩溃(它应该崩溃),因为我正试图使用格式化程序将一个已经
NSDate
的对象从一个非字符串变成
NSDate
。崩溃日志显示以下内容:

OBJECT: 2016-01-19T01:09:18.239Z
CLASS: __NSCFString
OBJECT: 2016-01-20 05:50:34 +0000
CLASS: __NSDate
-[__NSDate length]: unrecognized selector sent to instance 0x7ffcf3f9c480
所以我想也许这些物体实际上是不同的。我去Parse.com复制对象。它们是一样的:

["2015-12-20T17:47:38.134Z"]
["2016-01-20T05:50:34.253Z"]

我知道我可以检查它是
NSCFString
还是
NSDate
并相应地处理它们,但我很想知道问题出在哪里。感谢您的帮助。我整天都在做这件事。干杯

编辑:在我的物理设备上再运行几次之后。一些对象将显示为
NSCFString
,其他对象将显示为
NSDate
。再次检查Parse.com以确保所有对象都相同,再次检查true

["2016-01-20T06:07:41.435Z"]

["2016-01-20T06:07:39.454Z"]

["2016-01-20T04:53:53.419Z"]

这里有两种方法可以帮助您处理此类问题。第一个返回NSDate对象,第二个返回字符串日期格式。您可以使用其中一个或两者

- (NSDate*) getDateFromObject:(id) object{
if ([object isKindOfClass:[NSDate class]]) {
    return (NSDate*)object;
}
else if ([object isKindOfClass:NSString.class]){
    NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
    NSDate *date = [dateFormatter dateFromString:object];

    return date;
}
// return nil otherwise
return nil;
}

- (NSString*) getStringFromObject:(id) object{
if ([object isKindOfClass:NSString.class]) {
    return (NSString*)object;
}
else if ([object isKindOfClass:NSDate.class]){
    NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
    NSString *dateString = [dateFormatter stringFromDate:object];

    return dateString;
}
// return nil otherwise
return nil;

}我在这里面临同样的问题。我的情况是,我正在接收一个JSON对象数据,并将其序列化为具有NSDate值的对象。。为此,您只需使用少量手动编码将文本作为NSDate而不是web服务中的普通字符串传递:

示例(显示错误):

正确的一点:

- (void)userFromJson:(NSData *)object error:(NSError **)error {
    NSError *localError = nil;
    NSDictionary *parsedObject = [NSJSONSerialization JSONObjectWithData:object options:0 error:&localError];

    if (localError != nil) {
        *error = localError;
        return;
    }

    for (NSDictionary *keyDict in parsedObject) {
        for (NSString *key in keyDict) {
            if ([self respondsToSelector:NSSelectorFromString(key)]) {
                if ([key isEqualToString:@"dateOfBirth"]) {
                    NSDateFormatter *df = [[NSDateFormatter alloc] init];
                    [df setDateFormat:@"yyyy-mm-dd"];
                    NSDate *d = [df dateFromString:[keyDict valueForKey:key]];
                    [self setValue:d forKey:key];
                }
                else {
                    [self setValue:[keyDict valueForKey:key] forKey:key];
                }
            }
        }
    }
}
这对我来说现在是可行的,但我正试图找到一种方法,以避免硬编码可能的变量名(你知道…模块编码)

希望这对你有帮助

编辑: 经过一番努力,我得出了以下解决方案:

在你的相关课程中,你必须导入。。。然后,您可以检查本地对象参数类型并相应地执行操作:

- (void)userFromJson:(NSData *)object error:(NSError **)error {
    NSError *localError = nil;
    NSDictionary *parsedObject = [NSJSONSerialization JSONObjectWithData:object options:0 error:&localError];

    if (localError != nil) {
        *error = localError;
        return;
    }

    for (NSDictionary *keyDict in parsedObject) {
        for (NSString *key in keyDict) {
            if ([self respondsToSelector:NSSelectorFromString(key)]) {

                objc_property_t property = class_getProperty([self class], [key UTF8String]);

                if ([[self getClassOfProperty:property] isSubclassOfClass:[NSNumber class]]) {
                    NSScanner *scanner = [NSScanner scannerWithString:[keyDict valueForKey:key]];
                    if ([scanner scanFloat:NULL]) {
                        [self setValue:@([[keyDict valueForKey:key] floatValue]) forKey:key];
                    }
                    else if ([scanner scanInt:NULL]) {
                        [self setValue:@([[keyDict valueForKey:key] integerValue]) forKey:key];
                    }
                    else {

                    }
                }
                else if ([[self getClassOfProperty:property] isSubclassOfClass:[NSDate class]]) {
                    NSDateFormatter *df = [[NSDateFormatter alloc] init];
                    NSString *stDate = (NSString *)[keyDict valueForKey:key];
                    if ([stDate length] == 10) {
                        [df setDateFormat:@"yyyy-MM-dd"];
                    }
                    else {
                        [df setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
                    }

                    NSDate *d = [df dateFromString:stDate];
                    [self setValue:d forKey:key];
                }
                else if ([[self getClassOfProperty:property] isSubclassOfClass:[NSString class]]) {
                    [self setValue:[keyDict valueForKey:key] forKey:key];
                }
                else if ([[self getClassOfProperty:property] isSubclassOfClass:[NSData class]]) {
                    [self setValue:[NSData dataWithBytes:[[keyDict valueForKey:key] UTF8String] length:[[keyDict valueForKey:key] length]]
                         forKey:key];
                }
                else {

                }
            }
        }
    }
}

-(Class)getClassOfProperty:(objc_property_t)property {
    NSString *objType = [NSString stringWithUTF8String:property_getAttributes(property)];
    NSUInteger start =  [objType rangeOfString:@"\""].location;
    NSUInteger end =  [objType rangeOfString:@"\","].location;
    NSRange range;
    range.location = start + 1;
    range.length = end - range.location;
    NSString *objType2 = [objType substringWithRange:range];

    return [NSClassFromString(objType2) class];
}

现在,为了更加通用,您可以使用
NSDataDetector
,而不是使用特定的格式,请参阅

您可以使用isKindOfClass来检查对象是nsstring还是nsdate,并相应地检查。我想您首先要检查的是值是nsdate对象吗?为此,这里是帮助您的链接。你能看看这个答案,看看这是否有帮助。我认为他们发送的对象是自定义解析对象。您是从PFObject还是PFObject子类访问日期?@kRiZ from
PFObject
“我知道我可以检查它是NSCFString还是NSDate并相应地处理它们,但我很想知道问题出在哪里。”谢谢您,我将使用您的代码:-)