13_python基础—函数进阶(参数、返回值、递归)

简介: 13_python基础—函数进阶(参数、返回值、递归)

一、函数参数和返回值的作用


函数根据 有没有参数 以及 有没有返回值,可以 相互组合,一共有 4 种 组合形式


无参数,无返回值

无参数,有返回值

有参数,无返回值

有参数,有返回值


7daf3a041a04717a7e497510be250db7.png


定义函数时,是否接收参数,或者是否返回结果,是根据 实际的功能需求 来决定的!


如果函数 内部处理的数据不确定,就可以将外界的数据以参数传递到函数内部

如果希望一个函数 执行完成后,向外界汇报执行结果,就可以增加函数的返回值


1.1 无参数,无返回值


此类函数,不接收参数,也没有返回值,应用场景如下:


只是单纯地做一件事情,例如 显示菜单

在函数内部 针对全局变量进行操作,例如:新建名片,最终结果 记录在全局变量 中


注意:


如果全局变量的数据类型是一个 可变类型,在函数内部可以使用 方法 修改全局变量的内容 —— 变量的引用不会改变

在函数内部,使用赋值语句 才会 修改变量的引用


1.2 无参数,有返回值


此类函数,不接收参数,但是有返回值,应用场景如下:


采集数据,例如 温度计,返回结果就是当前的温度,而不需要传递任何的参数


1.3 有参数,无返回值


此类函数,接收参数,没有返回值,应用场景如下:


函数内部的代码保持不变,针对 不同的参数 处理 不同的数据

例如 名片管理系统 针对 找到的名片 做 修改、删除 操作


1.4 有参数,有返回值


此类函数,接收参数,同时有返回值,应用场景如下:


函数内部的代码保持不变,针对 不同的参数 处理 不同的数据,并且 返回期望的处理结果

例如 名片管理系统 使用 字典默认值 和 提示信息 提示用户输入内容

如果输入,返回输入内容

如果没有输入,返回字典默认值


二、函数的返回值 进阶


在程序开发中,有时候,会希望 一个函数执行结束后,告诉调用者一个结果,以便调用者针对具体的结果做后续的处理

返回值 是函数 完成工作后,最后 给调用者的 一个结果

在函数中使用 return 关键字可以返回结果

调用函数一方,可以 使用变量 来 接收 函数的返回结果

问题:一个函数执行后能否返回多个结果?


2.1 示例 —— 温度和湿度测量


假设要开发一个函数能够同时返回当前的温度和湿度

先完成返回温度的功能如下:


def measure():
    """返回当前的温度"""
    print("开始测量...")
    temp = 39
    print("测量结束...")
    return temp
result = measure()
print(result)
  • 在利用 元组 在返回温度的同时,也能够返回 湿度


  • 改造如下:
def measure():
    """测量温度和湿度"""
    print("测量开始...")
    temp = 39
    wetness = 50
    print("测量结束...")
    # 元组-可以包含多个数据,因此可以使用元组让函数一次返回多个值
    # 如果函数返回的类型是元组,小括号可以省略
    # return (temp, wetness)
    return temp, wetness
result = measure()
print(result)


提示:如果一个函数返回的是元组,括号可以省略


技巧


  • Python 中,可以 将一个元组 使用 赋值语句 同时赋值给 多个变量
  • 注意:变量的数量需要和元组中的元素数量保持一致
result = temp, wetness = measure()
def measure():
    """测量温度和湿度"""
    print("测量开始...")
    temp = 39
    wetness = 50
    print("测量结束...")
    # 元组-可以包含多个数据,因此可以使用元组让函数一次返回多个值
    # 如果函数返回的类型是元组,小括号可以省略
    # return (temp, wetness)
    return temp, wetness
# 元组
result = measure()
print(result)
# 需要单独的处理温度或者湿度 - 不方便
print(result[0])
print(result[1])
# 如果函数返回的类型是元组,同时希望单独的处理元组中的元素
# 可以使用多个变量,一次接收函数的返回结果
# 注意:使用多个变量接收结果时,变量的个数应该和元组中元素的个数保持一致
gl_temp, gl_wetness = measure()
print(gl_temp)
print(gl_wetness)


2.2 面试题 —— 交换两个数字


题目要求


  1. 有两个整数变量 a = 6, b = 100
  2. 不使用其他变量,交换两个变量的值


解法 1 —— 使用其他变量


# 解法 1 - 使用临时变量
c = b
b = a
a = c


