Ios NSDate读取为NSDate和NSCFString
我的后端是Parse.com。我让用户使用一组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 -[__
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 fromPFObject
“我知道我可以检查它是NSCFString还是NSDate并相应地处理它们,但我很想知道问题出在哪里。”谢谢您,我将使用您的代码:-)