iPhone: Как узнать текущие миллисекунды?


Ответы:


274
[[NSDate date] timeIntervalSince1970];

Возвращает количество секунд с начала эпохи как двойное. Я почти уверен, что вы можете получить доступ к миллисекундам из дробной части.


29
никто не убивает, чтобы написать это так: NSTimeInterval myInterval = NSDate.timeIntervalSince1970; // все эти скобки действительно старомодны, если вы спросите меня.
Пиццайола Горгонзола

11
«эти скобки» являются более новой версией синтаксиса в соответствии со stackoverflow.com/q/11474284/209828
Матфея

119
«эти скобки» являются объективными. Если вы хотите разрабатывать для iOS, вам, вероятно, стоит освоиться с ними.
Ampers4nd

5
Я использовал этот метод некоторое время и пришел к выводу, что он может вернуть более раннее значение, когда вы вызываете его через пару миллисекунд (например, вызываете его последовательно, и у вас может не получиться увеличивающаяся последовательность)
Ege Akpinar

5
[NSDate timeIntervalSinceReferenceDate]дешевле. (Хотя это относится к другой «эпохе» - если вы хотите, чтобы 1970 добавили константу NSTimeIntervalSince1970.)
Hot Licks

311

Если вы хотите использовать это для относительного времени (например, для игр или анимации), я бы лучше использовал CACurrentMediaTime ()

double CurrentTime = CACurrentMediaTime();

Какой рекомендуемый способ; NSDateизвлекает данные из сетевых синхронизирующих часов и иногда приводит к сбоям при повторной синхронизации с сетью.

Возвращает текущее абсолютное время в секундах.


Если вы хотите только десятичную часть (часто используется при синхронизации анимации),

let ct = CACurrentMediaTime().truncatingRemainder(dividingBy: 1)

7
Но, похоже, для [[NSDate date] timeIntervalSince1970 требуется вдвое больше времени. Я измерил 0,065 мс против 0,033 мс на 15000 звонков.
Кей

1
Обратите внимание, что вам нужно будет включить Quartz Framework и #import <Quartz / CABase.h>, чтобы сделать этот вызов.
BadPirate

8
Упс - вот импорт #import <QuartzCore / CAAnimation.h>
BadPirate

6
Для новичков в Xcode (как и я) «включить Quartz Framework» означает добавить его в набор библиотек в «Link Binary With Libraries».
Гейб Джонсон

3
Если кто-то ищет это в отношении SpriteKit, «текущее время» в методе обновления SKScene - это действительно CACurrentMediaTime ();
Энтони Мэттокс

92

Я проверил все остальные ответы на iPhone 4S и iPad 3 (сборка релиза). CACurrentMediaTimeимеет наименьшие накладные расходы с небольшим запасом. timeIntervalSince1970гораздо медленнее, чем другие, вероятно, из-за NSDateнакладных расходов на создание экземпляров, хотя это может не иметь значения для многих случаев использования.

Я бы порекомендовал, CACurrentMediaTimeесли вы хотите наименьших накладных расходов и не возражаете добавить зависимость Quartz Framework. Или gettimeofdayесли мобильность является приоритетом для вас.

Айфон 4С

CACurrentMediaTime: 1.33 µs/call
gettimeofday: 1.38 µs/call
[NSDate timeIntervalSinceReferenceDate]: 1.45 µs/call
CFAbsoluteTimeGetCurrent: 1.48 µs/call
[[NSDate date] timeIntervalSince1970]: 4.93 µs/call

iPad 3

CACurrentMediaTime: 1.25 µs/call
gettimeofday: 1.33 µs/call
CFAbsoluteTimeGetCurrent: 1.34 µs/call
[NSDate timeIntervalSinceReferenceDate]: 1.37 µs/call
[[NSDate date] timeIntervalSince1970]: 3.47 µs/call

1
+1, хотя это, очевидно, очень информативно и полезно, я бы не назвал это отличной инженерной работой, не увидев некоторый исходный код;)
Стивен Лу

