Getting UTC time from Posix time - leap seconds
    조회 수: 32 (최근 30일)
  
       이전 댓글 표시
    
Posix time is almost based on UTC, but doesn't include leap seconds that get thrown in at unpredictable (at least to me) intervals every few years.
I can convert posix time to calendar time in the date time format using datetime(xxx,'ConvertFrom','posixtime') but I need time in UTC. Does the
conversion add back in the missing leap seconds? There have been 26 leap seconds since 1970, and I suspect we're due for another soon. I
don't want to have to modify my code everytime there's a leap second.
댓글 수: 1
  Yair Altman
      
 2020년 11월 14일
				For the benefit of potential users who just need plain POSIX-to-UTC datetime conversion and don't care about the leap seconds, an easy and super-fast method is to use the vectorized input format of the datenum function. i.e. 
datenum([1970 1 1 0 0 posixValue])
For example:
>> datestr(datenum([1970 1 1 0 0 129450789]))
ans =
    '07-Feb-1974 06:33:09'
I am aware that this does not directly answer the poster's question (where leap secs are indeed needed), but in most cases for readers coming across this post, these leap seconds are not very important and then they could use datenum, on all Matlab releases (even R2019b or earlier).
답변 (2개)
  Swatantra Mahato
    
 2020년 10월 23일
        Hi,
I am assuming you want to convert posixtime to UTC and take into account the leap seconds.
This can be done using the "datetime" function by also passing the 'TimeZone' Property as 'UTCLeapSeconds'. As an example to show the difference, consider the script:
d1=datetime(129450789,'ConvertFrom','posixtime','TimeZone','UTC'); %07-Feb-1974 06:33:09
d2=datetime(129450789,'ConvertFrom','posixtime','TimeZone','UTCLeapSeconds'); 
d3=datetime(14383421,'ConvertFrom','posixtime','TimeZone','UTC'); %16-Jun-1970 11:23:41
d4=datetime(14383421,'ConvertFrom','posixtime','TimeZone','UTCLeapSeconds');
a=d1-d3;
b=d2-d4;
On executing "a-b" on the MATLAB command line we get:
>> a-b
ans =
duration
-00:00:03
which is in line with there being 3 leap seconds in the duration considered.
You can refer to the documentation for the "datetime" function for more information:
Hope this helps
댓글 수: 4
  Steven Lord
    
      
 2020년 10월 27일
				According to the entry at the end of the leapseconds documentation page it was Introduced in R2020a.
  Peter Perkins
    
 2020년 11월 18일
        Bruce, you are getting tripped up by the difference between clockface time, and the elapsed time since 1970. Here's the deal:
As you already know, the "real actual" UTC time line includes leap seconds that are inserted at somewhat unpredictable intervals. So in the "real actual" UTC time line, if you count the number of seconds since 1970, you get something that's 27s longer than what you might have expected. In MATLAB, you can demonstrate that using datetime, and MATLAB calls that timeline 'UTCLeapSeconds':
>> t = datetime(2020,11,18,'TimeZone','UTCLeapSeconds')
t = 
  datetime
   2020-11-18T00:00:00.000Z
>> t - datetime(1970,1,1,'TimeZone','UTCLeapSeconds')
ans = 
  duration
   446016:00:27
But here's the thing: almost noone wants to hear that "truth". So the POSIX time line, and datetime's default "unzoned" behavior, and datetime's behavior for 'UTC' tell a white lie
>> t = datetime(2020,11,18,'TimeZone','UTC','Format','dd-MMM-uuuu HH:mm:ss.SSS')
t = 
  datetime
   18-Nov-2020 00:00:00.000
>> t - datetime(1970,1,1,'TimeZone','UTC')
ans = 
  duration
   446016:00:00
because most people don't want their calculations involving elapsed times to "randomly" be off by several seconds. I can't tell which group of people you are in, but if all you are ultimately doing is converting to a text clockface timestamp, it won't matter unless one of your timestamps falls on a leapsecond. Generally speaking, people who need to care about leaps seconds are often doing things with satellites or similar, and they can opt in.
More details:
When you say, "converting posix time to datetime format using the datetime command with arguement pair 'TimeZone', 'UTCleapseconds' does not add missing leap seconds to the result", that's not actually true. You have to consider that POSIX times are not on the "leap seconds" time line, so converting to the latter has to account for the leap seconds. For a POSIX time around now, that means adding in 27s, but what's tripping you up is that it does not change the clockface time. In other words, the POSIX time for midnight 18-Nov-2020 may claim to be 1605657600s since 1970, but midnight 18-Nov-2020 in the "real actual" UTC timeline that 'UTCLeapSeconds' represents is 1605657627s since 1970. These two things
>> t1 = datetime(2020,11,18,'TimeZone','UTCLeapSeconds')
t1 = 
  datetime
   2020-11-18T00:00:00.000Z
