ฉันจะแปลงวัตถุวันที่และเวลาเป็นมิลลิวินาทีตั้งแต่ยุค (เวลา unix) ใน Python ได้อย่างไร


356

ฉันมีdatetimeวัตถุPython ที่ฉันต้องการแปลงเป็นเวลา unix หรือวินาที / มิลลิวินาทีนับตั้งแต่ยุค 1970

ฉันจะทำสิ่งนี้ได้อย่างไร


3
ที่เกี่ยวข้อง: การแปลง datetime.date เป็นเวลา UTC ใน Python
jfs

หากคุณลงจอดที่นี่เพียงแค่ต้องการวินาทีปัจจุบันด้วยความแม่นยำมิลลิวินาทีลอง$ python -c 'import time; print(time.time())'ซึ่งให้:1584487455.698623
MarkHu

คำตอบ:


474

ฉันคิดว่าวิธีที่ง่ายที่สุดคือ

import datetime

epoch = datetime.datetime.utcfromtimestamp(0)

def unix_time_millis(dt):
    return (dt - epoch).total_seconds() * 1000.0

32
จนถึงตอนนี้เป็นทางออกที่ดีที่สุด (ถ้ารุ่นหลาม> 2.7) เพราะการใช้งาน%sขึ้นอยู่กับระบบปฏิบัติการ! ดังนั้นทุกคนต้องการรหัสทำงานเชื่อถือได้โดยไม่คำนึงถึงในที่ OS %sที่ใช้ไม่ควร สำหรับ 2.3 <py ver <2.7 เราสามารถสร้างสิ่งtotal_seconds()นี้ได้ง่ายๆ:delta.days*86400+delta.seconds+delta.microseconds/1e6
วัง

14
หมายเหตุ: dtต้องอยู่ใน UTC (ไม่ใช่ในพื้นที่) ดูคำตอบที่คล้ายกันกับ Python 2.6 / 3 รองรับ
jfs

7
ควรกล่าวถึงว่าหากสิ่งที่คุณต้องการคือการประทับเวลายูนิกซ์ที่แท้จริงตามที่กำหนดไว้ที่นี่en.wikipedia.org/wiki/Unix_time (และดังนั้นจะใช้ฟังก์ชัน unix_time จากคำตอบนี้เท่านั้น) คุณควรจะตัด delta.total_seconds () ด้วย int to หลีกเลี่ยงการจบด้วยการลอย
Corford

1
ยุคที่คำนวณโดยใช้utcfromtimestamp(0)อาจทำให้ 'tz' ออฟเซ็ตเฉพาะในกรณีที่ 'tz' ไม่ใช่ 0 นี่เป็นเพราะdt ในdt- epochมี 'tz' คำนวณในที่ที่เป็นยุคเวลา UTC วิธีที่ดีที่สุดในการคำนวณยุคคือการepoch = datetime.datetime(1970, 1, 1)พิจารณาของเขตเวลา
ahj

35
เหตุใดจึงไม่datetime.utcfromtimestamp(0)คืนค่าเวลาโดยไม่มี tzinfo มันมีสิทธิในชื่อวิธีการUTC fromtimestamp datetime.utcfromtimestamp(0).replace(tzinfo=pytz.UTC)เพื่อที่จะทำให้มันไม่ใช่ไร้เดียงสาที่ฉันต้องทำสิ่งที่ชอบ นี่เป็นสิ่งจำเป็นหากdtเขตเวลาทราบหรือไม่เช่นนั้นคุณจะได้รับTypeError: can't subtract offset-naive and offset-aware datetimes
FGreg

172

ใน Python 3.3 เพิ่มวิธีการใหม่timestamp:

import datetime
seconds_since_epoch = datetime.datetime.now().timestamp()

คำถามของคุณระบุว่าคุณต้องการมิลลิวินาทีซึ่งคุณจะได้รับเช่นนี้:

milliseconds_since_epoch = datetime.datetime.now().timestamp() * 1000