19849ab11317479cacceb2fd00775f74.png

解法 2 —— 不使用临时变量


# 解法 2 - 不使用临时变量
a = a + b
b = a - b
a = a - b


20d48efc7ceb41058662472241cd6d4e.png


解法 3 —— Python 专有,利用元组


# a, b = (b, a)
# 提示:等号右边是一个元组,只是把小括号省略了
a, b = b, a


三、函数的参数 进阶(重点)


3.1. 不可变和可变的参数


问题 1:在函数内部,针对参数使用 赋值语句,会不会影响调用函数时传递的 实参变量? —— 不会!


无论传递的参数是 可变 还是 不可变

只要 针对参数 使用 赋值语句,会在 函数内部 修改 局部变量的引用,不会影响到 外部变量的引用


def demo(num, num_list):
    print("函数内部的代码")
    # 在函数内部,针对参数使用赋值语句,不会修改到外部的实参变量
    num = 100
    num_list = [1, 2, 3]
    print(num)
    print(num_list)
    print("函数执行完成")
gl_num = 99
gl_list = [4, 5, 6]
demo(gl_num, gl_list)
print(gl_num)
print(gl_list)   

528db46c5b0aed1931978075f218139b.png


问题 2:如果传递的参数是 可变类型,在函数内部,使用 方法 修改了数据的内容,同样会影响到外部的数据

def demo(num_list):
    print("函数内部的代码")
    # 使用方法修改列表的内容
    num_list.append(9)
    print(num_list)
    print("函数执行完成")
gl_list = [1, 2, 3]
demo(gl_list)
print(gl_list)


1f368b3e84a24245987192606695f70e.png


面试题 —— +=


  • python 中,列表变量调用 += 本质上是在执行列表变量的 extend 方法(另外一个列表的内容整合到当前列表中),不会修改变量的引用
def demo(num, num_list):
    print("函数开始")
    # num = num + num
    num += num
    # 列表变量使用 + 不会做相加再赋值的操作 !
    # num_list = num_list + num_list
    # 本质上是在调用列表的 extend 方法
    # 函数执行结束后,外部数据同样会发生变化
    num_list += num_list
    # num_list.extend(num_list)
    print(num)
    print(num_list)
    print("函数完成")
gl_num = 9
gl_list = [1, 2, 3]
demo(gl_num, gl_list)
print(gl_num)
print(gl_list)


函数开始

18

[1, 2, 3, 1, 2, 3]

函数完成

9

[1, 2, 3, 1, 2, 3]


3.2 缺省参数


  • 定义函数时,可以给 某个参数 指定一个默认值,具有默认值的参数就叫做 缺省参数
  • 调用函数时,如果没有传入 缺省参数 的值,则在函数内部使用定义函数时指定的 参数默认值
  • 函数的缺省参数,将常见的值设置为参数的缺省值,从而 简化函数的调用
  • 例如:对列表排序的方法
gl_num_list = [6, 3, 9]
# 默认就是升序排序,因为这种应用需求更多
gl_num_list.sort()
print(gl_num_list)
# 只有当需要降序排序时,才需要传递 `reverse` 参数
gl_num_list.sort(reverse=True)
print(gl_num_list)


3.2.1 指定函数的缺省参数(参数后使用赋值语句)


  • 在参数后使用赋值语句,可以指定参数的缺省值
def print_info(name, gender=True):
    """
    :param name: 班上同学的姓名
    :param gender: True 男生 False 女生
    """
    gender_text = "男生"
    if not gender:
        gender_text = "女生"
    print("%s 是 %s" % (name, gender_text))
# 假设班上的同学,男生居多!
# 提示:在指定缺省参数的默认值时,应该使用最常见的值作为默认值!
print_info("小明")
print_info("小美", False)


提示


  1. 缺省参数,需要使用 最常见的值 作为默认值!
  2. 如果一个参数的值 不能确定,则不应该设置默认值,具体的数值在调用函数时,由外界传递!


3.2.2 缺省参数的注意事项


1) 缺省参数的定义位置
  • 必须保证带有默认值的缺省参数在参数列表末尾


  • 所以,以下定义是错误的!


def print_info(name, gender=True, title):


2) 调用带有多个缺省参数的函数


  • 调用函数时,如果有 多个缺省参数需要指定参数名,这样解释器才能够知道参数的对应关系!
