day16:一文弄懂函数对象、函数嵌套和函数闭包的关系和应用

news2025/1/18 18:43:11

1.函数的对象

精髓可以把函数当成变量去用

1.1 可以赋值

# func=内存地址
def func():
    print('from func')
    
f=func
print(f,func)
f()

输出:
<function func at 0x0000017A4E24F7F0> <function func at 0x0000017A4E24F7F0>
from func

1.2 可以把函数当做参数传给另外一个函数

'''示例1'''
def func():
    print('from func')
    
def foo(x): # x = func的内存地址
    print(x) 
    # x()

foo(func) # foo(func的内存地址)

输出:<function func at 0x0000028FE716F7F0>

'''示例2'''
def func():
    print('from func')
    
def foo(x):# x=func的内存地址
    # print(x)
    x()  # 调用func函数

foo(func) # foo(func的内存地址)

输出:from func

1.3 可以把函数当做另外一个函数的返回值

def foo(x): # x=func的内存地址
    return x # return func的内存地址

res=foo(func) # foo(func的内存地址)
print(res) # res=func的内存地址

res()

输出:
<function func at 0x0000011745EDF7F0>
from func

1.4 可以当做容器类型的一个元素

如:列表、字典、元组

def func():
    print('from func')

''''''
l=[func,]
print(l)   # 列表l的内存地址  输出:[<function func at 0x7f821a3dd1e0>]
l[0]()    # l[0]的结果为名字func的内存地址,指向函数func,l[0]加()结果为func()调用func函数

dic={'k1':func}
print(dic)   # 字典dic的输出:{'k1': <function func at 0x7fcd4bbaa1e0>}
dic['k1']()    # dic['k1']获取到func的内存地址,加()表示调用func函数

tup = (11, func)
print(tup[1])
tup[1]()

输出:
[<function func at 0x7efdfd9df1e0>]
from func
{‘k1’: <function func at 0x7efdfd9df1e0>}
from func
<function func at 0x7fe9718651e0>
from func

1.5 经典案例:函数对象应用示范

案例要求:写ATM机的基本业务功能,包括登录、转账、查询余额、提现、退出等功能

1.5.1 初级写法

'''###案例1:初级写法'''
def login():
    print('登录功能')

def transfer():
    print('转账功能')

def check_balance():
    print('查询余额')

def withdraw():
    print('提现功能')

while True:
    print('''
    0 退出
    1 登录
    2 转账
    3 查询余额
    4 提现
    ''')
    choice = input('请输入需要使用的功能编号:').strip()
    
    if not choice.isdigit():
        print('请输入范围内的功能编号!!!')
        continue
        
    if choice == '0':
        print('已退出')
        break
    elif choice == '1':
        login()
    elif choice == '2':
        transfer()
    elif choice == '3':
        check_balance()
    elif choice == '4':
        withdraw()
    else:
        print('输入的指令不存在,请重新输入')

点评:上述代码的可扩展性极差,如果需求变更,要求新增一个“注册”功能,我们需要修改三处,一是定义一个新的函数,功能为注册;二是while函数体类的用户信息需要新增;三是源代码中需要添加elif choice=5,再调用一下“注册”功能的函数;如果要增加10个功能,维护起来及其麻烦。我们来优化一下~~

1.5.2 中级写法

简化方向:见标题“1.4”,可以把函数当作容器类型的一个元素,将函数整合到字典里

'''###案例2:中级写法'''
def login():
    print('登录功能')
    
def transfer():
    print('转账功能')

def check_balance():
    print('查询余额')

def withdraw():
    print('提现功能')

func_dic = {
    '1':login,
    '2':transfer,
    '3':check_balance,
    '4':withdraw
}
# func_dic['1']() 
   # func_dic['1']值为login的内存地址,加()表示调用login()函数功能

while True:
    print('''
    0 退出
    1 登录
    2 转账
    3 查询余额
    4 提现
    ''')
    choice = input('请输入需要使用的功能编号:').strip()

    if not choice.isdigit():
        print('请输入范围内的功能编号!!!')
        continue

    if choice == '0':
        print('已退出')
        break

    if choice in func_dic:
        func_dic[choice]()
    else:
        print('输入的指令不存在,请重新输入')

点评:当需要新增功能时,上述代码也需要修改函数体,当函数体代码量过大时容易引入bug,思考下如何保证函数体内的函数不动,只修改函数体外的代码。

1.5.2 高级写法

简化方向:提高函数体的扩展性,将功能说明信息和功能整合到字典,把功能信息和函数功能当作容器的一个元素

