博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
函数对象
阅读量:7075 次
发布时间:2019-06-28

本文共 4993 字,大约阅读时间需要 16 分钟。

函数对象:

函数名存放的就是函数的地址,所以函数名也是对象,称之为函数对象

a = 10print(a)def fn():    num = 10    print('fn function run')print(fn)

1 可以直接被引用

func = fnprint(func)fn()func()

2 可以当作函数参数传递

def add(a, b):    return a + bdef low(a, b):    return a - bdef jump(a, b):    return a * bdef full(a, b):    return a / b# 计算: 通过该函数可以完成对任意两个数的四则运算某一运算def computed(fn, n1, n2):    # fn代表四则运算中的一种    # res为运算结果    res = fn(n1, n2)    return res# 完成两个数的某一运算,拿到结果while True:    cmd = input('cmd:')    if cmd == 'add':        result = computed(add, 100, 20)    elif cmd == 'low':        result = computed(low, 100, 20)    else:        print('输入有误')        break    print(result)

3 可以作为容器类型的元素

def add(a, b):    return a + bdef low(a, b):    return a - bdef jump(a, b):    return a * bdef full(a, b):    return a / bdef quyu(a, b):    return a % bdef computed(fn, n1, n2):    res = fn(n1, n2)    return resmethod_map = {    'add': add,    'low': low,    'jump': jump,    'full': full,    'quyu': quyu,}while True:    cmd = input('cmd: ')    # 用户输入的指令只要有对应关系,就会自动去走对应的计算方法    # 这样外界就不能去关系到底有哪些计算方法    if cmd in method_map:        cp_fn = method_map[cmd]  # 拿到计算方法        result = computed(cp_fn, 100, 20)  # 通过计算方法得到计算结果        print(result)    else:        print('输入有误,退出')        break

4 可以作为函数的返回值

def add(a, b):    return a + bdef low(a, b):    return a - bdef jump(a, b):    return a * bdef full(a, b):    return a / bdef quyu(a, b):    return a % bdef computed(fn, n1, n2):    res = fn(n1, n2)    return resmethod_map = {    'add': add,    'low': low,    'jump': jump,    'full': full,    'quyu': quyu,}# 根据指令获取计算方法def get_cp_fn(cmd):    if cmd in method_map:        return method_map[cmd]    return add  # 输入有误用默认方法处理

名称空间

名称空间:存放名字与内存空间地址对应关系的容器
作用:解决由于名字有限,导致名字重复发送冲突的问题

名称空间的加载顺序

python test.py#1、python解释器先启动,因而首先加载的是:内置名称空间#2、执行test.py文件,然后以文件为基础,加载全局名称空间#3、在执行文件的过程中如果调用函数,则临时产生局部名称空间

三种名称空间
Built-in:内置名称空间;系统级,一个;随解释器执行而产生,解释器停止而销毁
Global:全局名称空间;文件级,多个;随所属文件加载而产生,文件运行完毕而销毁
Local:局部名称空间;函数级,多个;随所属函数执行而产生,函数执行完毕而销毁
注:del 名字:可以移除查找最近的名字与内存空间地址的对应关系
加载顺序:Built-in > Global > Local

名字的查找程序:局部名称空间--->全局名称空间--->内置名称空间 

需要注意的是:在全局无法查看局部的,在局部可以查看全局的

while True:    cmd = input('cmd: ')    if cmd == 'quit':        break    cp_fn = get_cp_fn(cmd)    result = computed(cp_fn, 100, 20)    print(result)    print(len('abc'))    len = len('abcdef')    print(len)    del len    # del len    print(len('000111222'))    '''    [        # ['len': 100001]  # 保存长度的变量        ['len': 200001]  # 计算长度的函数    ]    '''    def fn1():        len = 10        print(len)    def fn2():        len = 20        print(len)    fn1()    fn2()

global关键词

part1print(len)len = 10def fn():    len = 20    print(len)fn()print(len) part2ls = []def fn1():    ls = []    ls.append(100)    print(ls)fn1()print(ls)# 定义一个函数,函数中有一个变量def fn2():    num = 20    return num# 再定义一个函数,该函数要使用上一个函数中的变量def fn3():    print(num)num = fn2()fn3() part3num = 10def fn2():    global num  # 将Local:num => Global: num    num = 20    num = 30# Local的名字一旦global,就办成Global的名字,一个文件中的Global名字就是一个def fn3():    global num      num = 40    print(num)fn2()print(num)fn3()print(num)
函数的嵌套
将函数直接定义到另一个函数内部,就可以使用外部函数的中的名字def outer():    num = 20    def inner():        print(num)  # inner就可以直接使用outer中的名字    inner()outer()

作用域:名字起作用的范围

作用:解决同名字可以共存问题
四种作用域
Built-in:内置作用域,所有文件所有函数
Global:全局作用域,当前文件所有函数
Enclosing:嵌套作用域,当前函数与当前函数的内部函数
Local:局部作用域,当前函数

1、作用域即范围 

全局范围(内置名称空间与全局名称空间属于该范围):全局存活,全局有效  

局部范围(局部名称空间属于该范围):临时存活,局部有效

2、作用域关系是在函数定义阶段就已经固定的,与函数的调用位置无关

len = 10def outer():    len = 20  # 外层函数的局部变量:Enclosing - 嵌套作用域    def inner():        len = 30        print('1:', len)  # inner -> outer -> global -> built-in    inner()    print('2:', len)  # outer -> global -> built-inouter()print('3:', len)  # global -> built-indel lenprint('4:', len)  # built-inLEGB不同作用域之间名字不冲突,以达到名字的重用查找顺序:Local -> Enclosing -> Global -> Built-in -> 抛异常闭包就是函数嵌套(格式稍作改良)

closure:被包裹的函数,称之为闭包

完整的闭包结构:1.将函数进行闭包处理;2.提升函数名的作用域,将内部函数对象作为外部函数的返回值:1.可以使用局部变量; 2.不改变函数的调用位置

inner可以使用outer的局部变量:可以将inner定义在outer中inner的调用还是在外部:inner函数对象能被outer返回def outer():    num = 10    def inner():  # 闭包:定义在函数内部的函数称之为闭包        print(num)    return innerfn = outer()  # fn = innerfn()

案例一:外部函数可以为闭包传递参数 (了解)

import timedef download():    print('开始下载')    time.sleep(2)    print('下载完成')    data = "下载得到的数据"    outer(data)# 为闭包传参def outer(data):    def inner():        # 保存,播放,删除等操作        print("闭包打印:", data)    inner()download()
案例二:延迟执行
import requests# get_html一执行就获取页面def get_html(url):    html = requests.get(url)    print(html.text)get_html('https://www.baidu.com')get_html('https://www.python.org')get_html('https://www.sina.com.cn')get_html('https://www.baidu.com')def outer(url):    def get_html():        html = requests.get(url)        print(html.text)    return get_html# 先预定义多个爬虫方法,爬页面操作并未执行baidu = outer('https://www.baidu.com')python = outer('https://www.python.org')sina = outer('https://www.sina.com.cn')# 什么时候想爬什么页面就调用指定页面的爬虫方法baidu()sina()baidu()
View Code

 

转载于:https://www.cnblogs.com/wangtenghui/p/10638322.html

你可能感兴趣的文章