Come determinare se un NSDate è oggi?
-
22-09-2019 - |
Domanda
Come verificare se un NSDate
appartiene a oggi?
Ho usato per controllare utilizzando primi 10 caratteri di [aDate description]
. [[aDate description] substringToIndex:10]
restituisce la stringa "YYYY-MM-DD"
quindi l'ho confrontato la stringa con la stringa restituita da [[[NSDate date] description] substringToIndex:10]
.
C'è di più veloce e/o modo efficace per controllare?
Grazie.
Soluzione
in MacOS 10.9+ e iOS 8+, c'è un metodo su NSCalendar / calendario che fa esattamente questo!
- (BOOL)isDateInToday:(NSDate *)date
Quindi, si sarebbe semplicemente fare
Objective-C:
BOOL today = [[NSCalendar currentCalendar] isDateInToday:date];
Swift 3:
let today = Calendar.current.isDateInToday(date)
Altri suggerimenti
È possibile confrontare la data di componenti:
NSDateComponents *otherDay = [[NSCalendar currentCalendar] components:NSCalendarUnitEra | NSCalendarUnitYear | NSCalendarUnitMonth | NSCalendarUnitDay fromDate:aDate];
NSDateComponents *today = [[NSCalendar currentCalendar] components:NSCalendarUnitEra | NSCalendarUnitYear | NSCalendarUnitMonth | NSCalendarUnitDay fromDate:[NSDate date]];
if([today day] == [otherDay day] &&
[today month] == [otherDay month] &&
[today year] == [otherDay year] &&
[today era] == [otherDay era]) {
//do stuff
}
Edit:
Mi piace stefan metodo di più, penso che la rende più pulita e comprensibile se l'istruzione:
NSCalendar *cal = [NSCalendar currentCalendar];
NSDateComponents *components = [cal components:(NSCalendarUnitEra | NSCalendarUnitYear | NSCalendarUnitMonth | NSCalendarUnitDay) fromDate:[NSDate date]];
NSDate *today = [cal dateFromComponents:components];
components = [cal components:(NSCalendarUnitEra | NSCalendarUnitYear | NSCalendarUnitMonth | NSCalendarUnitDay) fromDate:aDate];
NSDate *otherDate = [cal dateFromComponents:components];
if([today isEqualToDate:otherDate]) {
//do stuff
}
Chris, ho incorporato il tuo suggerimento.Ho dovuto cercare ciò che era stato, quindi, per chiunque altro che non conosce, distingue tra AC e DC.Questo è probabilmente inutile per la maggior parte delle persone, ma è facile controllare e aggiunge un po ' di certezza, così ho incluso.Se si sta andando per la velocità, questo probabilmente non è un buon metodo comunque.
NOTA come con molte risposte COSÌ, dopo 7 anni, questo è completamente fuori di data.In Swift ora basta usare .isDateInToday
Questo è un ramo alla tua domanda, ma se si desidera stampare un NSDate con "Oggi" o "Yesterday", utilizzare la funzione
- (void)setDoesRelativeDateFormatting:(BOOL)b
per NSDateFormatter
Vorrei cercare di ottenere la data odierna normalizzato a mezzanotte e la seconda data, normalizzare a mezzanotte quindi confrontare se è lo stesso NSDate.
Da un esempio di Apple ecco come si normalizzano a mezzanotte data odierna, lo stesso per la seconda data e confrontare:
NSCalendar * gregorian = [[NSCalendar alloc]
initWithCalendarIdentifier:NSGregorianCalendar];
NSDateComponents * components =
[gregorian components:
(NSYearCalendarUnit|NSMonthCalendarUnit|NSDayCalendarUnit)
fromDate:[NSDate date]];
NSDate * today = [gregorian dateFromComponents:components];
Facendo Swift 3 & 4 estensione del suggerimento di Catfish_Man:
extension Date {
func isToday() -> Bool {
return Calendar.current.isDateInToday(self)
}
}
Non c'è bisogno di destreggiarsi con i componenti, epoche e roba.
NSCalendar fornisce un metodo per ottenere l'inizio di una certa unità di tempo per una data già esistente.
Questo codice otterrà l'inizio di oggi e un'altra data e confrontare questo. Se si restituisce NSOrderedSame
, sono entrambe le date durante lo stesso giorno - così oggi
NSDate *today = nil;
NSDate *beginningOfOtherDate = nil;
NSDate *now = [NSDate date];
[[NSCalendar currentCalendar] rangeOfUnit:NSDayCalendarUnit startDate:&today interval:NULL forDate:now];
[[NSCalendar currentCalendar] rangeOfUnit:NSDayCalendarUnit startDate:&beginningOfOtherDate interval:NULL forDate:beginningOfOtherDate];
if([today compare:beginningOfOtherDate] == NSOrderedSame) {
//otherDate is a date in the current day
}
extension NSDate {
func isToday() -> Bool {
let cal = NSCalendar.currentCalendar()
var components = cal.components([.Era, .Year, .Month, .Day], fromDate:NSDate())
let today = cal.dateFromComponents(components)!
components = cal.components([.Era, .Year, .Month, .Day], fromDate:self)
let otherDate = cal.dateFromComponents(components)!
return today.isEqualToDate(otherDate)
}
Ha lavorato per me su Swift 2.0
Versione Swift della migliore risposta:
let cal = NSCalendar.currentCalendar()
var components = cal.components([.Era, .Year, .Month, .Day], fromDate:NSDate())
let today = cal.dateFromComponents(components)!
components = cal.components([.Era, .Year, .Month, .Day], fromDate:aDate);
let otherDate = cal.dateFromComponents(components)!
if(today.isEqualToDate(otherDate)) {
//do stuff
}
Fare riferimento alla voce di documentazione di Apple dal titolo "Esecuzione Calendario Calcoli" [link] .
Listing 13 in quella pagina suggerisce che per determinare il numero di midnights tra giorni, si utilizza:
- (NSInteger)midnightsFromDate:(NSDate *)startDate toDate:(NSDate *)endDate
{
NSCalendar *calendar = [NSCalendar autoupdatingCurrentCalendar];
NSInteger startDay = [calendar ordinalityOfUnit:NSDayCalendarUnit
inUnit:NSEraCalendarUnit
forDate:startDate];
NSInteger endDay = [calendar ordinalityOfUnit:NSDayCalendarUnit
inUnit:NSEraCalendarUnit
forDate:endDate];
return endDay - startDay;
}
Si può quindi determinare se due giorni sono gli stessi utilizzando tale metodo e vedere se restituisce 0 o meno.
Si potrebbe anche verificare l'intervallo di tempo tra la data che hai, e la data corrente:
[myDate timeIntervalSinceNow]
Questo vi darà l'intervallo di tempo, in secondi, tra myDate e la data / ora.
link .
Modifica: Nota a tutti:. Sono ben consapevole del fatto che [myDate timeIntervalSinceNow] non senza ambiguità determinare se myDate oggi
Lascio questa risposta come è così che se qualcuno è alla ricerca di qualcosa di simile e [myDate timeIntervalSinceNow] è utile, possono trovare qui.
Swift estensione sulla base delle migliori risposte:
extension NSDate {
func isToday() -> Bool {
let cal = NSCalendar.currentCalendar()
if cal.respondsToSelector("isDateInToday:") {
return cal.isDateInToday(self)
}
var components = cal.components((.CalendarUnitEra | .CalendarUnitYear | .CalendarUnitMonth | .CalendarUnitDay), fromDate:NSDate())
let today = cal.dateFromComponents(components)!
components = cal.components((.CalendarUnitEra | .CalendarUnitYear | .CalendarUnitMonth | .CalendarUnitDay), fromDate:self);
let otherDate = cal.dateFromComponents(components)!
return today.isEqualToDate(otherDate)
}
}
Se si dispone di un sacco di questi confronti data, quindi le chiamate per calendar:components:fromDate
iniziare a prendere un sacco di tempo. Secondo alcuni profilazione che ho fatto, sembrano essere piuttosto costosi.
Di 'si sta cercando di determinare quale da alcune serie di date, dire NSArray *datesToCompare
, sono lo stesso giorno di un dato giorno, dicono NSDate *baseDate
, allora si può usare qualcosa di simile a quanto segue (in parte ricavato da una risposta di cui sopra):
NSDate *baseDate = [NSDate date];
NSArray *datesToCompare = [NSArray arrayWithObjects:[NSDate date],
[NSDate dateWithTimeIntervalSinceNow:100],
[NSDate dateWithTimeIntervalSinceNow:1000],
[NSDate dateWithTimeIntervalSinceNow:-10000],
[NSDate dateWithTimeIntervalSinceNow:100000],
[NSDate dateWithTimeIntervalSinceNow:1000000],
[NSDate dateWithTimeIntervalSinceNow:50],
nil];
// determine the NSDate for midnight of the base date:
NSCalendar* calendar = [NSCalendar currentCalendar];
NSDateComponents* comps = [calendar components:(NSYearCalendarUnit|NSMonthCalendarUnit|NSDayCalendarUnit)
fromDate:baseDate];
NSDate* theMidnightHour = [calendar dateFromComponents:comps];
// set up a localized date formatter so we can see the answers are right!
NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
[dateFormatter setDateStyle:NSDateFormatterMediumStyle];
[dateFormatter setTimeStyle:NSDateFormatterMediumStyle];
// determine which dates in an array are on the same day as the base date:
for (NSDate *date in datesToCompare) {
NSTimeInterval interval = [date timeIntervalSinceDate:theMidnightHour];
if (interval >= 0 && interval < 60*60*24) {
NSLog(@"%@ is on the same day as %@", [dateFormatter stringFromDate:date], [dateFormatter stringFromDate:baseDate]);
}
else {
NSLog(@"%@ is NOT on the same day as %@", [dateFormatter stringFromDate:date], [dateFormatter stringFromDate:baseDate]);
}
}
Output:
Nov 23, 2011 1:32:00 PM is on the same day as Nov 23, 2011 1:32:00 PM
Nov 23, 2011 1:33:40 PM is on the same day as Nov 23, 2011 1:32:00 PM
Nov 23, 2011 1:48:40 PM is on the same day as Nov 23, 2011 1:32:00 PM
Nov 23, 2011 10:45:20 AM is on the same day as Nov 23, 2011 1:32:00 PM
Nov 24, 2011 5:18:40 PM is NOT on the same day as Nov 23, 2011 1:32:00 PM
Dec 5, 2011 3:18:40 AM is NOT on the same day as Nov 23, 2011 1:32:00 PM
Nov 23, 2011 1:32:50 PM is on the same day as Nov 23, 2011 1:32:00 PM
C'è un modo più semplice rispetto a molte delle risposte di cui sopra!
NSDate *date = ... // The date you wish to test
NSCalendar *calendar = [NSCalendar currentCalendar];
if([calendar isDateInToday:date]) {
//do stuff
}
Questo potrebbe probabilmente essere rielaborato come categoria NSDate, ma ho usato:
// Seconds per day (24h * 60m * 60s)
#define kSecondsPerDay 86400.0f
+ (BOOL) dateIsToday:(NSDate*)dateToCheck
{
// Split today into components
NSCalendar* gregorian = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar];
NSDateComponents* comps = [gregorian components:(NSYearCalendarUnit|NSMonthCalendarUnit|NSDayCalendarUnit|NSHourCalendarUnit|NSMinuteCalendarUnit|NSSecondCalendarUnit)
fromDate:[NSDate date]];
// Set to this morning 00:00:00
[comps setHour:0];
[comps setMinute:0];
[comps setSecond:0];
NSDate* theMidnightHour = [gregorian dateFromComponents:comps];
[gregorian release];
// Get time difference (in seconds) between date and then
NSTimeInterval diff = [dateToCheck timeIntervalSinceDate:theMidnightHour];
return ( diff>=0.0f && diff<kSecondsPerDay );
}
(Tuttavia, confrontando le due stringhe di data come nella domanda iniziale si sente quasi 'pulito' ..)
per iOS7 e precedenti:
//this is now => need that for the current date
NSDate * now = [NSDate date];
NSCalendar *calendar = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar];
[calendar setTimeZone:[NSTimeZone systemTimeZone]];
NSDateComponents * components = [calendar components:( NSYearCalendarUnit| NSMonthCalendarUnit | NSDayCalendarUnit | NSHourCalendarUnit | NSMinuteCalendarUnit | NSSecondCalendarUnit) fromDate: now];
[components setMinute:0];
[components setHour:0];
[components setSecond:0];
//this is Today's Midnight
NSDate *todaysMidnight = [calendar dateFromComponents: components];
//now timeIntervals since Midnight => in seconds
NSTimeInterval todayTimeInterval = [now timeIntervalSinceDate: todaysMidnight];
//now timeIntervals since OtherDate => in seconds
NSTimeInterval otherDateTimeInterval = [now timeIntervalSinceDate: otherDate];
if(otherDateTimeInterval > todayTimeInterval) //otherDate is not in today
{
if((otherDateTimeInterval - todayTimeInterval) <= 86400) //86400 == a day total seconds
{
@"yesterday";
}
else
{
@"earlier";
}
}
else
{
@"today";
}
now = nil;
calendar = nil;
components = nil;
todaysMidnight = nil;
NSLog("Thank you :-)");
Controlla grande NSDate extension
della nostra Erica Sadun. Molto semplice da usare. Multa qui:
http://github.com/erica/NSDate-Extensions
E 'già presente in questo post: https://stackoverflow.com/a/4052798/362310
La soluzione corretta e sicura senza forza-unwrapping, lavorando su Swift 2.2 e prima di iOS 8:
func isToday() -> Bool {
let calendar = NSCalendar.currentCalendar()
if #available(iOS 8.0, *) {
return calendar.isDateInToday(self)
}
let todayComponents = calendar.components([.Era, .Year, .Month, .Day], fromDate:NSDate())
let dayComponents = calendar.components([.Era, .Year, .Month, .Day], fromDate:self)
guard let today = calendar.dateFromComponents(todayComponents),
day = calendar.dateFromComponents(dayComponents) else {
return false
}
return today.compare(day) == .OrderedSame
}
Ecco la mia risposta 2 cent edificio sulla risposta accettata, ma sostenere l'API più recente pure. Nota: Io uso il calendario gregoriano, come la maggior parte delle marche temporali sono GMT, ma cambiare la vostra, come si vede in forma
func isDateToday(date: NSDate) -> Bool {
let calendar = NSCalendar(calendarIdentifier: NSCalendarIdentifierGregorian)!
if calendar.respondsToSelector("isDateInToday:") {
return calendar.isDateInToday(date)
}
let dateComponents = NSCalendarUnit.CalendarUnitYear | NSCalendarUnit.CalendarUnitMonth | NSCalendarUnit.CalendarUnitDay
let today = calendar.dateFromComponents(calendar.components(dateComponents, fromDate: NSDate()))!
let dateToCompare = calendar.dateFromComponents(calendar.components(dateComponents, fromDate: date))!
return dateToCompare == today
}
La mia soluzione è calcolare quanta giorni trascorsi dal 1970 dalla divisione e confrontare la parte intera
#define kOneDay (60*60*24)
- (BOOL)isToday {
NSInteger offset = [[NSTimeZone defaultTimeZone] secondsFromGMT];
NSInteger days =[self timeIntervalSince1970] + offset;
NSInteger currentDays = [[NSDate date] timeIntervalSince1970] + offset;
return (days / kOneDay == currentDays / kOneDay);
}
NSDate *dateOne = yourDate;
NSDate *dateTwo = [NSDate date];
switch ([dateOne compare:dateTwo])
{
case NSOrderedAscending:
NSLog(@”NSOrderedAscending”);
break;
case NSOrderedSame:
NSLog(@”NSOrderedSame”);
break;
case NSOrderedDescending:
NSLog(@”NSOrderedDescending”);
break;
}