>> t1 = datetime(2020,11,18,'TimeZone','UTC','Format','dd-MMM-uuuu HH:mm:ss.SSS')
t1 = 
  datetime
   18-Nov-2020 00:00:00.000
look the same, and refer to the same instant in time, but calculating elapsed times with them is not the same.
In short: converting POSIX to/from (MATLAB's) 'UTC' is straight-forward because they both tell the same white lie. Converting to/from 'UTCLeapSeconds' needs, internally, to add in leap seconds in one direction, and remove them in the other. And of course, what is the POSIX time for 31-Dec-2016 23:59:60Z? Uhhhh ... it doesn't exist. It's an ill-defined question. So datetime maps that back to 31-Dec-2016 23:59:59Z.
댓글 수: 1
  James Tursa
      
      
 2023년 8월 7일
				
      편집: James Tursa
      
      
 2023년 8월 14일
  
			"...  And of course, what is the POSIX time for 31-Dec-2016 23:59:60Z? Uhhhh ... it doesn't exist. It's an ill-defined question ..."
To my understanding, posix times during +1 leap seconds certainly do exist, it is just that the resulting posix times are ambiguous (for -1 leap seconds there would be undefined UTC counterparts). E.g., for this +1 leap second:
format longg
dt = datetime(2016,12,31,23,59,59,'TimeZone','UTCLeapSeconds') + seconds(0:0.5:3)';
posixtime_dt = posixtime(dt);
table(dt,posixtime_dt)
Note that there are two datetimes associated with the posix time 1483228800 when using the 'UTCLeapSeconds' time zone, and this is totally correct.  In fact, all the posix times in the interval [1483228800,1483228801) certainly exist but they are ambiguous because of the 1 second jump back in time.  It is when you try to convert this to MATLAB's 'UTC' time zone that you run into problems because that time zone doesn't have this leap second. But MATLAB has to do something, so this is what it does:
dt_utc = datetime(dt,'TimeZone','UTC');
posixtime_dt_utc = posixtime(dt_utc);
table(dt,posixtime_dt,dt_utc,posixtime_dt_utc)
The posixtime_dt column is correct and the jump back in time happens at the proper instant. Converting dt to 'UTC' results in the jump occurring back to the 59 second mark. OK, that part is fine if that is how MATLAB wants to do it (I would have done it differently to make it consistent with posixtime), but then calling posixtime on that 'UTC' datetime results in an incorrect posix time of the original dt ... the jump back in time does not occur at the proper instant.  Bottom line is you need to be careful how you use the MATLAB functions when dealing with leap seconds.  Converting 'UTCLeapSeconds' to 'UTC' and then calling posixtime( ) on that will not get you the correct result during the leap second. 
Also, on older versions of MATLAB the posixtime( ) and convertTo(__,'posix') functions cannot handle non-scalar variables with  'UTCLeapSeconds' time zone inputs. So here you are forced to write different code, like do the calculations individually in a loop or convert to 'UTC' first and then do a correction. E.g.,
unix_time = posixtime_dt_utc + (second(dt)>=60); % add 1 to seconds in 60+ range
table(dt,posixtime_dt,unix_time)
Finally, I would add that 'UTCLeapSeconds' also affects how the juliandate( ) function behaves near leap seconds.  This function will smear the leap second across the +/-12 hour vicinity of the leap second, with the Julian Date day boundary occurring at the 60.5 second mark. Something to be aware of if you are working with Julian Dates. E.g.,
juliandate_dt = juliandate(dt);
juliandate_dt_utc = juliandate(dt_utc);
table(dt,juliandate_dt,dt_utc,juliandate_dt_utc)
And the smearing effect:
dt = datetime(2016,12,31,23,59,59,'TimeZone','UTCLeapSeconds') + seconds(-43200:43200)';
jd_dt = juliandate(dt);
1-mean(diff(jd_dt)*86400)
1/86400
The average value of a second during the +/-12 hours surrounding the leap second time is slightly less than 1 second, and the amount of this difference is pretty darn close to 1/86400.  That is, the 1 leap second has been smeared across the +/- 12 hour time interval. So Julian Dates calculated from the 'UTCLeapSeconds' dt variable will be slightly different from the Julian Dates calculated from their 'UTC' counterparts during this 24 hour interval. If you don't want this smearing, then you will need to convert to 'UTC' first and then write some code for how you want to handle the leap second itself.
*** Update 8/14/2023 ***
I have been informed by Mathworks that they are looking into this and may change the behavior of datetime when converting 'UTCLeapSeconds' to 'UTC' during the leap second in future MATLAB releases.
참고 항목
카테고리
				Help Center 및 File Exchange에서 Dates and Time에 대해 자세히 알아보기
			
	Community Treasure Hunt
Find the treasures in MATLAB Central and discover how the community can help you!
Start Hunting!





