【Python】链式、嵌套调用、递归、函数栈帧、参数默认值和关键字参数

devtools/2024/9/25 2:33:43/

链式调用

用一个函数的返回值,作为另一个函数的参数

python">def isOdd(num):  if num % 2 == 0:  return False  return True  def add(x, y):  return x + y  print(isOdd(add(3,4)))"""
运行结果"""
  • 这里就是先算出 add 的值,然后将 add 的结果传给 isOdd 进行奇偶数判断,最后将结果给 print 进行打印

链式调用中,是先执行 ( ) 里面的函数,然后执行外面的函数。换句话说就是:调用一个函数,就需要先对他的参数求值

嵌套调用

一个函数体内部,还可以调用其他函数

python">def a():  num1 = 10  print('函数 a')  def b():  num2 = 20  print('函数 b')  a()  def c():  num3 = 30  print('函数 c')  b()  c()"""
运行结果
函数 c
函数 b
函数 a
"""
  • 调用 c 的时候,会打印 c,也会调用 b
  • 调用 b 的时候,会打印 b,也会调用 a
  • 调用 a 的时候,会打印 a

函数栈帧

  • 调试器的左下角,能看到函数之间的“调用栈
  • 调用栈里面描述了当前这个代码的函数之间的调用关系是怎样的
  • 每一层这个调用关系就称为“函数的栈帧”,每个函数的局部变量就在这个栈帧中体现
    image.png|435
  • 每一层栈帧,在你选中之后,都能看到里面的局部变量,每个函数的局部变量就保存在对应的栈帧之中
  • 调用函数,则生成对应的栈帧;函数结束,则对应的栈帧消亡(里面的局部变量也就没了)

如果将每个函数里面的变量名都改为一样的,但里面的变量仍是不同的变量,属于不同的函数作用域

  • 每个变量是保存在各自的栈帧中的,每个栈帧是保存在内存上的
  • 变量的本质是一块内存空间

函数递归

函数递归,就是一个函数自己调用自己

python">def factor(n):  if n == 1:  return 1  return n * factor(n-1)  result = factor(5)  
print(result)"""
运行结果
120
"""

image.png

  • 虽然都是 n,但是 n 是函数的形参,形参相当于函数的局部变量,而局部变量是在函数自己的栈帧上的
  • 虽然几个函数的局部变量名相同,但是他们是在不同的栈帧中,是在不同的内存空间中,也就是不同的变量
  • 另一方面,看起来是同一个函数,但是这里的多次调用,其实是多个不同的栈帧

递归的两个要素:

  1. 递归结束条件
  2. 递归的递推公式

缺点

递归的缺点:

  1. 执行过程非常复杂,难以理解
  2. 递归代码容易出现“栈溢出”的情况
    • 代码不小心写错了,导致每次递归的时候,参数不能正确的接近递归结束的条件,就会出现“无限递归”的情况
  3. 递归代码一般都是可以转换成等价的循环代码的,循环的版本通常运行速度要比递归的版本有优势(函数的调用也是有开销的)

优点

代码非常简洁,尤其是处理一些“问题本身就是通过递归方式定义的”问题非常方便(二叉树)

参数默认值

  • Python 中的函数,可以给形参指定默认值
  • 带有默认值的参数,可以在调用的时候不传参

在函数内部加上打印信息,方便我们进行调试。但是,像这种调试信息,希望在正式发布的时候不要有,只是在调试的阶段才有
为了实现这一点,我们就可以给函数加上特定的“开关

python">def add(x, y, debug=False):  if debug:  print(f'x = {x}, y = {y}')  return x + y  result = add(10,20, True)  
print(result)
  • 这里的 debug = False 就是形参的默认值
  • 带有默认值的形参,就可以在调用函数的时候,不必传参,使用默认值。但也可以手动传参,不使用默认值

通过这样的默认值,就可以让函数的设计更灵活
但要求带有默认值的形参,得在形参列表的后面,而不能在前面或者中间,带有多个默认参数的形参,就都得放在后面

但像默认值这样的语法,在编程界是存在争议的

  • C++也支持形参默认参数
  • Java 不支持

关键字参数

在调用函数的时候,需要给函数指定实参,一般默认情况下是按照形参的顺序,来依次传递实参的

  • 按照先后顺序来传参,这种传参风格,称为“位置参数”,这是各个编程语言中最普遍的方式

关键字传参,是按照形参的名字来进行传参

python">def test(x, y):  print(f'x = {x}')  print(f'y = {y}')  test(x=10, y=20)
test(y=100, x=200)
test(1000, y=2000)"""
运行结果
x = 10
y = 20
x = 200
y = 100
x = 1000
y = 2000
"""
  • 这样的设定,能非常明显地告诉程序猿,你的参数要传给谁
  • 并且有了关键字参数之后,传参的顺序也可以随意,可以无视形参和实参的顺序
  • 位置参数和关键字参数还可以混着用,只不过混着用的时候要求位置参数在前,关键字参数在后

关键字参数一般是搭配默认参数来使用。一个函数,可以提供很多的参数,来实现对这个函数的内部功能做出一些调整设定,为了降低调用者的使用成本,就可以把大部分参数设定出默认值,当调用这需要调整其中的一部分参数的时候,就可以搭配关键字参数来进行操作


http://www.ppmy.cn/devtools/108344.html

相关文章

PostgreSQL的基础知识

什么是数据库? 数据库(Database)是一个用于存储、检索、管理和分析数据的集合。它是按照一定的数据模型组织、存储的集合,具有统一的结构形式、定义的相互关系、制定的约束条件和一定的冗余度,以便于在各种用户、各种…

uni-app填坑指南——解决处理处理静态资源的问题

说实在话,这个标题其实有点夸大了。 uni-app并没有不解决,而是解决的不够充分不够彻底。这里我们来讨论一下uni-app在处理静态资源的问题上的一些不足之处。 1. 问题描述 在uni-app中,我们可以将静态资源放在static目录下,然后通…

【Oracle篇】全面理解优化器和SQL语句的解析步骤(含执行计划的详细分析和四种查看方式)(第二篇,总共七篇)

💫《博主介绍》:✨又是一天没白过,我是奈斯,DBA一名✨ 💫《擅长领域》:✌️擅长Oracle、MySQL、SQLserver、阿里云AnalyticDB for MySQL(分布式数据仓库)、Linux,也在扩展大数据方向的知识面✌️…

数据赋能(198)——开发:数据应用——技术方法、主要工具

技术方法 数据应用的技术方法主要包括以下几种: 数据采集技术: 网络爬虫技术:通过模拟人的行为在网上抓取数据,能够获取互联网上的大量数据,并按照一定的规则和筛选条件进行筛选和提取。传感器技术:通过各…

python深度学习:从注意力机制到生成模型,全面解析现代AI技术

近年来,伴随着以卷积神经网络(CNN)为代表的深度学习的快速发展,人工智能迈入了第三次发展浪潮,AI技术在各个领域中的应用越来越广泛。注意力机制、Transformer模型(BERT、GPT-1/2/3/3.5/4、DETR、ViT、Swin…

TL431精密可调并联稳压器

TL431是一种由德州仪器(TI)公司生产的可调精密并联稳压器,具有较广的工作电压范围(2.5V-36V)和优异的电流调节能力。作为一种基准电压源,它广泛用于电源设计、稳压电路以及精密参考电源等应用场合。 TL431的内部结构 TL431具有三…

C/C++经典排序问题,sort函数使用

目录 1. 前言 2. 正文 2.1 问题 2.2 解决办法 2.2.1 思路 2.2.2 代码实现 2.2.3 测试结果 3. 备注 1. 前言 大家在学习C语言的时候,是不是经常被排序算法折磨的很难那首,其实都是但是在C中有专门的,排序函数,而且支持自定…

leetcode 1651 Hopper公司查询3(postgresql)

需求 Table: Drivers -------------------- | Column Name | Type | -------------------- | driver_id | int | | join_date | date | -------------------- driver_id是该表的主键。 该表的每一行均包含驾驶员的ID以及他们加入Hopper公司的日期。 Table: Rides ---------…