0

0

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

雪夜

雪夜

发布时间:2025-08-17 17:12:01

|

1352人浏览过

|

来源于php中文网

原创

使用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

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

超能文献
超能文献

超能文献是一款革命性的AI驱动医学文献搜索引擎。

下载
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
这样的任意关键字参数,并把它们打包成一个字典。

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

另外一个和

*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函数带来了极大的灵活性。但在使用它时,我们应该权衡其带来的便利性和可能对代码可读性、可维护性造成的影响。选择最能清晰表达意图的方式,永远是编程的最佳实践。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

910

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

597

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

294

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

210

2025.08.29

treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

546

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

27

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

43

2026.01.06

什么是低代码
什么是低代码

低代码是一种软件开发方法,使用预构建的组件可快速构建应用程序,无需大量编程。想了解更多低代码的相关内容,可以阅读本专题下面的文章。

298

2024.05.21

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

48

2026.02.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Django 教程
Django 教程

共28课时 | 4.7万人学习

Excel 教程
Excel 教程

共162课时 | 19.7万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

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