Разница в датах между UIDatepicker и NSDate

У меня есть датапикер, и я сохраняю значение в файл .plist с ключом UserDate в строковом формате.

Значение, возвращаемое из datepicker, имеет формат 2013-02-13 11:17:34 +0000.

Чтобы сохранить его в файл .plist, мне нужно преобразовать datepicker в строку 2013-02-14T19:17:34Z

В другом представлении я хочу получить обратно UserDate из plist и сравнить с текущей датой пользователя. Я хочу другие дни и часы.

Я конвертирую и передаю 2012-02-12 19:17:34 +0800 как destinationDate для сравнения с [NSDate new]. Мне удалось вычислить разницу между двумя датами. Но результат неверный.

Как получить правильный результат разницы для двух дат между датой iPhone и датой Datepicker? Мне нужно использовать определенный часовой пояс?

Дополнительная информация: я живу в GMT + 8. Я установил датупикер симулятора на 13 февраля 2012 года, но результат по-прежнему показывает, что обратный отсчет составляет 18 часов 4 минуты.

NSCalendar *calendar = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar];
int units = NSMonthCalendarUnit | NSDayCalendarUnit | NSHourCalendarUnit | NSMinuteCalendarUnit | NSSecondCalendarUnit;

NSDateComponents *components = [calendar components:units fromDate:[NSDate date] toDate:destinationDate options:0];
[datelabel setText:[NSString stringWithFormat:@"%d%c %d%c %d%c %d%c %d%c", [components month], 'm', [components day], 
                    'd', [components hour], 'h', [components minute], 'm', [components second], 's']];

NSLog

введите описание изображения здесь

Изменить

Plist файл.

введите описание изображения здесь

метод сохранения

NSArray *paths = NSSearchPathForDirectoriesInDomains (NSDocumentDirectory, NSUserDomainMask, YES);
// get documents path
NSString *documentsPath = [paths objectAtIndex:0];
// get the path to our Data/plist file
NSString *plistPath = [documentsPath stringByAppendingPathComponent:@"Data.plist"];

// set the variables to the values in the text fields
self.userTitle = myTitle.text;
self.userMessage = myMessage.text;  

NSLog(@"METHOD SAVE");

NSLog(@"Datepicker value =  %@", myDate.date );

NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
[dateFormatter setFormatterBehavior:NSDateFormatterBehavior10_4];
[dateFormatter setDateFormat:@"yyyy-MM-dd'T'HH:mm:ss'Z'"];
NSString *mdate = [dateFormatter stringFromDate:myDate.date];
//  [dateFormatter setDateFormat:@"yyyy-MM-dd'T'HH:mm:ss'Z'"];

NSLog(@"Converted from date picker, save into plist =  %@", mdate );

self.saveDate = mdate;
//self.userDate = myDate.date;

// create dictionary with values in UITextFields
NSDictionary *plistDict = [NSDictionary dictionaryWithObjects: [NSArray arrayWithObjects: userTitle, userMessage, saveDate, nil] forKeys:[NSArray arrayWithObjects: @"Title", @"Message", @"UserDate", nil]]; NSString *error = nil;
// create NSData from dictionary
NSData *plistData = [NSPropertyListSerialization dataFromPropertyList:plistDict format:NSPropertyListXMLFormat_v1_0 errorDescription:&error];

// check is plistData exists
if(plistData)
{
    // write plistData to our Data.plist file
    [plistData writeToFile:plistPath atomically:YES];
    
    UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Success" message:@"Your settings have been saved." delegate:self cancelButtonTitle:nil otherButtonTitles:@"Ok",nil];
    [alert show];
}
else
{
    NSLog(@"Error in saveData: %@", error);
    //[error release];
}

viewWillAppear