2
Остерегайтесь этой проблемы, хотя: bendodson.com/weblog/2013/01/29/ca-current-media-time Эти часы, по-видимому, останавливаются, когда устройство переходит в спящий режим.
Моджуба

1
Есть NSTimeIntervalSince1970макрос, который является самым быстрым.
Озгур

@ozgur NSTimeIntervalSince1970- это константа, описывающая секунды между 1970-01-01 и «контрольной датой» (т.е. 2001-01-01), поэтому она всегда равна 978307200
YourMJK

53

В Swift мы можем сделать функцию и сделать следующее

func getCurrentMillis()->Int64{
    return  Int64(NSDate().timeIntervalSince1970 * 1000)
}

var currentTime = getCurrentMillis()

Хотя его работает отлично в Swift 3.0 , но мы можем изменить и использовать Dateкласс вместо NSDateв 3,0

Swift 3.0

func getCurrentMillis()->Int64 {
    return Int64(Date().timeIntervalSince1970 * 1000)
}

var currentTime = getCurrentMillis()

30

До сих пор я нашел gettimeofdayхорошее решение для iOS (iPad), когда вы хотите выполнить некоторую оценку интервала (скажем, частоту кадров, синхронизацию кадра рендеринга ...):

#include <sys/time.h>
struct timeval time;
gettimeofday(&time, NULL);
long millis = (time.tv_sec * 1000) + (time.tv_usec / 1000);

2
Мне это тоже нравится, так как он очень портативный. Обратите внимание, что в зависимости от операционной системы вам может потребоваться использовать «long long» вместо «long», а также приведите time.tv_sec к «long long» перед выполнением остальных вычислений.
AbePralle

static unsigned long getMStime (void) {struct timeval time; gettimeofday (& time, NULL); return (time.tv_sec * 1000) + (time.tv_usec / 1000); }
Дэвид Х

Интересно, что это хорошо работает на моем iPhone 5S и Mac, но возвращает неправильное значение на iPad 3.
Hyndrix

Чтобы избежать получения отрицательных чисел, мне пришлось приводить перед математикой: int64_t result = ((int64_t) tv.tv_sec * 1000) + ((int64_t) tv.tv_usec / 1000);
Джейсон Гилберт

это время возвращения в -ve
Шаукет Шейх

23

Чтобы получить миллисекунды на текущую дату.

Свифт 4+:

func currentTimeInMilliSeconds()-> Int
    {
        let currentDate = Date()
        let since1970 = currentDate.timeIntervalSince1970
        return Int(since1970 * 1000)
    }

18

Swift 2

let seconds = NSDate().timeIntervalSince1970
let milliseconds = seconds * 1000.0

Свифт 3

let currentTimeInMiliseconds = Date().timeIntervalSince1970.milliseconds

2
TimeIntervalУ aka Doubleнет свойства миллисекунд. Date().timeIntervalSince1970 * 1000
Лео Дабус

10

Может быть полезно знать о CodeTimestamps, которые предоставляют обертку вокруг основанных на мах функций синхронизации. Это дает вам данные синхронизации с наносекундным разрешением - в 1000000 раз точнее, чем миллисекунды. Да, в миллион раз точнее. (Префиксы - это milli, micro, nano, каждый из которых в 1000 раз точнее предыдущего.) Даже если вам не нужны CodeTimestamps, проверьте код (он с открытым исходным кодом), чтобы узнать, как они используют mach для получения данных синхронизации. Это было бы полезно, когда вам нужна большая точность и более быстрый вызов метода, чем подход NSDate.

http://eng.pulse.me/line-by-line-speed-analysis-for-ios-apps/


И вам, вероятно, понадобится, -fno-objc-arcесли вы используете ARC :)
Дэн Розенстарк



8
// Timestamp after converting to milliseconds.

NSString * timeInMS = [NSString stringWithFormat:@"%lld", [@(floor([date timeIntervalSince1970] * 1000)) longLongValue]];

6

Мне нужен NSNumberобъект, содержащий точный результат [[NSDate date] timeIntervalSince1970]. Поскольку эту функцию вызывали много раз, и мне не нужно было создавать NSDateобъект, производительность была невелика.

