args将任意数量的位置参数收集为元组,*kwargs将任意数量的关键字参数收集为字典,二者结合可提升函数灵活性和扩展性。

在Python里,
*args
**kwargs
*args
**kwargs
当我们在Python中定义一个函数时,如果事先不确定调用者会传入多少个参数,或者传入哪些关键字参数,
*args
**kwargs
具体来说,
*args
*args
def my_sum(*numbers):
# numbers 在这里是一个元组
total = 0
for num in numbers:
total += num
return total
print(my_sum(1, 2, 3)) # 输出 6
print(my_sum(10, 20, 30, 40)) # 输出 100
print(my_sum()) # 输出 0而
**kwargs
立即学习“Python免费学习笔记(深入)”;
def print_config(**options):
# options 在这里是一个字典
print("配置详情:")
for key, value in options.items():
print(f" {key}: {value}")
print_config(host="localhost", port=8080, debug=True)
# 输出:
# 配置详情:
# host: localhost
# port: 8080
# debug: True
print_config(user="admin")
# 输出:
# 配置详情:
# user: admin这两者结合使用,就能创建一个极其通用的函数签名,能够处理几乎所有可能的参数组合。通常,我们会把它们放在函数参数列表的末尾,遵循的顺序是:普通位置参数 ->
*args
**kwargs
def comprehensive_function(a, b, *args, default_val=100, **kwargs):
print(f"a: {a}")
print(f"b: {b}")
print(f"额外的位置参数 (args): {args}")
print(f"默认值 (default_val): {default_val}")
print(f"额外的关键字参数 (kwargs): {kwargs}")
comprehensive_function(1, 2, 3, 4, name="Alice", age=30, default_val=200)
# 输出:
# a: 1
# b: 2
# 额外的位置参数 (args): (3, 4)
# 默认值 (default_val): 200
# 额外的关键字参数 (kwargs): {'name': 'Alice', 'age': 30}看到没,这种灵活性是无与伦比的。它让我的代码在面对需求变化时,能有更大的余地去适应,而不是动不动就改函数签名。
*args
*args
想象一下,你有一个函数,它的核心任务是处理一系列同类型的数据,但你不知道这次会有多少个数据。比如,计算平均值,或者把多个文件路径合并。如果没有
*args
*args
举个例子,我经常需要写一些日志记录的辅助函数,可能会有不同的消息段:
import datetime
def log_message(level, *parts):
"""记录一条带有不同部分的日志消息"""
timestamp = datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S")
# parts 是一个元组,包含了 'Hello', 'World', '!'
full_message = " ".join(str(p) for p in parts)
print(f"[{timestamp}] [{level.upper()}]: {full_message}")
log_message("INFO", "应用程序启动", "版本", 1.0)
# 输出: [2023-10-27 10:30:00] [INFO]: 应用程序启动 版本 1.0
log_message("WARNING", "配置项缺失")
# 输出: [2023-10-27 10:30:00] [WARNING]: 配置项缺失这里
*parts
需要注意的是,
*args
**kwargs
def mixed_params(fixed_arg, *dynamic_args, key_arg="default"):
print(f"Fixed: {fixed_arg}")
print(f"Dynamic: {dynamic_args}")
print(f"Key: {key_arg}")
mixed_params(1, 2, 3, 4, key_arg="custom")
# Fixed: 1
# Dynamic: (2, 3, 4)
# Key: custom如果我尝试把
*dynamic_args
fixed_arg
key_arg
一个常见的误区是,当你想把
*args
*
another_function(*args)
another_function(args)
**kwargs
**kwargs
我个人在开发Web应用或API客户端时,就特别依赖
**kwargs
**kwargs
import requests
def make_api_call(url, method="GET", **request_options):
"""
模拟一个灵活的API调用函数
request_options 可以包含 headers, params, json, timeout 等
"""
print(f"正在向 {url} 发送 {method} 请求...")
print(f"请求选项: {request_options}")
# 实际项目中,这里会调用 requests.request(method, url, **request_options)
# 模拟返回一个响应对象
class MockResponse:
def __init__(self, status_code, text):
self.status_code = status_code
self.text = text
def json(self):
import json
return json.loads(self.text)
if url.endswith("/success"):
return MockResponse(200, '{"status": "success", "data": {"id": 123}}')
else:
return MockResponse(404, '{"error": "Not Found"}')
# 调用示例
response = make_api_call("https://api.example.com/data/success",
method="POST",
headers={"Authorization": "Bearer token123"},
json={"query": "test"},
timeout=5)
print(f"响应状态码: {response.status_code}")
print(f"响应内容: {response.json()}")
# 另一个调用
response = make_api_call("https://api.example.com/users", params={"page": 1, "limit": 10})
print(f"响应状态码: {response.status_code}")在这个
make_api_call
request_options
headers
json
timeout
**request_options
requests.request
和
*args
**kwargs
*args
**kwargs
一个很重要的点是,如果传入的关键字参数名与函数中已有的命名参数(无论是普通参数还是带有默认值的参数)冲突,那么该参数值会被对应的命名参数接收,而不会进入
**kwargs
def process_data(name, age=30, **extra_info):
print(f"Name: {name}")
print(f"Age: {age}")
print(f"Extra Info: {extra_info}")
process_data("Bob", age=25, city="New York", occupation="Engineer")
# Name: Bob
# Age: 25 (这里的 age=25 覆盖了默认值 30)
# Extra Info: {'city': 'New York', 'occupation': 'Engineer'}
process_data("Charlie", city="London")
# Name: Charlie
# Age: 30 (使用了默认值)
# Extra Info: {'city': 'London'}可以看到,
age=25
age
extra_info
*args
**kwargs
*args
**kwargs
1. 函数装饰器 (Decorators): 这是它们最经典、也是最有力的应用之一。装饰器本质上是一个函数,它接收另一个函数作为参数,并返回一个新的函数。为了让装饰器能够作用于任何签名的函数(无论它接受多少位置参数或关键字参数),
*args
**kwargs
import time
def timer(func):
"""一个简单的计时装饰器"""
def wrapper(*args, **kwargs):
start_time = time.time()
result = func(*args, **kwargs) # 使用 *args 和 **kwargs 转发所有参数
end_time = time.time()
print(f"函数 {func.__name__} 执行耗时: {end_time - start_time:.4f} 秒")
return result
return wrapper
@timer
def long_running_task(iterations, message):
for _ in range(iterations):
_ = 1 + 1 # 模拟计算
print(f"任务完成: {message}")
return "Done"
@timer
def short_task():
print("短任务完成")
return "Short Done"
long_running_task(1_000_000, "大量计算")
short_task()在这个
timer
wrapper
*args
**kwargs
func
timer
2. 函数转发/代理 (Function Forwarding/Proxying): 当一个函数的主要职责是调用另一个函数,并把自己的参数全部传递过去时,
*args
**kwargs
class APIClient:
def __init__(self, base_url):
self.base_url = base_url
def _send_request(self, method, path, **kwargs):
"""内部方法,处理实际的HTTP请求"""
url = f"{self.base_url}{path}"
print(f"Sending {method} request to {url} with options: {kwargs}")
# 实际会用 requests.request(method, url, **kwargs)
return {"status": "success", "data": "mock_data"}
def get(self, path, **kwargs):
"""GET请求的封装,转发所有参数"""
return self._send_request("GET", path, **kwargs)
def post(self, path, data, **kwargs):
"""POST请求的封装,转发所有参数,同时添加 data 参数"""
kwargs['json'] = data # 或者直接在 kwargs 里添加
return self._send_request("POST", path, **kwargs)
client = APIClient("https://myapi.com")
client.get("/users", params={"id": 123}, headers={"Auth": "token"})
client.post("/items", data={"name": "New Item"}, timeout=10)这里的
get
post
**kwargs
_send_request
requests
3. 构建灵活的类初始化 (__init__
**kwargs
class DynamicConfig:
def __init__(self, default_setting="foo", **custom_settings):
self.settings = {"default_setting": default_setting}
self.settings.update(custom_settings) # 合并自定义设置
def get_setting(self, key):
return self.settings.get(key, "Setting not found")
config1 = DynamicConfig()
print(config1.get_setting("default_setting")) # foo
config2 = DynamicConfig(default_setting="bar", database_url="sqlite:///db.db", cache_size=1024)
print(config2.get_setting("default_setting")) # bar
print(config2.get_setting("database_url")) # sqlite:///db.db这里
__init__
**custom_settings
self.settings
4. 继承和方法重写: 在面向对象编程中,子类方法经常需要调用父类方法,并传递相同的参数。使用
*args
**kwargs
super()
class BaseProcessor:
def process(self, *args, **kwargs):
print(f"BaseProcessor processing with args: {args}, kwargs: {kwargs}")
# 实际处理逻辑
class AdvancedProcessor(BaseProcessor):
def process(self, *args, **kwargs):
print("AdvancedProcessor doing some pre-processing...")
# 添加子类特有的逻辑
super().process(*args, **kwargs) # 转发所有参数给父类
print("AdvancedProcessor doing some post-processing...")
processor = AdvancedProcessor()
processor.process(1, 2, name="test", debug=True)这种模式保证了继承链上的参数传递的完整性和一致性。
总的来说,
*args
**kwargs
以上就是args和**kwargs在python中是什么意思_Python中args与**kwargs的核心作用解析的详细内容,更多请关注php中文网其它相关文章!
python怎么学习?python怎么入门?python在哪学?python怎么学才快?不用担心,这里为大家提供了python速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号