Swift 3 - Сравнение объектов Date

95

Я обновляю свое приложение до синтаксиса Swift 3.0 (я знаю, что оно все еще находится в стадии бета-тестирования, но хочу быть готовым, как только оно будет выпущено).

До предыдущей бета-версии Xcode (Beta 5) я мог сравнивать два Dateобъекта, используя операнды <, >и ==. Но в последней бета-версии (Beta 6) это больше не работает. Вот несколько скриншотов:

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

Как вы можете видеть на обоих скриншотах, это два Dateобъекта. Но я получаю следующую ошибку: введите описание изображения здесь

Что я делаю не так? Функции все еще объявлены в Dateклассе:

static func >(Date, Date)

Возвращает истину, если дата слева позже по времени, чем дата справа.

Это просто ошибка бета-версии или я что-то делаю не так?

beeef
источник
2
let d1 = Date() ; let d2 = Date() ; if d1 > d2 { }работает в моем Xcode 8 beta 6.
Мартин Р
1
Да. - Вам следует просмотреть полное сообщение об ошибке в навигаторе отчетов.
Martin R
Спасибо @MartinR! Не знал, что могу «погрузиться» в ошибку и получить дополнительную информацию. Мне очень помогли!
beeef
примите правильный ответ
Анкит Такур

Ответы:

161

Я пробовал этот фрагмент (в Xcode 8 Beta 6), и он работает нормально.

let date1 = Date()
let date2 = Date().addingTimeInterval(100)

if date1 == date2 { ... }
else if date1 > date2 { ... }
else if date1 < date2 { ... }
Анкит Такур
источник
если я хочу игнорировать время. например, 2019-05-14 12:08:14 +0000== 2019-05-14должен возвращать истину. тогда что ?
Awais Fayyaz
@AwaisFayyaz Если вы хотите сравнивать даты без учета DateComponent. Вам нужно получить нужные вам компоненты с даты. Затем получите дату из DateComponents и сравните их.
Giorgio Doganiero
52

Dateесть Comparable& Equatable(начиная с Swift 3)

Этот ответ дополняет ответ @Ankit Thakur.

Так как Swift 3, Dateструктура (на основе базового NSDateкласса) принимает сии Comparableи Equatableпротоколы.

  • Comparableтребует , чтобы Dateреализовать операторы: <, <=, >, >=.
  • Equatableтребует, чтобы Dateреализовать ==оператор.
  • Equatableпозволяет Dateиспользовать реализацию !=оператора по умолчанию (которая является обратной реализации Equatable ==оператора).

В следующем примере кода используются эти операторы сравнения и подтверждается, какие сравнения верны с printоператорами.

Функция сравнения

import Foundation

func describeComparison(date1: Date, date2: Date) -> String {

    var descriptionArray: [String] = []

    if date1 < date2 {
        descriptionArray.append("date1 < date2")
    }

    if date1 <= date2 {
        descriptionArray.append("date1 <= date2")
    }

    if date1 > date2 {
        descriptionArray.append("date1 > date2")
    }

    if date1 >= date2 {
        descriptionArray.append("date1 >= date2")
    }

    if date1 == date2 {
        descriptionArray.append("date1 == date2")
    }

    if date1 != date2 {
        descriptionArray.append("date1 != date2")
    }

    return descriptionArray.joined(separator: ",  ")
}

Образец использования

let now = Date()

describeComparison(date1: now, date2: now.addingTimeInterval(1))
// date1 < date2,  date1 <= date2,  date1 != date2

describeComparison(date1: now, date2: now.addingTimeInterval(-1))
// date1 > date2,  date1 >= date2,  date1 != date2

describeComparison(date1: now, date2: now)
// date1 <= date2,  date1 >= date2,  date1 == date2
Мобильный Дэн
источник
31

из Swift 3 и выше Date is Comparable, поэтому мы можем напрямую сравнивать даты, например

let date1 = Date()
let date2 = Date().addingTimeInterval(50)

let isGreater = date1 > date2
print(isGreater)

let isSmaller = date1 < date2
print(isSmaller)

let isEqual = date1 == date2
print(isEqual)

В качестве альтернативы мы можем создать расширение на дату

extension Date {

  func isEqualTo(_ date: Date) -> Bool {
    return self == date
  }
  
  func isGreaterThan(_ date: Date) -> Bool {
     return self > date
  }
  
  func isSmallerThan(_ date: Date) -> Bool {
     return self < date
  }
}

Использование: let isEqual = date1.isEqualTo(date2)

Сухит Патил
источник
если я хочу игнорировать время. например, 2019-05-14 12:08:14 +0000== 2019-05-14должен возвращать истину. тогда что ?
Awais Fayyaz
16

Посмотрите это http://iswift.org/cookbook/compare-2-dates

Получить даты:

// Get current date
let dateA = NSDate()

// Get a later date (after a couple of milliseconds)
let dateB = NSDate()

Использование инструкции SWITCH

// Compare them
switch dateA.compare(dateB) {
    case .OrderedAscending     :   print("Date A is earlier than date B")
    case .OrderedDescending    :   print("Date A is later than date B")
    case .OrderedSame          :   print("The two dates are the same")
}