Итак, чтобы получить формат, который мне давала оригинальная функция, попробуйте это:

#include <sys/time.h>
struct timeval tv;
gettimeofday(&tv,NULL);
double perciseTimeStamp = tv.tv_sec + tv.tv_usec * 0.000001;

Что должно дать вам тот же результат, что и [[NSDate date] timeIntervalSince1970]


4

CFAbsoluteTimeGetCurrent ()

Абсолютное время измеряется в секундах относительно абсолютной контрольной даты 1 января 2001 года 00:00:00 по Гринвичу. Положительное значение представляет дату после ссылочной даты, отрицательное значение представляет дату перед ней. Например, абсолютное время -32940326 эквивалентно 16 декабря 1999 года в 17:54:34. Повторные вызовы этой функции не гарантируют монотонно увеличивающиеся результаты. Системное время может уменьшиться из-за синхронизации с внешними временными привязками или из-за явной смены часов пользователем.


4

Попробуй это :

NSDate * timestamp = [NSDate dateWithTimeIntervalSince1970:[[NSDate date] timeIntervalSince1970]];

NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
[dateFormatter setDateFormat:@"YYYY-MM-dd HH:mm:ss.SSS"];

NSString *newDateString = [dateFormatter stringFromDate:timestamp];
timestamp = (NSDate*)newDateString;

В этом примере dateWithTimeIntervalSince1970 используется в сочетании с форматером @ "ГГГГ-ММ-дд ЧЧ: мм: ss.SSS", который будет возвращать дату с годом, месяцем, днем ​​и временем с часами, минутами, секундами и миллисекундами , Смотрите пример: «2015-12-02 04: 43: 15.008». Я использовал строку NSString, чтобы быть уверенным, что формат будет записан ранее.


4

Это в основном тот же ответ, что и @TristanLorach, только что перекодированный для Swift 3:

   /// Method to get Unix-style time (Java variant), i.e., time since 1970 in milliseconds. This 
   /// copied from here: http://stackoverflow.com/a/24655601/253938 and here:
   /// http://stackoverflow.com/a/7885923/253938
   /// (This should give good performance according to this: 
   ///  http://stackoverflow.com/a/12020300/253938 )
   ///
   /// Note that it is possible that multiple calls to this method and computing the difference may 
   /// occasionally give problematic results, like an apparently negative interval or a major jump 
   /// forward in time. This is because system time occasionally gets updated due to synchronization 
   /// with a time source on the network (maybe "leap second"), or user setting the clock.
   public static func currentTimeMillis() -> Int64 {
      var darwinTime : timeval = timeval(tv_sec: 0, tv_usec: 0)
      gettimeofday(&darwinTime, nil)
      return (Int64(darwinTime.tv_sec) * 1000) + Int64(darwinTime.tv_usec / 1000)
   }

2
 func currentmicrotimeTimeMillis() -> Int64{
let nowDoublevaluseis = NSDate().timeIntervalSince1970
return Int64(nowDoublevaluseis*1000)

}


1

Это то, что я использовал для Swift

var date = NSDate()
let currentTime = Int64(date.timeIntervalSince1970 * 1000)

print("Time in milliseconds is \(currentTime)")

использовал этот сайт для проверки точности http://currentmillis.com/


1
let timeInMiliSecDate = Date()
let timeInMiliSec = Int (timeInMiliSecDate.timeIntervalSince1970 * 1000)
print(timeInMiliSec)

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

0

[NSDate timeIntervalSinceReferenceDate]другой вариант, если вы не хотите включать каркас Quartz. Возвращает двойное значение, представляющее секунды.


0
NSTimeInterval time = ([[NSDate date] timeIntervalSince1970]); //double
long digits = (long)time; //first 10 digits        
int decimalDigits = (int)(fmod(time, 1) * 1000); //3 missing digits
/*** long ***/
long timestamp = (digits * 1000) + decimalDigits;
/*** string ***/
NSString *timestampString = [NSString stringWithFormat:@"%ld%03d",digits ,decimalDigits];
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.