ถ้าคุณใช้timestampกับวัตถุ datetime ไร้เดียงสามันจะสันนิษฐานว่ามันอยู่ในเขตเวลาท้องถิ่น ใช้ออบเจ็กต์ datetime ที่รับรู้เขตเวลาหากนี่ไม่ใช่สิ่งที่คุณตั้งใจจะเกิดขึ้น


28
หมายเหตุ: .timestamp()เมธอดสันนิษฐานว่า datetime ที่ไร้เดียงสาอยู่ในเขตเวลาท้องถิ่น (และเวลาท้องถิ่นอาจไม่ชัดเจน) ถ้ามันอยู่ใน UTC แล้วใช้แทนdt.replace(tzinfo=timezone.utc).timestamp()
jfs

11
datetime.timestamp () ส่งคืนค่าวินาทีของวินาทีเป็นทศนิยม เพื่อรับ miliseconds: int (datetime.timestamp () * 1,000)
Solar.gy

9
คำตอบนี้ไม่ใช่ตัวอย่างที่คัดลอกและวางในใจของเอกสารที่ไม่เป็นมิตรกับมนุษย์ที่docs.python.org/3/library/… -นี่คือ: datetime.datetime.timestamp(datetime.datetime.now())
MarkHu

3
ใน 3.6 (อย่างน้อย) datetime.timestamp () จะถือว่าเขตเวลาไร้เดียงสา (tzinfo = None) datetime อยู่ใน UTC ดังนั้นดีกว่าเสมอที่จะมีการตั้งค่าเขตเวลาของคุณ: datetime.datetime.now(pytz.timezone('Europe/Paris')).timestamp() == datetime.datetime.now(pytz.utc).timestamp() == datetime.datetime.utcnow().timestamp()แต่ไม่เท่ากับ (เสมอ) เท่ากับdatetime.datetime.now().timestamp()(อันสุดท้ายนี้เท่ากับส่วนที่เหลือถ้า local tz คือ UTC ... )
Bluu

1
เพื่อความสนใจของคุณการย้อนกลับคือ] fromtimestamp( docs.python.org/3/library/… )
Flimm

98
>>> import datetime
>>> # replace datetime.datetime.now() with your datetime object
>>> int(datetime.datetime.now().strftime("%s")) * 1000 
1312908481000

หรือความช่วยเหลือของโมดูลเวลา (และไม่มีการจัดรูปแบบวันที่):

>>> import datetime, time
>>> # replace datetime.datetime.now() with your datetime object
>>> time.mktime(datetime.datetime.now().timetuple()) * 1000
1312908681000.0

ตอบด้วยความช่วยเหลือจาก: http://pleac.sourceforge.net/pleac_python/datesandtimes.html

เอกสารอ้างอิง:


6
Btw, strftime ("% s") จะส่งคืนสตริงว่างสำหรับฉัน วิธีที่สองทำงานได้ดี
Pavel Vlasov

23
มีเพียงความแม่นยำที่สอง
shuckc

6
Python ไม่รองรับ '% s' เช่นอาจมาจาก Windows .timetuple()ส่งคืนtm_isdst=-1มันบังคับmktime()ให้คาดเดา อาจเดาผิดระหว่าง DST (โอกาส 50% ของข้อผิดพลาด +/- ชั่วโมง) ทั้ง '% s' และmktime()อาจใช้อ็อฟเซ็ต utc ผิดสำหรับวันที่ผ่านมา คุณต้องการเขตเวลาในอดีต db เช่นจัดเตรียมโดยpytzโมดูลเพื่อแปลงเวลาท้องถิ่นได้อย่างน่าเชื่อถือเป็นเวลาประทับ POSIX (เว้นแต่ว่า OS ได้จัดเตรียมฐานข้อมูลไว้แล้ว)
jfs

1
time.mktime(ts.timetuple())โดยที่ ts เป็นวัตถุ datetime ของไพ
ธ อน

1
@suhail: mktime/timetupleอ่านความคิดเห็นของฉันข้างต้นเกี่ยวกับ นอกจากนี้ยังตัดtimetuple()เศษส่วนของหนึ่งวินาทีและจุดของคำถามคือการได้รับการประทับเวลาด้วยความแม่นยำมิลลิวินาที
jfs


