freeze_time 不适用于默认参数

Kan*_*thy 5 python-unittest freezegun

我有一个参数默认值为 的函数datetime.now()。方法如下,

def as_standard_format(p_date=datetime.now(), fmt=sdk_constants.DEFAULT_DATE_TIME_FORMAT):
    return p_date.strftime(fmt)
Run Code Online (Sandbox Code Playgroud)

我有一个类似下面的测试方法,

@freeze_time(datetime(year=2018, month=9, day=25, hour=15, minute=46, second=36))
def testNowWithDefaultFormat(self):

    # The below method gives the frozen time
    print(datetime.now())

    # But the below test fails
    assert utils.format_date(
        datetime.now()) == "20180925T154636Z", "Default call not giving current time in " \
                                               "standard format \"%Y%m%dT%H%M%SZ\""
Run Code Online (Sandbox Code Playgroud)

为什么不能freeze_time使用默认参数值?

gma*_*uch 4

您是Python 中与可变默认值参数相关的常见问题的受害者,而不仅仅特定于freeze_time.

为了便于解释,请考虑以下代码:

import time from datetime import datetime

def what_time_is_it(p_date=datetime.now()):
    return p_date.strftime('"%Y-%m-%d %X"')

if __name__ == '__main__':
    print(what_time_is_it())
    time.sleep(10)
    print(what_time_is_it())
Run Code Online (Sandbox Code Playgroud)

您可能预期会发生什么

"2019-02-28 13:47:24"
"2019-02-28 13:47:34" # This line showing a datetime 10 seconds after the first one
Run Code Online (Sandbox Code Playgroud)

datetime.now()每次调用都会调用一个新的what_time_is_it()

发生了什么

"2019-02-28 13:47:24"
"2019-02-28 13:47:24" # EXACTLY the same time in both lines!
Run Code Online (Sandbox Code Playgroud)

datetime.now()定义函数时会创建一次新值,并且在每次连续调用中使用相同的值。

你应该做什么?

每次调用函数时创建一个新对象,通过使用默认参数来表示未提供任何参数(None通常是一个不错的选择):

def as_standard_format(p_date=None, fmt=sdk_constants.DEFAULT_DATE_TIME_FORMAT):
    if p_date is None:
        return datetime.now().strftime(fmt)
    else:
        return p_date.strftime(fmt)
Run Code Online (Sandbox Code Playgroud)