def login():
    print('登录功能')


def transfer():
    print('转账功能')


def check_banlance():
    print('查询余额')


def withdraw():
    print('提现')


def register():
    print('注册')


func_dic={
    '0':['退出'],
    '1': ['登录功能', login],
    '2': ['转账功能', transfer],
    '3': ['查询余额', check_banlance],
    '4': ['提现', withdraw],
    '5': ['注册', register],
}

while True:
    for k in func_dic:
        print(k, func_dic[k][0])

    choice = input('请输入命令编号:').strip()
    if not choice.isdigit():
        print('请输入范围内的功能编号,编号,编号!!!')
        continue

    if choice == '0':
        break

    if choice in func_dic:
        func_dic[choice][1]()
    else:
        print('输入的指令不存在')

2. 函数嵌套

2.1 函数的嵌套调用:

说明:在调用一个函数的过程中又调用其他函数

案例:比较一串数字的大小
思路分析:
1.第一个和第二个数字先比较大小,留下大的那一个;
2.第一步中最大的数与第三个数比较,留下大的那一个;
3.第二步中最大的数与第四个数比较,留下大的那一个;


所以 将两两比较大小的功能封装成函数

'''案例:筛选出一串数字中最大的那个'''
def max2(x,y):
    if x > y:
        return x
    else:
        return y

def max4(a,b,c,d):
    # 第一步:比较a,b得到res1
    res1=max2(a,b)
    # 第二步:比较res1,c得到res2
    res2=max2(res1,c)
    # 第三步:比较res2,d得到res3
    res3=max2(res2,d)
    return res3

res=max4(1,2,3,4)
print(res)

2.2 函数的嵌套定义

说明:在函数内定义其他函数

def f1():
    def f2():
        pass
''' 求圆形的周长:2*pi*radius
	求圆形的面积:pi*(radius**2)
'''
def circle(radius,action=0):
    from math import pi

    def perimiter(radius):
        return 2*pi*radius

    # 求圆形的求面积:pi*(radius**2)
    def area(radius):
        return pi*(radius**2)

    if action == 0: 
        return 2*pi*radius   # 返回函数式
 
    elif action == 1:
        return area(radius) # 返回函数
# 调用方法1
result = circle(2,action=0)
print(result)

# 调用方法2
result = circle(2,action=0)
print(result)

3. 闭包函数

3.1 前期提要

闭包函数 = 名称空间与作用域+函数嵌套+函数对象
核心点:名字的查找关系是以函数定义阶段为准

3.2 什么是闭包函数

“闭”函数:指的该函数是内嵌函数
“包”函数:指的该函数包含对外层函数作用域名字的引用(不是对全局作用域)

3.1.1 案例1:闭包函数之名称空间与作用域的应用

''' 案例1 闭包函数:名称空间与作用域的应用'''
def f1():
    x = 33333333333333333333
    def f2():
        print(x)
    f2()

f1()  # 输出:33333333333333333333

补充说明:也就是说函数被当做数据处理时,始终以自带的作用域为准。若内嵌函数包含对外部函数作用域(而非全局作用域)中变量的引用,那么该’内嵌函数’就是闭包函数,简称闭包(Closures)

3.1.2 案例2:闭包函数之名称空间与作用域的应用+函数嵌套

''' 案例2 闭包函数:名称空间与作用域的应用+函数嵌套'''
def f1():
    x = 33333333333333333333
    def f2():
        print(x)
    f2()

x=11111
def bar():
    x=444444
    f1()

def foo():
    x=2222
    bar()

foo()  # 输出:33333333333333333333

在这里插入图片描述

3.3 闭包函数:函数对象

def f1():
    x = 33333333333333333333
    def f2():
        print('函数f2:',x)
    return f2

f=f1()     # 返回函数f2的内存地址
print(f)   # 输出:<function f1.<locals>.f2 at 0x7ff3a9819d08>

x=4444
f()     # 输出:函数f2: 33333333333333333333
        # 解释:1、f存的是f1(),f1()返回值为函数f2的内存地址,故f()相当于调用了函数f2
        #      2、f2是f1的内嵌函数,即函数f2为“闭”函数,f2包含对外层函数作用域名字的引用
def foo():
    x=5555
    f()

foo()  # 相当于调用f()==>f2(),故输出为:函数f2: 33333333333333333333

3.4 闭包函数的应用

为何要有闭包函数,两种为函数体传参的方式

3.4.1 方式一:直接把函数体需要的参数定义成形参