13

นี่คือวิธีที่ฉันทำ:

from datetime import datetime
from time import mktime

dt = datetime.now()
sec_since_epoch = mktime(dt.timetuple()) + dt.microsecond/1000000.0

millis_since_epoch = sec_since_epoch * 1000


2
@JFSebastian ขอบคุณสำหรับหัวขึ้น! แน่นอนไม่มีการพิจารณา dst หากเซิร์ฟเวอร์ของคุณอยู่ในเวลาท้องถิ่นแทนที่จะเป็น UTC มันจะสร้างความแตกต่าง ฉันยังไม่พบเหตุผลที่น่าสนใจสำหรับการตั้งค่าเซิร์ฟเวอร์ในสิ่งอื่นที่ไม่ใช่ UTC Moto ของฉันคือ "เขียน UTC อ่านเวลาท้องถิ่น" เพื่อให้คุณรู้เสมอที่คุณจะเข้าพัก ...
estani

2
คุณสามารถใช้ calendar.timegm แทน mktime เพื่อหลีกเลี่ยงปัญหาของ mktime ที่พยายามคาดเดาเขตเวลา
Decko

13

@ChristopherBull เพียงแค่หารจำนวนมิลลิวินาทีด้วย 1,000 ถึงวินาที
Jonas

5
คุณเข้าใจผิดและเข้าใจผิด วินาทีพร้อมใช้งานได้ในฟังก์ชั่นด้านบน คุณสามารถแปลงเป็นมิลลิวินาที แต่มันจะเป็นความแม่นยำของวินาที
Christopher Bull

1
คำตอบนี้ใช้timeโมดูล แต่ OP ถามเกี่ยวกับdatetimeโมดูล FWIW ยุคปัจจุบันที่ง่ายที่สุดคือint(time.time())
MarkHu

7
from datetime import datetime
from calendar import timegm

# Note: if you pass in a naive dttm object it's assumed to already be in UTC
def unix_time(dttm=None):
    if dttm is None:
       dttm = datetime.utcnow()

    return timegm(dttm.utctimetuple())

print "Unix time now: %d" % unix_time()
print "Unix timestamp from an existing dttm: %d" % unix_time(datetime(2014, 12, 30, 12, 0))

timegm()ใช้ได้กับเวลา utc เท่านั้น มันไม่ได้ใช้tm_isdstดังนั้นคุณสามารถใช้แทนutcnow.timetuple() utcnow.utctimetuple()หมายเหตุ: การใช้naive_local_datetime.utctimetuple()จะผิดที่นี่ มันไม่ได้แปลเวลาท้องถิ่นเป็น utc นอกจากนี้timetuple()เรียกแถบเศษส่วนของวินาทีจากผลลัพธ์ (ไม่ว่ามันจะขึ้นอยู่กับแอพพลิเคชั่น) คำถามถามเกี่ยวกับ * milli * วินาทีไม่ใช่วินาที
jfs

ฉันชอบที่จะใช้ utcnow () และ utctimetuple () เพื่อทำให้รหัสชัดเจนอย่างแน่นอนว่าคุณกำลังจัดการกับ UTC (และวิธีนี้ทุกคนที่อ่านไม่จำเป็นต้องจำไว้ว่า timegm เป็น UTC เท่านั้น) utctimetuple () ไม่ได้บอกเป็นนัยถึงการแปลบนวัตถุไร้เดียงสา dttm (ดังนั้นการอ้างถึง dttm ด้วย utcnow ()) นอกจากนี้คำถามที่ถามถึงวินาทีหรือมิลลิวินาที
corford

หมายเหตุ: ควรพูดในความคิดเห็นล่าสุดฉันอ่านคำถามโดยบอกว่าเขาต้องการวินาทีหรือมิลลิวินาที (อาจเป็นความผิดพลาดของฉัน) สำหรับมิลลิวินาทีเพียงคูณด้วย 1,000 (ตามคำตอบที่ให้คะแนนสูงสุด)
corford

utctimetuple()แถบเศษส่วนของวินาที การคูณด้วย1000จะไม่ทำให้พวกมันกลับมา
jfs

