Python函数默认参数在定义时求值,需用None占位+运行时判断、可调用对象延迟执行或**kwargs兜底实现动态默认;禁用修改__defaults__等不安全方式。
函数默认参数在定义时就完成求值,这是 Python 的固定行为。若想实现“运行时动态计算”,必须绕过默认参数机制,改用其他方式延迟求值。
最常用、最清晰的做法:把默认值设为 None,在函数体内检查并按需计算。
import time
def log_message(msg=None, timestamp=None):
if msg is None:
msg = "default message"
if timestamp is None:
timestamp = time.time() # 每次调用都重新获取当前时间
print(f"[{timestamp:.0f}] {msg}")
l

og_message() # [1715234567] default message(每次不同)
log_message("hi") # [1715234568] hi(时间仍动态)
把“计算逻辑”本身作为默认参数传入,调用时再执行它。
import random
def get_random_id():
return f"id_{random.randint(1000, 9999)}"
def create_user(name, uid_gen=get_random_id): # 注意:没加括号!
uid = uid_gen() # 这里才真正调用,每次不同
return {"name": name, "uid": uid}
create_user("Alice") # {'name': 'Alice', 'uid': 'id_3842'}
create_user("Bob") # {'name': 'Bob', 'uid': 'id_7105'}(不同)
当参数数量或类型较灵活时,可结合解包与运行时填充。
from datetime import datetime
def report(title, **options):
now = datetime.now()
# 动态填充未提供的字段
data = {
"title": title,
"generated_at": options.get("generated_at", now),
"version": options.get("version", f"v{now.month}.{now.year}"),
"items": options.get("items", []),
}
return data
report("daily") # generated_at 和 version 都是本次调用时刻计算的
虽然技术上可通过 func.__defaults__ 修改,但强烈不建议:
本质上不是“动态默认”,而是“偷偷篡改”,应彻底避免。