с использованием оператора IF

 if dateA.compare(dateB) == .orderedAscending {
     datePickerTo.date = datePicker.date
 }

 //OR

 if case .orderedAcending = dateA.compare(dateB) {

 }
Луис Абреу Асеведо
источник
9

Для меня проблема заключалась в том, что у меня было собственное расширение класса Date, которое определяло все операторы сравнения. Теперь (начиная с swift 3), что Date сопоставима, эти расширения не нужны. Я прокомментировал их, и это сработало.

Джитендра Кулкарни
источник
8

SWIFT 3: Не знаю, ищете ли вы это. Но я сравниваю строку с текущей меткой времени, чтобы узнать, старше ли моя строка.

func checkTimeStamp(date: String!) -> Bool {
        let dateFormatter: DateFormatter = DateFormatter()
        dateFormatter.dateFormat = "yyyy-MM-dd HH:mm:ss"
        dateFormatter.locale = Locale(identifier:"en_US_POSIX")
        let datecomponents = dateFormatter.date(from: date)

        let now = Date()

        if (datecomponents! >= now) {
            return true
        } else {
            return false
        }
    }

Чтобы использовать это:

if (checkTimeStamp(date:"2016-11-21 12:00:00") == false) {
    // Do something
}
NB
источник
5

Чтобы сравнить дату только с годом - месяцем - днем ​​и без времени, у меня получалось так:

     let order = Calendar.current.compare(self.startDate, to: compareDate!, toGranularity: .day)  

                      switch order {
                        case .orderedAscending:
                            print("\(gpsDate) is after \(self.startDate)")
                        case .orderedDescending:
                            print("\(gpsDate) is before \(self.startDate)")
                        default:
                            print("\(gpsDate) is the same as \(self.startDate)")
                        }
Алессандро Маттиуцци
источник
5

Например, если вы хотите игнорировать секунды, вы можете использовать

func isDate (Date, equalTo: Date, toUnitGranularity: NSCalendar.Unit) -> Bool

Пример сравнения, если это тот же день:

Calendar.current.isDate(date1, equalTo: date2, toGranularity: .day)
Здравко Здравкин
источник
1

По состоянию на момент написания этой статьи, Swift изначально поддерживает сравнение дат со всеми операторами сравнения (т.е. <, <=, ==, >=, и >). Вы можете также сравнить дополнительные даты , но ограничены <, ==и >. Если вам нужно сравнить две необязательные даты с помощью <=или >=, т.е.

let date1: Date? = ...
let date2: Date? = ...
if date1 >= date2 { ... }

Вы можете перегрузить <=и >=оператор поддержки дополнительных опции:

func <= <T: Comparable>(lhs: T?, rhs: T?) -> Bool {
    return lhs == rhs || lhs < rhs
}

func >= <T: Comparable>(lhs: T?, rhs: T?) -> Bool {
    return lhs == rhs || lhs > rhs
}
NSExceptional
источник
1
extension Date {
 func isBetween(_ date1: Date, and date2: Date) -> Bool {
    return (min(date1, date2) ... max(date1, date2)).contains(self)
  }
}

let resultArray = dateArray.filter { $0.dateObj!.isBetween(startDate, and: endDate) }
Арджун Патель
источник
1

Другой способ сделать это:

    switch date1.compare(date2) {
        case .orderedAscending:
            break

        case .orderedDescending:
            break;

        case .orderedSame:
            break
    }
Шадрос
источник
1
   var strDateValidate = ""
            let dateFormatter = DateFormatter()
            dateFormatter.dateFormat = "yyyy-MM-dd"

            let firstDate = dateFormatter.date(from:lblStartTime.text!)
            let secondDate = dateFormatter.date(from:lblEndTime.text!)

            if firstDate?.compare(secondDate!) == .orderedSame || firstDate?.compare(secondDate!) == .orderedAscending {
                print("Both dates are same or first is less than scecond")
                strDateValidate = "yes"
            }
            else
            {
                //second date is bigger than first
                strDateValidate = "no"
            }

            if strDateValidate == "no"
            {
                alertView(message: "Start date and end date for a booking must be equal or Start date must be smaller than the end date", controller: self)
            }
Давендер Верма
источник
1

Swift 5:

1) Если вы используете тип даты:

let firstDate  = Date()
let secondDate = Date()

print(firstDate > secondDate)  
print(firstDate < secondDate)
print(firstDate == secondDate)

2) Если вы используете тип String:

let firstStringDate  = "2019-05-22T09:56:00.1111111"
let secondStringDate = "2019-05-22T09:56:00.2222222"

print(firstStringDate > secondStringDate)  // false
print(firstStringDate < secondStringDate)  // true
print(firstStringDate == secondStringDate) // false 

Не уверен или второй вариант работает на все 100%. Но сколько бы я не изменил значения firstStringDate и secondStringDate, результат был правильным.

Максвелл
источник
1

в Swift 3,4 вы должны использовать «Сравнить». например:

DateArray.sort { (($0)?.compare($1))! == .orderedDescending }
Алиреза
источник