Python函数怎样用args 接收任意数量的位置参数 Python函数可变位置参数的使用技巧​

雪夜
发布: 2025-08-17 17:12:01
原创
1220人浏览过
使用args可接收任意数量的位置参数,将其打包为元组,适用于参数数量不确定的场景。在函数定义中,args应置于普通参数之后,可与普通参数和kwargs混合使用,但需注意参数顺序。*args提升函数通用性,但可能降低可读性,当参数语义明确或数据为逻辑集合时,建议使用命名参数、列表传参或kwargs替代。

python函数怎样用args 接收任意数量的位置参数 python函数可变位置参数的使用技巧​

Python函数通过

*args
登录后复制
语法能够非常灵活地接收任意数量的位置参数。简单来说,它就像一个“收集器”,把所有多余的、没有被明确定义为命名参数的位置参数,打包成一个元组(tuple)供函数内部使用。这在编写通用性更强的函数时尤其方便,比如你想写一个可以处理不定数量输入数据的函数。

在Python里,当你在函数定义时看到一个参数前面带着星号(

*
登录后复制
),比如
*args
登录后复制
,它就代表着“可变位置参数”。这意味着,当你调用这个函数时,无论你传入多少个位置参数(除了那些已经被明确命名的参数),它们都会被这个
*args
登录后复制
收集起来,然后以一个元组的形式,赋值给
args
登录后复制
这个变量。

举个例子,假设我们想写一个函数来计算任意多个数字的和:

def calculate_sum(*numbers):
    total = 0
    for num in numbers:
        total += num
    print(f"这些数字的总和是: {total}")

# 调用时可以传入不同数量的参数
calculate_sum(1, 2, 3)
calculate_sum(10, 20)
calculate_sum(5)
calculate_sum() # 也可以不传,此时numbers会是一个空元组
登录后复制

这里,

numbers
登录后复制
在函数内部就是一个元组。第一次调用时,
numbers
登录后复制
(1, 2, 3)
登录后复制
;第二次是
(10, 20)
登录后复制
,以此类推。这种机制极大地提升了函数的通用性和适应性。

立即学习Python免费学习笔记(深入)”;

如何在函数定义中正确使用 *args?

使用

*args
登录后复制
其实挺直观的,但有几个小细节值得我们注意。最关键的一点是它在函数参数列表中的位置。通常情况下,
*args
登录后复制
应该放在所有普通的位置参数之后。这是因为普通参数会优先匹配传入的值,剩下的那些没有“归宿”的位置参数,才会一股脑儿地被
*args
登录后复制
收纳。

比如,如果你有一个函数需要一个固定参数,然后再接受任意数量的额外参数:

def greet_and_others(main_person, *guests):
    print(f"你好,{main_person}!")
    if guests:
        print("今天还有这些朋友来了:")
        for guest in guests:
            print(f"- {guest}")
    else:
        print("今天只有你一个人。")

greet_and_others("张三", "李四", "王五")
greet_and_others("赵六")
登录后复制

这里,

main_person
登录后复制
会首先接收第一个传入的参数,而
*guests
登录后复制
则会收集剩下的所有位置参数。如果
*guests
登录后复制
前面还有其他参数,它们会按照位置顺序依次匹配。

值得一提的是,

args
登录后复制
这个名字只是一个约定俗成的惯例(convention),你可以用任何合法的变量名来代替它,比如
*items
登录后复制
*data
登录后复制
*values
登录后复制
等,只要前面带个星号就行。不过,我个人还是倾向于使用
*args
登录后复制
,因为它能一眼就让人明白这个参数的用途。

一个常见的误区是,你不能在

*args
登录后复制
之后再定义普通的位置参数。Python会认为
*args
登录后复制
已经把所有位置参数都“吃”掉了,所以后面的参数就没法按位置匹配了。当然,你可以在
*args
登录后复制
之后定义关键字参数(keyword-only arguments),这通常需要一个单独的星号或者
**kwargs
登录后复制
来分隔,但那是另一个话题了。

*args 与普通参数、**kwargs 的混合使用场景与注意事项

在实际开发中,我们经常会遇到需要同时处理固定参数、任意位置参数和任意关键字参数的场景。Python的函数签名设计得非常巧妙,允许我们以一种清晰的方式来组合它们。标准的顺序是:普通位置参数,然后是

*args
登录后复制
,接着是关键字参数(如果有的话,通常用
*
登录后复制
来分隔),最后是
**kwargs
登录后复制

一个典型的函数签名看起来可能是这样的:

def process_data(required_id, *values, description="Default", **options):
    print(f"处理ID: {required_id}")
    if values:
        print(f"收集到的值: {values}")
    print(f"描述: {description}")
    if options:
        print(f"额外选项: {options}")

# 示例调用
process_data(101, 1, 2, 3, description="详细数据", verbose=True, debug_mode=False)
process_data(202, "apple", "banana")
process_data(303, verbose=True)
登录后复制

这里,