1
เนื่องจากวิธีที่ OP ถามคำถามนี้จึงไม่ชัดเจนว่าเขา / เธอต้องการอะไร (เช่นยูนิกซ์จริงหรือเวลาประทับที่มีความแม่นยำมิลลิวินาที) ไม่ว่าคำถามทั้งสองจะถูกถามและตอบแล้ว ต้องบอกว่าฉันคิดว่าคำตอบที่นี่เป็นวิธีที่เร็วและสะอาดที่สุดสำหรับผู้คนที่จะคร่ำครวญ
corford

3
>>> import datetime
>>> import time
>>> import calendar

>>> #your datetime object
>>> now = datetime.datetime.now()
>>> now
datetime.datetime(2013, 3, 19, 13, 0, 9, 351812)

>>> #use datetime module's timetuple method to get a `time.struct_time` object.[1]
>>> tt = datetime.datetime.timetuple(now)
>>> tt
time.struct_time(tm_year=2013, tm_mon=3, tm_mday=19, tm_hour=13, tm_min=0, tm_sec=9,     tm_wday=1, tm_yday=78, tm_isdst=-1)

>>> #If your datetime object is in utc you do this way. [2](see the first table on docs)
>>> sec_epoch_utc = calendar.timegm(tt) * 1000
>>> sec_epoch_utc
1363698009

>>> #If your datetime object is in local timeformat you do this way
>>> sec_epoch_loc = time.mktime(tt) * 1000
>>> sec_epoch_loc
1363678209.0

[1] http://docs.python.org/2/library/datetime.html#datetime.date.timetuple

[2] http://docs.python.org/2/library/time.html


3

นี่เป็นอีกรูปแบบหนึ่งของการแก้ปัญหาด้วยการทำให้วัตถุเวลาของคุณเป็นปกติ

def to_unix_time(timestamp):
    epoch = datetime.datetime.utcfromtimestamp(0) # start of epoch time
    my_time = datetime.datetime.strptime(timestamp, "%Y/%m/%d %H:%M:%S.%f") # plugin your time object
    delta = my_time - epoch
    return delta.total_seconds() * 1000.0


1
import time
seconds_since_epoch = time.mktime(your_datetime.timetuple()) * 1000

9
นี่มันผิด! ตารางเวลาไม่รวมมิลลิวินาทีดังนั้น mktime จะไม่ส่งคืนค่ายุคที่มีความละเอียดเป็นมิลลิวินาที มันไร้ประโยชน์ในกรณีนี้
วัง

@ วัง - คุณเป็นคนที่ถูกต้องนี่จะไม่ส่งคืน
มิลลิ


1
หากคุณลบแม้ว่าคุณจะได้รับ* 1000 seconds_since_epochการถอนคำตอบนี้เนื่องจากฉันไม่สนใจมิลลิวินาทีในขณะนี้
Michael Scheper

0

นี่คือฟังก์ชั่นที่ฉันทำตามคำตอบข้างต้น

def getDateToEpoch(myDateTime):
    res = (datetime.datetime(myDateTime.year,myDateTime.month,myDateTime.day,myDateTime.hour,myDateTime.minute,myDateTime.second) - datetime.datetime(1970,1,1)).total_seconds()
    return res

คุณสามารถตัดค่าที่ส่งคืนเช่นนี้: str (int (res)) หากต้องการส่งคืนโดยไม่มีค่าทศนิยมที่จะใช้เป็นสตริงหรือเพียงแค่ int (ไม่มี str)


-14

โซลูชันอื่นสำหรับการแปลงข้อมูลวันที่เป็น unixtimestampmillis

private static readonly DateTime UnixEpoch = new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc);

    public static long GetCurrentUnixTimestampMillis()
    {
        DateTime localDateTime, univDateTime;
        localDateTime = DateTime.Now;          
        univDateTime = localDateTime.ToUniversalTime();
        return (long)(univDateTime - UnixEpoch).TotalMilliseconds;
    } 

2
คำถามเกี่ยวกับภาษา Python
Stéphane
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.