• 函数对象


    函数对象:

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

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

    1 可以直接被引用

    func = fn
    print(func)
    fn()
    func()

    2 可以当作函数参数传递

    def add(a, b):
        return a + b
    def low(a, b):
        return a - b
    def jump(a, b):
        return a * b
    def 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 + b
    def low(a, b):
        return a - b
    def jump(a, b):
        return a * b
    def full(a, b):
        return a / b
    def quyu(a, b):
        return a % b
    def computed(fn, n1, n2):
        res = fn(n1, n2)
        return res
    method_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 + b
    def low(a, b):
        return a - b
    def jump(a, b):
        return a * b
    def full(a, b):
        return a / b
    def quyu(a, b):
        return a % b
    def computed(fn, n1, n2):
        res = fn(n1, n2)
        return res
    method_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关键词

    part1
    print(len)
    
    len = 10
    
    def fn():
        len = 20
        print(len)
    
    fn()
    print(len)
     part2
    ls = []
    def fn1():
        ls = []
        ls.append(100)
        print(ls)
    fn1()
    print(ls)
    
    # 定义一个函数,函数中有一个变量
    
    def fn2():
        num = 20
        return num
    
    # 再定义一个函数,该函数要使用上一个函数中的变量
    def fn3():
        print(num)
    num = fn2()
    fn3()
    
     part3
    num = 10
    def 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 = 10
    def outer():
        len = 20  # 外层函数的局部变量:Enclosing - 嵌套作用域
        def inner():
            len = 30
            print('1:', len)  # inner -> outer -> global -> built-in
        inner()
        print('2:', len)  # outer -> global -> built-in
    outer()
    print('3:', len)  # global -> built-in
    
    del len
    print('4:', len)  # built-in
    
    LEGB
    不同作用域之间名字不冲突,以达到名字的重用
    查找顺序: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 inner
    fn = outer()  # fn = inner
    fn()

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

    import time
    def 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
  • 相关阅读:
    第八章 路由器交换机及其操作系统的介绍
    k-Tree DP计数
    Drop Voicing 最长升序
    高精度
    1196D2
    C
    POJ 3974 马拉车
    2020.8.1第二十六天
    2020.7.31第二十五天
    每日日报
  • 原文地址:https://www.cnblogs.com/wangtenghui/p/10638322.html
Copyright © 2020-2023  润新知