อะไรคือวิธีที่ดีงามในการตัดทอนออบเจกต์ python datetime?
ในกรณีนี้โดยเฉพาะอย่างยิ่งวันที่ ดังนั้นโดยทั่วไปการตั้งค่าชั่วโมงนาทีวินาทีและไมโครวินาทีเป็น 0
ฉันต้องการให้ผลลัพธ์เป็นวัตถุ datetime ด้วยไม่ใช่สตริง
อะไรคือวิธีที่ดีงามในการตัดทอนออบเจกต์ python datetime?
ในกรณีนี้โดยเฉพาะอย่างยิ่งวันที่ ดังนั้นโดยทั่วไปการตั้งค่าชั่วโมงนาทีวินาทีและไมโครวินาทีเป็น 0
ฉันต้องการให้ผลลัพธ์เป็นวัตถุ datetime ด้วยไม่ใช่สตริง
คำตอบ:
ฉันคิดว่านี่คือสิ่งที่คุณกำลังมองหา ...
>>> import datetime
>>> dt = datetime.datetime.now()
>>> dt = dt.replace(hour=0, minute=0, second=0, microsecond=0) # Returns a copy
>>> dt
datetime.datetime(2011, 3, 29, 0, 0)
แต่ถ้าคุณไม่สนใจเกี่ยวกับเวลาของสิ่งต่าง ๆ จริง ๆ แล้วคุณควรจะผ่านdate
วัตถุเท่านั้น ...
>>> d_truncated = datetime.date(dt.year, dt.month, dt.day)
>>> d_truncated
datetime.date(2011, 3, 29)
datetime
ไม่ใช่date
วัตถุ (ที่คุณสามารถใช้dt.date()
สาย (ไม่จำเป็นต้องใช้ตัวสร้างที่ชัดเจน)) .replace()
วิธีการอาจล้มเหลวหากเพิ่มการสนับสนุนdatetime
nanosecond คุณสามารถใช้datetime.combine()
แทน
datetime.datetime.now().replace(hour=0, minute=0, second=0, microsecond=0)
ล่ะ?
ใช้ a date
ไม่ใช่datetime
ถ้าคุณไม่สนใจเวลา
>>> now = datetime.now()
>>> now.date()
datetime.date(2011, 3, 29)
คุณสามารถอัปเดตช่วงเวลาเช่นนี้:
>>> now.replace(minute=0, hour=0, second=0, microsecond=0)
datetime.datetime(2011, 3, 29, 0, 0)
replace
ฉันรู้คำตอบที่ยอมรับได้เมื่อสี่ปีที่แล้ว แต่ดูเหมือนว่าเบากว่าการใช้replace
:
dt = datetime.date.today()
dt = datetime.datetime(dt.year, dt.month, dt.day)
หมายเหตุ
datetime
วัตถุโดยไม่ผ่านคุณสมบัติเวลาไปยังตัวสร้างคุณจะได้รับเที่ยงคืนdt = datetime.datetime.now()
คุณไม่สามารถตัดวัตถุ datetime เพราะมันจะไม่เปลี่ยนรูป
อย่างไรก็ตามนี่เป็นวิธีหนึ่งในการสร้างดาต้าไทม์ใหม่ด้วยฟิลด์ 0 ชั่วโมงนาทีวินาทีและไมโครวินาทีโดยไม่ต้องทิ้งวันที่ดั้งเดิมหรือ tzinfo:
newdatetime = now.replace(hour=0, minute=0, second=0, microsecond=0)
replace
ตัวเลือกแรกเนื่องจากอาจเป็นสิ่งที่พวกเขาต้องการ
tzinfo=now.tzinfo
มันไม่ถูกต้องที่จะใช้ tzinfo
เวลาเที่ยงคืนเช่นอาจจะแตกต่างกัน UTC ชดเชยที่2012-04-01 00:09:00
(09:00) ในประเทศออสเตรเลีย / เมลเบิร์นเป็นเขตเวลาAEST+10:00
แต่ก็เป็นAEDT+11:00
ที่2012-04-01 00:00:00
(เที่ยงคืน) - มีจุดสิ้นสุดของเวลาการเปลี่ยนแปลงในวันนั้น คุณสามารถใช้pytz
โมดูลเพื่อแก้ไขดูคำตอบของฉัน
รับเที่ยงคืนที่สอดคล้องกับวัตถุ datetime ที่กำหนดคุณสามารถใช้datetime.combine()
วิธีการ :
>>> from datetime import datetime, time
>>> dt = datetime.utcnow()
>>> dt.date()
datetime.date(2015, 2, 3)
>>> datetime.combine(dt, time.min)
datetime.datetime(2015, 2, 3, 0, 0)
ข้อได้เปรียบเมื่อเทียบกับวิธีคือวิธีการแก้ชั่นจะทำงานต่อไปแม้ว่าโมดูลแนะนำการสนับสนุนนาโนวินาที.replace()
datetime.combine()
datetime
tzinfo
สามารถเก็บรักษาไว้ได้หากจำเป็น แต่อ็อฟเซ็ต utc อาจแตกต่างกันในเวลาเที่ยงคืนเช่นเนื่องจากการเปลี่ยน DST และวิธีการแก้ปัญหาไร้เดียงสา (การตั้งค่าtzinfo
แอตทริบิวต์เวลา) อาจล้มเหลว ดูฉันจะรับเวลา UTC ของ“ เที่ยงคืน” สำหรับเขตเวลาที่กำหนดได้อย่างไร
คุณสามารถใช้นุ่นสำหรับมัน (แม้ว่ามันอาจจะเป็นค่าใช้จ่ายสำหรับงานนั้น) คุณสามารถใช้รอบ , พื้นและceilเช่นสำหรับตัวเลขปกติและความถี่หมีแพนด้าใด ๆ จากการชดเชยนามแฝง :
import pandas as pd
import datetime as dt
now = dt.datetime.now()
pd_now = pd.Timestamp(now)
freq = '1d'
pd_round = pd_now.round(freq)
dt_round = pd_round.to_pydatetime()
print(now)
print(dt_round)
"""
2018-06-15 09:33:44.102292
2018-06-15 00:00:00
"""
คุณสามารถใช้วันที่และเวลาได้เพื่อแยกวัน, เดือน, ปี ...
ตัวอย่าง:
from datetime import datetime
d = datetime.today()
# Retrieves the day and the year
print d.strftime("%d-%Y")
ผลผลิต (สำหรับวันนี้):
29-2011
หากคุณต้องการเรียกคืนวันนี้คุณสามารถใช้คุณลักษณะวันเช่น:
from datetime import datetime
d = datetime.today()
# Retrieves the day
print d.day
Ouput (สำหรับวันนี้):
29
d.day
ฯลฯ
มีห้องสมุดที่ยอดเยี่ยมที่ใช้จัดการกับวันที่: Delorean
import datetime
from delorean import Delorean
now = datetime.datetime.now()
d = Delorean(now, timezone='US/Pacific')
>>> now
datetime.datetime(2015, 3, 26, 19, 46, 40, 525703)
>>> d.truncate('second')
Delorean(datetime=2015-03-26 19:46:40-07:00, timezone='US/Pacific')
>>> d.truncate('minute')
Delorean(datetime=2015-03-26 19:46:00-07:00, timezone='US/Pacific')
>>> d.truncate('hour')
Delorean(datetime=2015-03-26 19:00:00-07:00, timezone='US/Pacific')
>>> d.truncate('day')
Delorean(datetime=2015-03-26 00:00:00-07:00, timezone='US/Pacific')
>>> d.truncate('month')
Delorean(datetime=2015-03-01 00:00:00-07:00, timezone='US/Pacific')
>>> d.truncate('year')
Delorean(datetime=2015-01-01 00:00:00-07:00, timezone='US/Pacific')
และถ้าคุณต้องการรับค่า datetime กลับมา:
>>> d.truncate('year').datetime
datetime.datetime(2015, 1, 1, 0, 0, tzinfo=<DstTzInfo 'US/Pacific' PDT-1 day, 17:00:00 DST>)
ดูเพิ่มเติมได้ที่https://pandas.pydata.org/pandas-docs/stable/reference/api/pandas.Series.dt.floor.html
ตอนนี้เป็นปี 2019 ฉันคิดว่าวิธีที่มีประสิทธิภาพที่สุดในการทำคือ:
df['truncate_date'] = df['timestamp'].dt.floor('d')
มีโมดูล datetime_truncate ที่จัดการสิ่งนี้ให้คุณ มันแค่เรียก datetime.replace
6 ปีต่อมา ... ฉันพบโพสต์นี้และฉันชอบ aproach จำนวนมาก:
import numpy as np
dates_array = np.array(['2013-01-01', '2013-01-15', '2013-01-30']).astype('datetime64[ns]')
truncated_dates = dates_array.astype('datetime64[D]')
ไชโย
>>> import datetime
>>> dt = datetime.datetime.now()
>>> datetime.datetime.date(dt)
datetime.date(2019, 4, 2)
คุณสามารถใช้
datetime.date.today()
มันเบาและส่งคืนสิ่งที่คุณต้องการ
การตัดทอนหมายความว่าอย่างไร
คุณสามารถควบคุมการจัดรูปแบบได้เต็มรูปแบบโดยใช้เมธอด strftime () และการใช้สตริงรูปแบบที่เหมาะสม
http://docs.python.org/library/datetime.html#strftime-strptime-behavior
หากคุณกำลังจัดการกับชุดประเภท DateTime มีวิธีที่มีประสิทธิภาพมากขึ้นในการตัดทอนพวกเขาโดยเฉพาะอย่างยิ่งเมื่อวัตถุชุดมีแถวจำนวนมาก
คุณสามารถใช้ฟังก์ชั่นพื้น
ตัวอย่างเช่นหากคุณต้องการตัดให้เหลือชั่วโมง:
สร้างช่วงของวันที่
times = pd.Series(pd.date_range(start='1/1/2018 04:00:00', end='1/1/2018 22:00:00', freq='s'))
เราสามารถตรวจสอบได้โดยเปรียบเทียบเวลาทำงานระหว่างฟังก์ชั่นการแทนที่และการทำงานของพื้น
%timeit times.apply(lambda x : x.replace(minute=0, second=0, microsecond=0))
>>> 341 ms ± 18.2 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)
%timeit times.dt.floor('h')
>>>>2.26 ms ± 451 µs per loop (mean ± std. dev. of 7 runs, 100 loops each)
นี่เป็นอีกวิธีหนึ่งที่เหมาะสมในบรรทัดเดียว แต่ไม่สง่างามเป็นพิเศษ:
dt = datetime.datetime.fromordinal(datetime.date.today().toordinal())