我收到各种格式化的日期或部分日期,并且在将它们转换为datetime.date
格式后,我想调整month字段或day字段而不调用构造函数以获取全新的datetime.date
对象
数据可以是以下任何一种:
'2012.01.01'
'194311'
'1865/08/30'
'1701'
... etc
对于未指定的日期,我希望将它们推送到1月1日,提供月份的第一个月,到提供的月份结束,或者到12月31日。必须提供一年且不会更改
dateutil.parser.parse
可以很好地处理所有这些问题,除了它填补了当前月份和/或日期缺少的月份和/或日期的惯例:
In [11]: dateutil.parser.parse('2012/04')
Out[11]: datetime.datetime(2012, 4, 10, 0, 0)
In [12]: dateutil.parser.parse('1865')
Out[12]: datetime.datetime(1865, 2, 10, 0, 0)
In [13]: dateutil.parser.parse('1920.03.27')
Out[13]: datetime.datetime(1920, 3, 27, 0, 0)
进入datetime.datetime
格式后,我可以调用datetime
函数date
来有效地投射为datetime.date
,因此该部分很简单。
我想避免的是编写一个必须经历调用构造函数的辅助函数,如下所示:
def adjust_to_jan1(date_str):
temp = dateutil.parser.parse(date_str).date()
return datetime.date(temp.year, 1, 1)
或
def adjust_to_month_end(date_str):
import calendar
temp = dateutil.parser.parse(date_str).date()
last_day = calendar.monthrange(temp.year, temp.month)
return datetime.date(temp.year, temp.month, last_day)
或
def adjust_to_last_weekdat(date_str):
import calendar
temp = dateutil.parser.parse(date_str).date()
last_weekday = max([elem for x in calendar.monthcalendar(temp.year,
temp.month)
for elem in x[0:5]])
return datetime.date(temp.year, temp.month, last_weekday)
由于我经常以交互方式处理这些数据,所以始终定义这些类型的函数会很麻烦。它变得更加复杂:上周这一天不是任何一组国家的假期...上个工作日除了如果年份在某一年之前可能是上周六......
我想做的只是声明我想要的更改,例如:
some_date = datetime.date(2012, 1, 20)
some_date.day = 1
但这是不允许的。
我可以继承datetime
,但后来我不得不担心我的子类没有造成任何损害,以及是否可以使用来自交互式工作的一些脚本对其他人进行移植。
任何其他解决方案都将受到赞赏。
如果您认为没有比继承或编写大量这些帮助函数更好的方法,那很好,但我更愿意为任何提案留下评论和答案 与此不同 - 而不是得到许多回答说“这看起来很好”或者其他什么。
答案 0 :(得分:6)
datetime.date
个对象是不可变的,但它们确实有.replace()
method来做你想做的事情:
somedate = somedate.replace(day=1)
该方法返回一个 new 对象,其中包含所需的值。
为了完整起见,还有datetime.datetime.replace()
和datetime.time.replace()
方法。