def print_info(name, title="", gender=True):
    """
    :param title: 职位
    :param name: 班上同学的姓名
    :param gender: True 男生 False 女生
    """
    gender_text = "男生"
    if not gender:
        gender_text = "女生"
    print("%s%s 是 %s" % (title, name, gender_text))
# 提示:在指定缺省参数的默认值时,应该使用最常见的值作为默认值!
print_info("小明")
print_info("老王", title="班长")
print_info("小美", gender=False)


3.3 多值参数(元组、字典)


3.3.1 定义支持多值参数的函数


不定长参数也叫可变参数。


有时可能需要 一个函数 能够处理的参数 个数 是不确定的,这个时候,就可以使用 多值参数


python 中有 两种 多值参数:


参数名前增加 一个 * 可以接收 元组

参数名前增加 两个 * 可以接收 字典

一般在给多值参数命名时,习惯使用以下两个名字


*args —— 存放 元组 参数,前面有一个 *

**kwargs —— 存放 字典 参数,前面有两个 *

args 是 arguments 的缩写,有变量的含义


kw 是 keyword 的缩写,kwargs 可以记忆 键值对参数


def demo(num, *args, **kwargs):
    print(num)
    print(args)
    print(kwargs)
demo(1, 2, 3, 4, 5, name="小明", age=18, gender=True)


1
(2, 3, 4, 5)
{‘name’: ‘小明’, ‘age’: 18, ‘gender’: True}


提示:多值参数 的应用会经常出现在网络上一些大牛开发的框架中,知道多值参数,有利于我们能够读懂大牛的代码


3.3.2 多值参数案例 —— 计算任意多个数字的和


需求


定义一个函数 sum_numbers,可以接收的 任意多个整数

功能要求:将传递的 所有数字累加 并且返回累加结果


def sum_numbers(*args):
    num = 0
    # 遍历 args 元组顺序求和
    for n in args:
        num += n
    return num
print(sum_numbers(1, 2, 3))      # 6


3.3.3 元组和字典的拆包(知道)


在调用带有多值参数的函数时,如果希望:

将一个 元组变量,直接传递给 args

将一个 字典变量,直接传递给 kwargs

就可以使用 拆包,简化参数的传递,拆包 的方式是:

在 元组变量前,增加 一个 *

在 字典变量前,增加 两个 *


def demo(*args, **kwargs):
    print(args)
    print(kwargs)
# 需要将一个元组变量/字典变量传递给函数对应的参数
gl_nums = (1, 2, 3)
gl_xiaoming = {"name": "小明", "age": 18}
# 会把 num_tuple 和 xiaoming 作为元组传递个 args
# demo(gl_nums, gl_xiaoming)
# 拆包语法,简化元组变量/字典变量的传递
demo(*gl_nums, **gl_xiaoming)


四、函数的递归


函数调用自身的 编程技巧 称为递归


4.1 递归函数的特点


特点


一个函数 内部 调用自己

函数内部可以调用其他函数,当然在函数内部也可以调用自己

代码特点


函数内部的 代码 是相同的,只是针对 参数 不同,处理的结果不同

当 参数满足一个条件 时,函数不再执行

这个非常重要,通常被称为递归的出口,否则 会出现死循环!


示例代码

def sum_numbers(num):
    print(num)
    # 递归的出口很重要,否则会出现死循环
    if num == 1:
        return  # 返回函数调用的位置
    # 自己调用自己,传递另外一个参数
    sum_numbers(num - 1)
sum_numbers(3)   


e8173b1a77162892e7706c99b659a827.png


4.2 递归案例 —— 计算数字累加


需求


  1. 定义一个函数 sum_numbers
  2. 能够接收一个 num 的整数参数
  3. 计算 1 + 2 + … num 的结果
def sum_numbers(num):
    # 1.出口
    if num == 1:
        return 1
    # 2.假设 sum_numbers 能够完成 num - 1 的累加
    temp = sum_numbers(num - 1)
    # 函数内部的核心算法就是 两个数字的相加
    return num + temp
print(sum_numbers(3))     # 6


c06e82f2cbec4e685e380fc592f4fc83.png

提示:递归是一个 编程技巧,初次接触递归会感觉有些吃力!在处理 不确定的循环条件时,格外的有用,例如:遍历整个文件目录的结构

