飞翔飞翔
主页
  • 计算机基础

    • TCP/IP协议
    • Linux命令
  • 数据库

    • SQL教程
  • 编程语言

    • C语言
    • Python2
    • Python3
  • 数据格式

    • JSON教程
  • 工具

    • Markdown指南
  • Git

    • GitFlow
  • Quartz

    • Quartz教程
  • Java

    • Java设计模式
  • 缓存

    • Redis教程
联系
阿里云
主页
  • 计算机基础

    • TCP/IP协议
    • Linux命令
  • 数据库

    • SQL教程
  • 编程语言

    • C语言
    • Python2
    • Python3
  • 数据格式

    • JSON教程
  • 工具

    • Markdown指南
  • Git

    • GitFlow
  • Quartz

    • Quartz教程
  • Java

    • Java设计模式
  • 缓存

    • Redis教程
联系
阿里云
  • 学习路径
  • 第1章 认识Python

    • Python 历史与特点
    • Python 2 与 Python 3 的核心差异
    • 安装与运行 Python 2.7.18
    • 编码规范 PEP 8
  • 第2章 基础语法

    • 变量与对象
    • 数字类型
    • 字符串 str
    • Unicode 字符串
    • 运算符
    • 空值 None
  • 第3章 流程控制

    • if 条件语句
    • if-else 条件语句
    • if-elif-else 多分支
    • 条件表达式(三元运算符)
    • while 循环
    • for 循环
    • range 与 xrange
    • 循环控制:break、continue、pass
    • 循环 else 子句
  • 第4章 数据结构

    • 列表基础
    • 列表方法
    • 列表推导式
    • 元组
    • 字典基础
    • 字典方法
    • 字典循环技巧
    • 集合
    • 序列解包
    • 序列比较
  • 第5章 函数

    • 定义函数
    • 参数传递机制
    • 默认参数
    • 关键字参数
    • 可变参数
    • Lambda 表达式
    • 文档字符串
    • 函数对象
  • 第6章 模块与包

    • import 导入
    • 模块搜索路径
    • name 与主程序
    • 编译文件 .pyc 与 .pyo
    • 包结构
    • dir() 函数
  • 第7章 文件与IO

    • 打开与关闭文件
    • 文件读写方法
    • with 上下文管理器
    • 格式化输出:% 操作符
    • 格式化输出:str.format()
    • JSON 序列化
  • 第8章 面向对象

    • 类定义与实例化
    • init 构造方法
    • 类变量与实例变量
    • 方法调用与 self
    • 继承基础
    • 多重继承
    • 新式类与旧式类
    • 私有变量与名称改写
    • 属性装饰器 property
    • 类方法与静态方法
    • 魔术方法
    • 空类与数据记录
  • 第9章 异常处理

    • 异常类型
    • try-except
    • try-except-else-finally
    • 抛出异常 raise
    • 自定义异常
    • with 语句与上下文管理器
  • 第10章 迭代器与生成器

    • 迭代器协议
    • 生成器函数
    • 生成器表达式
    • itertools模块
  • 第11章 标准库精要

    • os模块
    • sys模块
    • datetime模块
    • re模块
    • json模块
    • collections模块
    • math与random模块
    • urllib2与网络请求
    • subprocess与命令执行
    • threading与并发
    • unittest与测试
    • 虚拟环境与包管理
  • 第12章 工程实践

    • 调试技巧
    • 性能分析
    • 文档与注释
    • 下一步学习

可变参数

有时函数需要接受任意数量的参数。Python 提供了 *args 和 **kwargs 两种机制,分别收集多余的位置参数和关键字参数。

*args:收集多余位置参数

def total(*args):
    result = 0
    for num in args:
        result += num
    return result

print total()           # 0
print total(1)          # 1
print total(1, 2, 3)    # 6
print total(1, 2, 3, 4, 5)  # 15

*args 把多余的位置参数收集成一个元组。在函数内部,args 是一个普通元组,可以遍历、索引、求长度。

def show_args(first, *args):
    print "first:", first
    print "rest:", args
    print "count:", len(args)

show_args("a", "b", "c", "d")
# first: a
# rest: ('b', 'c', 'd')
# count: 3

**kwargs:收集多余关键字参数

def show_info(**kwargs):
    for key, value in kwargs.iteritems():
        print "%s = %s" % (key, value)

show_info(name="Alice", age=25, city="Beijing")
# name = Alice
# age = 25
# city = Beijing

**kwargs 把多余的关键字参数收集成一个字典。在函数内部,kwargs 是一个普通字典。

组合使用

def func(a, b, *args, **kwargs):
    print "a:", a
    print "b:", b
    print "args:", args
    print "kwargs:", kwargs

func(1, 2, 3, 4, 5, x=10, y=20)
# a: 1
# b: 2
# args: (3, 4, 5)
# kwargs: {'y': 20, 'x': 10}

参数顺序必须是:位置参数 → *args → 关键字参数 → **kwargs。

解包调用

调用函数时,可以用 * 和 ** 解包序列和字典:

def add(a, b, c):
    return a + b + c

nums = [1, 2, 3]
print add(*nums)        # 6,等价于 add(1, 2, 3)

params = {"a": 1, "b": 2, "c": 3}
print add(**params)     # 6,等价于 add(a=1, b=2, c=3)

组合使用:

def func(a, b, c, d, e):
    return a + b + c + d + e

args = [1, 2]
kwargs = {"d": 4, "e": 5}
print func(*args, 3, **kwargs)      # 15
# 等价于 func(1, 2, 3, d=4, e=5)

实际应用

装饰器:

def my_decorator(func):
    def wrapper(*args, **kwargs):
        print "Before call"
        result = func(*args, **kwargs)
        print "After call"
        return result
    return wrapper

@my_decorator
def greet(name):
    print "Hello,", name

greet("Alice")
# Before call
# Hello, Alice
# After call

装饰器不知道被装饰函数的参数签名,用 *args, **kwargs 可以适配任何函数。

转发参数:

def log_and_call(func, *args, **kwargs):
    print "Calling %s with %s, %s" % (func.__name__, args, kwargs)
    return func(*args, **kwargs)

log_and_call(max, 1, 5, 3)
# Calling max with (1, 5, 3), {}
# 返回 5

命名约定

  • *args:多余位置参数(可以叫其他名字,但 args 是约定)
  • **kwargs:多余关键字参数(可以叫其他名字,但 kwargs 是约定)

常见错误

*args 已经是元组,不要再用 *:

def bad(*args):
    print *args         # SyntaxError!print 是语句,不能这样用

def good(*args):
    print args          # 正确
    print " ".join(str(a) for a in args)

**kwargs 的键必须是字符串:

def func(**kwargs):
    pass

func(**{1: "a"})        # TypeError: func() keywords must be strings
上一页
关键字参数
下一页
Lambda 表达式