required_id
登录后复制
是必须提供的固定位置参数;
*values
登录后复制
会收集
1, 2, 3
登录后复制
"apple", "banana"
登录后复制
这些额外的位置参数;
description
登录后复制
是一个带默认值的关键字参数;而
**options
登录后复制
则会收集像
verbose=True
登录后复制
,
debug_mode=False
登录后复制
这样的任意关键字参数,并把它们打包成一个字典。

即构数智人
即构数智人

即构数智人是由即构科技推出的AI虚拟数字人视频创作平台,支持数字人形象定制、短视频创作、数字人直播等。

即构数智人 36
查看详情 即构数智人

我发现这种组合在编写一些通用工具函数时特别有用,比如一个日志记录器,你可能需要一个固定的消息级别,然后是任意数量的日志内容片段,最后是一些配置日志格式的选项。

另外一个和

*args
登录后复制
紧密相关的概念是“参数解包”(argument unpacking),也就是在调用函数时使用
*
登录后复制
操作符。这和函数定义时的
*args
登录后复制
正好是反过来的过程。当你有了一个列表或元组,想把它的元素作为单独的位置参数传给函数时,就可以用
*
登录后复制

def sum_all(a, b, c):
    print(a + b + c)

my_numbers = [10, 20, 30]
sum_all(*my_numbers) # 相当于 sum_all(10, 20, 30)

# 这在处理 *args 收集到的元组时也很有用
def log_messages(*messages):
    print("--- 日志开始 ---")
    for msg in messages:
        print(msg)
    print("--- 日志结束 ---")

all_logs = ("错误发生", "用户登录", "数据更新")
log_messages(*all_logs) # 将元组解包后传入 log_messages
登录后复制

这种解包能力,在我看来,是Python函数参数处理机制中最优雅和灵活的部分之一。它让数据的传递和转换变得异常流畅。

什么时候应该避免使用 *args,以及替代方案?

尽管

*args
登录后复制
非常强大和灵活,但并非所有场景都适合使用它。有时候,过度依赖
*args
登录后复制
反而会降低代码的可读性和可维护性。

我个人在决定是否使用

*args
登录后复制
时,会考虑以下几点:

  1. 参数语义是否明确? 如果函数接收的参数数量虽然可变,但每个参数都有其特定的、独立的含义(例如,一个函数需要接收不同形状的几何体的尺寸,但每个尺寸都代表长、宽、高),那么使用命名参数会比
    *args
    登录后复制
    清晰得多。比如,
    calculate_area(width, height)
    登录后复制
    calculate_area(*dimensions)
    登录后复制
    更直观。
  2. 可读性优先于“通用性”。如果参数数量通常是固定的几个,只是偶尔会多一个,那么明确列出这些参数通常是更好的选择。
    *args
    登录后复制
    虽然“通用”,但它隐藏了参数的具体含义,让其他人(包括未来的你)在阅读代码时,需要花更多精力去理解
    args
    登录后复制
    元组里每个元素到底代表什么。
  3. 类型提示和静态分析的便利性。虽然现在Python的类型提示系统也支持
    *args
    登录后复制
    (比如
    *args: int
    登录后复制
    *args: str
    登录后复制
    ),但它毕竟不如明确的命名参数那样能够提供细粒度的类型信息。对于需要严格类型检查的项目,这可能会是一个考虑因素。

那么,当

*args
登录后复制
不是最佳选择时,我们有哪些替代方案呢?

  • 使用明确的命名参数: 这是最常见、最推荐的方式。当参数数量固定且语义清晰时,直接定义它们的名字。

  • 传递列表或元组作为单个参数: 如果你的一组数据确实是一个逻辑上的集合,而不是多个独立的参数,那么直接将这个列表或元组作为函数的单个参数传入,会比用

    *args
    登录后复制
    更符合语义。

    # 避免 *args,直接传列表
    def process_items_list(item_list):
        for item in item_list:
            print(f"处理: {item}")
    
    my_items = ["apple", "banana", "cherry"]
    process_items_list(my_items)
    登录后复制
  • 使用`kwargs

    处理可选的、命名参数:** 如果你的函数需要接受大量可选的配置项,且这些配置项都是键值对形式的,那么
    登录后复制
    *kwargs
    (可变关键字参数)通常是比
    登录后复制
    args`更好的选择。它能让调用者通过参数名来指定配置,清晰明了。

  • 定义数据类或对象: 对于更复杂的数据结构,如果多个参数实际上是某个实体或概念的属性,那么定义一个类或使用

    dataclasses
    登录后复制
    来封装这些属性,并传递这个对象实例,会大大提高代码的组织性和可维护性。

总之,

*args
登录后复制
是一个非常有用的工具,它为Python函数带来了极大的灵活性。但在使用它时,我们应该权衡其带来的便利性和可能对代码可读性、可维护性造成的影响。选择最能清晰表达意图的方式,永远是编程的最佳实践。

以上就是Python函数怎样用args 接收任意数量的位置参数 Python函数可变位置参数的使用技巧​的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号