- (void)viewWillAppear:(BOOL)animated {
[self.navigationController setNavigationBarHidden:YES];

[super viewWillAppear:animated];

// Data.plist code
// get paths from root direcory
NSArray *paths = NSSearchPathForDirectoriesInDomains (NSDocumentDirectory, NSUserDomainMask, YES);
// get documents path
NSString *documentsPath = [paths objectAtIndex:0];
// get the path to our Data/plist file
NSString *plistPath = [documentsPath stringByAppendingPathComponent:@"Data.plist"];

// check to see if Data.plist exists in documents
if (![[NSFileManager defaultManager] fileExistsAtPath:plistPath])
{
    // if not in documents, get property list from main bundle
    plistPath = [[NSBundle mainBundle] pathForResource:@"Data" ofType:@"plist"];
}

// read property list into memory as an NSData object
NSData *plistXML = [[NSFileManager defaultManager] contentsAtPath:plistPath];
NSString *errorDesc = nil;
NSPropertyListFormat format;
// convert static property liost into dictionary object
NSDictionary *temp = (NSDictionary *)[NSPropertyListSerialization propertyListFromData:plistXML mutabilityOption:NSPropertyListMutableContainersAndLeaves format:&format errorDescription:&errorDesc];
if (!temp)
{
    NSLog(@"Error reading plist: %@, format: %d", errorDesc, format);
}

NSLog(@"Plist Title: %@", [temp objectForKey:@"Title"]);
NSLog(@"Plist Message: %@", [temp objectForKey:@"Message"]);
NSLog(@"Plist date: %@", [temp objectForKey:@"UserDate"]);

userTitle.text = [NSString localizedStringWithFormat:@"%@", [temp objectForKey:@"Title"]];
userMessage.text = [NSString localizedStringWithFormat:@"%@", [temp objectForKey:@"Message"]];

//http://stackoverflow.com/questions/1070354/how-do-i-get-the-current-date-in-cocoa

NSDate* now = [[NSDate alloc] initWithTimeIntervalSinceNow:3600];
userCoundown.text = [NSString stringWithFormat:@"%@",now];  

NSString *uDate = [[NSString alloc] init];
uDate = [temp objectForKey:@"UserDate"];
NSLog(@"Plist date in string - %@", uDate);

NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
[dateFormatter setFormatterBehavior:NSDateFormatterBehavior10_4];
[dateFormatter setDateFormat:@"yyyy-MM-dd'T'HH:mm:ss'Z'"];
NSDate *ddate = [dateFormatter dateFromString:uDate];
[dateFormatter setDateFormat:@"yyyy-MM-dd H:mm:ss Z"];

NSLog(@"Format date from Plist -  %@", [dateFormatter stringFromDate:ddate]);

//NSLog(@"GMT format from Plist - %@", timeStamp);

//http://www.epochconverter.com/#
destinationDate = ddate;
timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(updateLabel) userInfo:nil repeats:YES];
}

updatelabel

-(void)updateLabel {

NSCalendar *calendar = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar];
int units = NSMonthCalendarUnit | NSDayCalendarUnit | NSHourCalendarUnit | NSMinuteCalendarUnit | NSSecondCalendarUnit;

NSDateComponents *components = [calendar components:units fromDate:[NSDate date] toDate:destinationDate options:0];
[datelabel setText:[NSString stringWithFormat:@"%d%c %d%c %d%c %d%c %d%c", [components month], 'm', [components day], 
                    'd', [components hour], 'h', [components minute], 'm', [components second], 's']];
}

person neotorama    schedule 12.02.2012    source источник
comment
Вы можете сохранить NSDate прямо в список, он будет автоматически преобразован в строковый формат ISO 8601. При чтении списка NSData будет преобразовано обратно в NSDate.   -  person zaph    schedule 12.02.2012
comment
Вы должны показать нам больше вашего кода; в идеале целая последовательность операторов, воспроизводящих проблему, а не только ее небольшую часть, и особенно код, который вы используете для 1) преобразования NSDate в строку и сохранения ее в списке 2) извлечения даты из списка и превратить это в свидание.   -  person yuji    schedule 12.02.2012
comment
верхний комментарий правильный .. но неполный ..: P .. вы можете сохранить массив, словарь, данные, строку, целые числа и т. д. непосредственно в файл plist .. просто создайте массив с объектом NSDate и сделайте из него plist .. . затем позже получите plist .. получите массив .. и вы получите соответствующую дату в его objectAtIndex: 0 .. затем сравните ..   -  person Shubhank    schedule 12.02.2012
comment
Извините. Я добавил код. Спасибо   -  person neotorama    schedule 12.02.2012


Ответы (2)


NSDate хранит время в GMT. Если у вас есть часовые пояса, вам нужно будет их указать.

Вопрос не ясен, поэтому это не полный ответ, но он должен помочь с NSDate в plist.

Вот пример создания, записи и чтения plist с NSDate:

NSDictionary *plistDict = [NSDictionary dictionaryWithObjectsAndKeys:
                           userTitle,   @"Title",
                           userMessage, @"Message",
                           saveDate,    @"UserDate",
                           nil];

[plistDict writeToFile:filePath atomically:YES];

NSDictionary *plistDictRead = [NSDictionary dictionaryWithContentsOfFile:filePath];
NSDate *dateRead = [plistDictRead objectForKey:@"UserDate"];

Нет необходимости преобразовывать NSDate в строку и использовать NSPropertyListSerialization.

person zaph    schedule 12.02.2012
comment
Я отредактировал вопрос. Что мне нужно, так это правильный расчет результата между моим datepicker и датой iphone. Сегодня 13 февраля 2012 года, я установил дату для симулятора на 13 февраля 2012 года, но результат по-прежнему показывает, что обратный отсчет составляет 18 часов 4 минуты. Извините за недопонимание. - person neotorama; 12.02.2012

Не забудьте создать NSDate, а затем вывести его с действительным часовым поясом и календарем!

Проверьте аналогичный вопрос, на который я ответил в другом сообщении: UIDatePicker возвращает неверный NSDate

person Carlos Morales    schedule 09.08.2012