'''案例1 改革开放前:直接把函数体需要的参数定义成形参'''
def f2(x):
    print(x)

f2(1)
f2(2)
f2(3)

3.4.2 方式二:将需要传入的值包给函数

''' 案例2 改革开放后:'''
def f1(x):
    def f2():
        print(x)
    return f2

x=f1(3)
print(x) # 返回的是函数f2的内存地址 

x()  # 调用f2,即打印结果为 3

3.4.3 更加直观的案例

'''传参的方案一:把函数需要的值定义为形参'''

import requests

def get(url):
    response=requests.get(url)
    print(len(response.text))

get('https://www.baidu.com')
get('https://www.cnblogs.com/linhaifeng')
get('https://zhuanlan.zhihu.com/p/109056932')
'''传参的方案二:把函数需要的值包给外层函数'''

import requests

def outter(url):
    # url='https://www.baidu.com'  #  将url传到outter,相当于在outter的作用域内定义了 url='https://www.baidu.com'这个变量
    def get():
        response=requests.get(url)
        print(len(response.text))
    return get

baidu=outter('https://www.baidu.com')
baidu()
baidu()
baidu()

<分析>:对比两种方式,方案一在下载同一页面时需要重复传入url,而方案二只需要传一次值,就会得到一个包含指定url的闭包函数,以后调用该闭包函数无需再传url.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2034204.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

PythonStudio 控件使用常用方式(十四)TCheckListBox

PythonStudio是一个极强的开发Python的IDE工具&#xff0c;它使用的是Delphi的控件&#xff0c;常用的内容是与Delphi一致的。但是相关文档并一定完整。现在我试试能否逐步把它的控件常用用法写一点点&#xff0c;也作为PythonStudio的参考。 从1.2.1版开始&#xff0c;Python…

视频融合项目中的平台抉择:6大关键要素助力精准选型

随着安防监控系统行业的快速发展&#xff0c;视频融合项目逐渐成为城市治理、企业管理及智能建筑等领域的重要组成部分。视频融合平台作为视频数据整合、管理和分析的核心&#xff0c;其选择直接影响到项目的成功与否。 在当前智慧业务类项目的集成过程中&#xff0c;我们不仅…

漂亮网站是门面,和你的豪车、奢侈品、秘书一样,该投就得投。

对于企业来说&#xff0c;一个漂亮的网站可以起到很好的门面作用。一个好的网站可以让客户更容易地找到你的产品和服务&#xff0c;提高品牌形象和知名度&#xff0c;增加业务成功几率。因此&#xff0c;对于企业来说&#xff0c;投资于一个漂亮的网站是非常必要的。 经常看到…

用Python实现9大回归算法详解——01线形回归算法

1. 线性回归的基本概念 线性回归是一种最基本的监督学习算法&#xff0c;用于预测因变量&#xff08;目标变量&#xff09;和一个或多个自变量&#xff08;特征变量&#xff09;之间的关系。线性回归假设因变量与自变量之间的关系是线性的&#xff0c;即可以用以下形式的线性方…

2024国赛Word论文模板【一键生成式操作】

一、比赛介绍 该竞赛创办于1992年&#xff0c;每年一届&#xff0c;是首批列入“高校学科竞赛排行榜”的19项竞赛之一。2023年&#xff0c;来自全国及美国、澳大利亚、马来西亚的1685所院校/校区、59611队(本科54158队、专科5453队)、近18万人报名参赛。 而今年的国赛马上就要…

《Unity3D网络游戏实战》正确收发数据流

TCP数据流 系统缓冲区 当收到对端数据时&#xff0c;操作系统会将数据存入到Socket的接收缓冲区中 操作系统层面上的缓冲区完全由操作系统操作&#xff0c;程序并不能直接操作它们&#xff0c;只能通过socket.Receive、socket.Send等方法来间接操作。当系统的接收缓冲区为空&…

C#去掉文件夹或文件名非法字符

实现有输入字符串创建文件夹的功能&#xff0c;需要检查字符串中是否包含不能在文件夹中使用的非法字符 。C#中如果需要生成文件路径&#xff0c;有时会抛出“文件路径中出现非法字符”的异常&#xff0c;这时我们就会找到这个非法的字符串&#xff0c;然后Replace替换掉。net类…

牛客JS题(四十)字体高亮

注释很详细&#xff0c;直接上代码 涉及知识点&#xff1a; 正则表达式逆向思路 题干&#xff1a; 我的答案 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /></head><body><input type"text&…

图形编辑器基于Paper.js教程14:使用 Paper.js 绘制数学图形与交互的实现,画布缩放保持大小的圆,正弦,余弦,螺旋线