目录
相关文章
|
1月前
|
开发者 Python
Python入门:8.Python中的函数
### 引言 在编写程序时,函数是一种强大的工具。它们可以将代码逻辑模块化,减少重复代码的编写,并提高程序的可读性和可维护性。无论是初学者还是资深开发者,深入理解函数的使用和设计都是编写高质量代码的基础。本文将从基础概念开始,逐步讲解 Python 中的函数及其高级特性。
Python入门:8.Python中的函数
|
27天前
|
C语言 Python
Python学习:内建属性、内建函数的教程
本文介绍了Python中的内建属性和内建函数。内建属性包括`__init__`、`__new__`、`__class__`等,通过`dir()`函数可以查看类的所有内建属性。内建函数如`range`、`map`、`filter`、`reduce`和`sorted`等,分别用于生成序列、映射操作、过滤操作、累积计算和排序。其中,`reduce`在Python 3中需从`functools`模块导入。示例代码展示了这些特性和函数的具体用法及注意事项。
|
27天前
|
Go Python
Python中的round函数详解及使用示例
`round()`函数是Python内置的用于四舍五入数字的工具。它接受一个数字(必需)和可选的小数位数参数,返回最接近的整数或指定精度的浮点数。本文详细介绍其用法、参数及示例,涵盖基本操作、负数处理、特殊情况及应用建议,帮助你更好地理解和运用该函数。
|
2月前
|
Python
Python调用函数并获取返回值
通过本文的介绍,我们详细了解了如何在Python中定义和调用函数,传递参数,以及获取函数的返回值。掌握这些基本操作是编写高效、清晰和可维护Python代码的基础。希望这些内容能够帮助你在实际编程中更好地使用函数。
60 18
|
1月前
|
人工智能 数据库连接 开发工具
[oeasy]python069_当前作用域都有些什么_列表dir_函数_builtins
本文介绍了Python中`dir()`函数的使用方法及其作用。`dir()`可以列出当前作用域内的所有变量和成员,类似于`locals()`,但`dir()`不仅限于本地变量,还能显示模块中的所有成员。通过`dir(__builtins__)`可以查看内建模块中的所有内建函数,如`print`、`ord`、`chr`等。此外,还回顾了`try-except-finally`结构在数据库连接中的应用,并解释了为何`print`函数可以直接使用而无需导入,因为它位于`__builtins__`模块中。最后,简要提及了删除`__builtins__.print`的方法及其影响。
35 0
|
3月前
|
分布式计算 MaxCompute 对象存储
|
2月前
|
Python
[oeasy]python057_如何删除print函数_dunder_builtins_系统内建模块
本文介绍了如何删除Python中的`print`函数,并探讨了系统内建模块`__builtins__`的作用。主要内容包括: 1. **回忆上次内容**:上次提到使用下划线避免命名冲突。 2. **双下划线变量**:解释了双下划线(如`__name__`、`__doc__`、`__builtins__`)是系统定义的标识符,具有特殊含义。
44 3
|
2月前
|
JSON 监控 安全
深入理解 Python 的 eval() 函数与空全局字典 {}
`eval()` 函数在 Python 中能将字符串解析为代码并执行,但伴随安全风险,尤其在处理不受信任的输入时。传递空全局字典 {} 可限制其访问内置对象,但仍存隐患。建议通过限制函数和变量、使用沙箱环境、避免复杂表达式、验证输入等提高安全性。更推荐使用 `ast.literal_eval()`、自定义解析器或 JSON 解析等替代方案,以确保代码安全性和可靠性。
73 2
|
3月前
|
数据采集 JSON API
如何利用Python爬虫淘宝商品详情高级版(item_get_pro)API接口及返回值解析说明
本文介绍了如何利用Python爬虫技术调用淘宝商品详情高级版API接口(item_get_pro),获取商品的详细信息,包括标题、价格、销量等。文章涵盖了环境准备、API权限申请、请求构建和返回值解析等内容,强调了数据获取的合规性和安全性。
|
2月前
|
存储 人工智能 Python
[oeasy]python061_如何接收输入_input函数_字符串_str_容器_ 输入输出
本文介绍了Python中如何使用`input()`函数接收用户输入。`input()`函数可以从标准输入流获取字符串,并将其赋值给变量。通过键盘输入的值可以实时赋予变量,实现动态输入。为了更好地理解其用法,文中通过实例演示了如何接收用户输入并存储在变量中,还介绍了`input()`函数的参数`prompt`,用于提供输入提示信息。最后总结了`input()`函数的核心功能及其应用场景。更多内容可参考蓝桥、GitHub和Gitee上的相关教程。
36 0