技术分析&#xff1a;使用 Paper.js 绘制数学图形与交互的实现 在现代Web开发中&#xff0c;动态图形和交互式视觉表现已成为提升用户体验的重要手段。通过一个详细的示例&#xff0c;我们将探索如何使用 Paper.js 进行数学图形&#xff08;正弦曲线、余弦曲线和螺旋线&#x…

拯救打工人的4款可ai生成ppt神器大PK,谁是加班狗的最爱?

在忙碌的工作日常里&#xff0c;做PPT常常让不少上班族头疼。晚上加班时&#xff0c;办公室里总能看到那些为了做出完美PPT而熬夜的人。不过现在好了&#xff0c;科技进步带来了好消息&#xff0c;有几款AI生成PPT的神器出现了&#xff0c;今天&#xff0c;咱们就来比一比&…

网络编程----TCP/IP协议

使用TCP/IP协议实现客户端和服务器端进行通信: 1.服务器端(test1.c): #include <sys/socket.h> #include <sys/types.h> #include <arpa/inet.h> #include <stdio.h> #include <unistd.h>// 创建服务器端 int main() {//1.创建套接字int serfd…

锂电池剩余寿命预测 | Matlab基于Transformer-GRU的锂电池剩余寿命预测

目录 预测效果基本介绍程序设计参考资料 预测效果 基本介绍 Matlab基于Transformer-GRU的锂电池剩余寿命预测&#xff0c;Transformer结合门控循环单元。 Matlab基于Transformer-GRU的锂电池剩余寿命预测&#xff08;单变量&#xff09; 运行环境Matlab2023b及以上。 首先从…

API网关:SpringCloud GateWay

一. 网关的作用及背景 1.API网关的作用 请求路由 在我们的系统中由于同一个接口新老两套系统都在使用&#xff0c;我们需要根据请求上下文将请求路由到对应的接口。 统一鉴权 对于鉴权操作不涉及到业务逻辑&#xff0c;那么可以在网关层进行处理&#xff0c;不用下层到业务…

嵌入式开发--STM32的GPIO输入和输出复用

嵌入式开发–STM32的GPIO输入和输出复用 MCU的引脚数量非常有限&#xff0c;做项目时&#xff0c;经常是为了成本而选择引脚尽量少的芯片&#xff0c;这也给布线和编程带来更大的挑战。 最近一个项目&#xff0c;需要在某些时候通过拨码开关预置参数&#xff0c;预置完成后&am…

QT自定义系统快捷键任务

关键代码 //自定义快捷键检测 connect(this->ui->hotkeySequenceEdit_1, &QKeySequenceEdit::keySequenceChanged,this, &HotTestWidget::setShortcut_1);// 托盘显示 trayIcon new QSystemTrayIcon(this); QPixmap pixmap("tray.png"); QIcon icon(…

[Android] [解决]Bottom Navigation Views Activity工程带来的fragment顶部空白间距问题

用Android Stuio创建一个Bottom Navigation Views Activity工程&#xff0c; 我们刻意设置一下fragment背景为黑色&#xff0c;会发现&#xff0c;这个fragment离顶部还有一段不小空白距离&#xff0c; 怎么解决呢&#xff1f; 在activity_main.xml里面&#xff0c;删掉这句&a…

2024年【中级消防设施操作员(考前冲刺)】找解析及中级消防设施操作员(考前冲刺)试题及解析

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 中级消防设施操作员&#xff08;考前冲刺&#xff09;找解析考前必练&#xff01;安全生产模拟考试一点通每个月更新中级消防设施操作员&#xff08;考前冲刺&#xff09;试题及解析题目及答案&#xff01;多做几遍&a…

SuccBI+低代码文档中心 — 可视化分析(仪表板)(下)

制作仪表板 引入数据模型 仪表板所需模型已经在数据模块中准备好&#xff0c;可以将对应模型表添加到数据模型中。提供了两种添加方式&#xff1a; 在数据栏中点击添加按钮&#xff0c;在弹出框中通过搜索或直接在其所在目录下选中该模型&#xff0c;点击确定。 点击数据按钮…

【安卓】调用摄像头和相册

文章目录 调用摄像头拍照从相册中选择照片 调用摄像头拍照 新建一个CameraAlbumTest项目&#xff0c;然后修改activity_main.xml中的代码。在布局文件中添加两个控件&#xff1a;一个Button和一个ImageView。Button是用于打开摄像头进行拍照的&#xff0c;而ImageView则是用于将…