zoukankan      html  css  js  c++  java
  • 04-课堂笔记-函数相关

    一、复习

    '''
    字符编码
    文件头:py2 - ASCII py3 - UTF-8
    三种字符串:u'' b'' r''
    u''.encode('utf-8')
    b''.decode('utf-8')
    str(b'', encode='utf-8')
    bytes(u'', encode='utf-8')

    文件操作
    三步骤:1.开的文件 2.操作文件(读写) 3.关闭
    简写:with open => 可以自动关闭文件流
    操作模式:
    主:r w a x
    从:t b +

    seek:游标 (偏移量, 游标操作位置) 与b模式结合


    函数基础
    函数定义:def 函数名(): 函数体
    函数调用:函数名()
    函数的返回值:省略 空return return一个值 return多个值 => 本质返回一个装有多个值的元组
    '''

     

    二、今日内容

    '''
    1.函数的参数:实参与形参
    2.函数对象:函数名
    3.名称空间与作用域:LEGB
    4.闭包:函数的嵌套定义
    5.装饰器:闭包的一种应用场景
    6.迭代器对象:迭代器,生成器,for迭代

    7.匿名函数
    8.内置函数
    9.三元表达式、列表、字典生成式
    10.模块

    '''

     

    三、如何定义函数

    '''
    1、功能的单一化
    2、函数内部需要外部的资源:通过函数的参数来实现
    3、函数执行后的结果需要告诉外界:通过返回值告诉给外界
    '''

     

    四、函数的参数

    形参与实参

    '''
    形参:在函数定义时()里出现的参数
    -- 形参本身没有实际值(意义),在函数调用时,传入什么实参,形参就装有什么值
    实参:在函数调用时()里出现的参数
    -- 实参有实际值(意义)

    重点:函数调用传参:将实参的值赋值给形参 | 形参要获取外界的值只能通过实参进行获取
    '''

    def fn(a, b):  # a, b:形参
       pass

    a = 10
    fn(a, 20)  # a, 20:实参

    两种实参

    '''
    位置实参:
    1.传参两种方式:实参名 | 实参具体值
    2.必须按位置对形参进行传值


    关键字实参:
    1.传参两种方式:形参名=实参名 | 形参名=实参值
    2.可以指名道姓对形参进行传值,所以可以不用按位置进行传参
    '''

    def func(a, b, c):
       print(a, b, c)

    # func(10, b=20, 200) 报错:SyntaxError: positional argument follows keyword argument
    # 重点:两种实参在一起进行传参时:必须位置在前,关键字在后

     

    两大形参分类

    '''
    形参种类:
    1)位置形参
      -- 普通位置形参
      -- 默认值形参
      -- 可变长位置形参

    2)关键字形参
      -- 有默认值关键字形参
      -- 无默认值关键字形参
      -- 可变长关键字形参
    '''
    def fn(a, b, *, x, y):  # 位置形参:a, b 关键字形参: x, y
       pass

    '''
    重点:
    1.*为分水岭
    2.位置实参只能给位置形参进行传值
    3.关键字实参可以给位置形参与关键字形参进行传值
    '''

     

    两个带默认值的形参

    def fn2(a=10, *, x=20):
       print(a, x)
    fn2(100, x=200)
    # 总结:
    # 1.有默认值的参数可以不用传值
    # 2.*前有默认值的叫默认值参数,属于位置形参,可以被位置及关键字实参进行传值
    # 3.*后有默认值的叫有默认值的关键字形参,属于关键字形参,只能被关键字实参进行传值
    # 4.如果省略*, 有默认值的形参都是默认值参数

     

    不带默认值与带默认值形参结合使用

    def fn3(a, b=10, *, x, y=20, z):
       print(a, b, x, y, z)
    fn3(100, x=200, z=300)
    # 总结:
    # 1.没有默认值的必须传参,有默认值的可以传也可以不传
    # 2.位置有值的必须出现在无值之后,关键字顺序不做要求

     

    可变长位置形参与可变长关键字形参

    def fn4(a, b=10, *args, x, **kwargs):
        print(a, b, x)
        print(args)
        print(kwargs)
    fn4(10, 20, 30, x=100, y=200, z=300)
    # 总结:
    # 1.可变长是用来接收未接收完的值(接收0到n个):
    #   -- *args用来接收所有没有接收完的位置(只能接收位置实参)
    #   -- **kwargs用来接收所有没有接收完的关键字(只能接收关键字实参)
    # 2.*args必须出现在所以位置参数之后,**kwargs必须出现在所以参数之后
    
    
    # 常见应用场景
    # 假设第一个位置永远是参数name
    def func4(*args, **kwargs):
        name = args[0]  # 将name抽出来
    
    def func44(name, *args, **kwargs):
        # name 可以直接接收,省了抽出来的过程
        pass
    

     

    总结

    '''
    1.位置实参只能给位置形参传值
    2.关键字实参可以给位置及关键字形参传值
    3.有默认值的可以不用传参
    4.可变长位置形参只能接受位置实参,接受位置形参没有接收完的位置实参,存放到元组中
    5.可变长关键字形参只能接受关键字实参,接受关键字形参没有接收完的关键字实参,存放到字典中
    6.*args必须出现在所有位置形参之后,**kwargs必须在所有形参之后
    '''
    

     

    可变长整体传参:打散传值

    def fn(*args, **kwargs):
        print(args, kwargs)
        
    fn([1, 2, 3], {'a':1 , 'b': 2})  # =>接收到的 ([1, 2, 3], {'a':1 , 'b': 2})  {}
    fn(*[1, 2, 3], **{'a':1 , 'b': 2})  # =>接收到的 (1, 2, 3)  {'a':1 , 'b': 2}
    
    # 注:字符串也可以作为单列集合进行打散传递
    fn(*'abc')  # => ('a', 'b', 'c')  {}
    

     

    五、函数对象

    # 函数名就是存放了函数的内存地址,存放了内存地址的变量都是对象,即 函数名 就是 函数对象
    
    # 函数对象的应用场景
    # 1 可以直接被引用
    # 2 可以当作函数参数传递
    # 3 可以作为函数的返回值
    # 4 可以作为容器类型的元素
    
    # 功能体
    def add(n1, n2):
        return n1 + n2
    
    def low(n1, n2):
        return n1 - n2
    
    def jump(n1, n2):
        return n1 * n2
    
    # 完成功能
    def computed(n1, n2, fn):  # fn = add|low|jump
        res = fn(n1, n2)  # 调用具体的功能
        return res
    
    # 功能对应关系
    method_map = {  # 指令与函数对象的对应关系
        '1': add,
        '2': low,
        '3': jump
    }
    
    # 获取功能
    def get_method(cmd):
        if cmd in method_map:
            return method_map[cmd]  # 返回 add|low|jump
        return add  # 当指令错误,add作为默认功能
    
    while True:
        cmd = input('cmd: ')
        res = get_method(cmd)(10, 20)  # 根据指令获取功能并调用得到结果
        print(res)
    

     

    六、函数的嵌套调用

    # 函数的嵌套调用:在一个函数内部调用另一个函数
    # 求两个数最大值
    def max_two(n1, n2):
        if n1 > n2:
            return n1
        return n2
    
    # 求三个数最大值
    def max_three(n1, n2, n3):
        max = max_two(n1, n2)
        return max_two(max, n3)
    
    # 求四个数最大值
    def max_four(n1, n2, n3, n4):
        max = max_three(n1, n2, n3)
        return max_two(max, n4)
    print(max_four(20, 50, 30, 50))
    

     

     

    七、名称空间

    # 名称空间:存放名字与内存空间地址对应关系的容器
    # 作用:解决由于名字有限,导致名字重复发送冲突的问题 - 内置全局局部可以同时使用一个名字存放不同地址
    
    # 三种名称空间
    # Built-in:内置名称空间;系统级,一个;随解释器执行而产生,解释器停止而销毁
    # Global:全局名称空间;文件级,多个;随所属文件加载而产生,文件运行完毕而销毁
    # Local:局部名称空间;函数级,多个;随所属函数执行而产生,函数执行完毕而销毁
    
    # 加载顺序:Built-in > Global > Local
    # 	-- 采用堆栈存储数据的方式(压栈),导致内置最后被访问
    

     

    八、函数的嵌套定义

    # 函数的嵌套定义:在函数内部定义函数
    # 诞生的理由:一个函数想使用另一个函数内部的变量,可以定义在其内部
    def func():
        a = 10
        def fn():
            print(a)
        return fn
    
    new_fn = func()
    new_fn()
    

     

    两个与函数有关的关键字:global nonlocal

    # global:统一局部与全局的变量名
    num = 10
    def outer():
        # global num
        # num = 100
        def inner():
            global num
            num = 1000
            
    # nonlcal: 统一局部与嵌套局部的变量名
    def outer():
        num = 100
        def inner():
            nonlocal num
            num = 1000
    

     

    九、作用域

    # 作用域:名字起作用的范围
    # 作用:解决同名字可以共存问题 - 不同作用域相同名字的值都能在其作用域范围下进行使用
    '''
    四种作用域: LEGB
    Built-in:内置作用域 - 所有文件所有地方都可以被访问
    Global:全局作用域 - 在当前文件的所有位置
    Enclosing:嵌套作用域 - 自身内部与内部的子函数
    Local:局部作用域 - 只有自身内部
    '''
    # 加载顺序:Built-in > Global > Enclosing > Local
    # 访问(查找)顺序:报错 < Built-in < Global < Enclosing < Local
    # 作用范围:Built-in > Global > Enclosing > Local
    

     

    十、闭包

    # 闭包:定义在函数内部的函数,这个内部的函数就是闭包
    
    # 应用场景:
    # 1.可以去使用其他函数的内部变量,且还可以保证调用位置不变(闭包的函数对象作为那个函数的返回值)
    def outer():
        count = 3000
        def fn():
            print(count)  # 能使用outer内部的变量count
        return fn
    # 还是在外界调用
    outer()()  # outer()() => fn() => 调用fn
    
    
    # 2.延迟执行(外层函数可以为内存函数传递参数)
    import requests
    def outer(url):
        def show_html():
            response = requests.get(url)
            print(response.text)
        return show_html
    
    # 制作 爬百度与新浪的 函数对象
    show_baidu = outer('https://www.baidu.com')
    show_sina = outer('https://www.sina.com.cn')
    
    # 延迟到需求来了,需要爬百度,就用百度函数对象,需要爬新浪,就用新浪函数对象
    show_baidu()
    show_sina()
    show_baidu()
    

     

    十一、装饰器

    # 装饰器:装饰器就是闭包的一个应用场景
    #       -- 外层函数与内存函数形成的闭包结构的一种综合使用
    
    # 重点:开放封闭原则
    # 开放:拓展功能的点是开放的 - 可以为之前的函数添加新功能
    # 封闭:1.不能改变原函数的源代码  2.还有通过原函数的函数对象来调用函数
    
    def huaping():
        print('插花功能')
    
    temp = huaping
    def my_huaping():
        temp()
        print('观赏功能')
    huaping = my_huaping
    
    huaping()
    
    # ----------------------------------------
    
    def huaping():
        print('插花功能')
    
    def outer(temp):  # temp = huaping
        def my_huaping():
            temp()
            print('观赏功能')
        return my_huaping
    huaping = outer(huaping)  # huaping = my_huaping
    
    huaping()
    
    
    # ----------------------------------------------
    def outer(temp):  # temp = huaping
        def my_huaping():
            temp()
            print('观赏功能')
        return my_huaping
    
    @outer  # huaping = outer(huaping)
    def huaping():
        print('插花功能')
        
    huaping()
    
    
    # ------------------------------------------
    # 被装饰的函数可能有参有返:装饰器模板,可以满足所有参数,且能装饰原函数返回值
    def outer(func):  # temp = huaping
        def inner(*args, **kwargs):
            pass
            res = func(*args, **kwargs)
           	pass
        	return res
        return inner
    
    @outer
    def any_method():
        pass
    
    

    装饰器案例

    # 为登录功能添加账号检验功能:必须是3个及以上英文字母组成
    def check_user(func):
        def inner(user, pwd):
            if not (user.isalpha() and len(user) >= 3):
                return '账号不合法'
            res = func(user, pwd)
            return res
        return inner
    
    # 为登录功能添加密码检验功能:必须是3个及以上英文字母或数字组成
    def check_pwd(func):
        def inner(*args, **kwargs):
            pwd = args[1]
            if not (pwd.isalnum() and len(pwd) >= 3):
                return '密码不合法'
            res = func(*args, **kwargs)
            return res
        return inner
    
    # 对登录结果的修饰装饰器:True=>登录成功 False=>登录失败
    def change_res(func):
        def inner(*args, **kwargs):
            res = func(*args, **kwargs)
            if res == True:
                return '登录成功'
            return '登录失败'
        return inner
    
    
    # 装饰器被执行的过程是从上至下
    @check_user  # login = check_user(func=login) = inner
    @check_pwd
    @change_res
    def login(user, pwd):  # 被装饰的函数对象
        if user == 'owen' and pwd == '123':
            return True
        return False
    
    user = input('user: ')
    pwd = input('pwd: ')
    res = login(user, pwd)
    
    print(res)
    

     


     

     

    零、怎么定义函数

    '''
    1、功能的单一化
    2、参数需要外界提供、通过函数的参数来实现
    3、函数执行后的结果,通过返回值告诉给外界
    '''

     

    一、形参与实参

    '''
    def fn(形参们):
    pass

    fn(实参们)
    '''
    # 形参:定义函数,在括号内声明的变量名,用来结束外界传来的值
    # 实参:调用函数,在括号内传入的实际值,值可以为常量、变量、表达式或三者的组合

    # 注:形参随着函数的调用而产生,随着调用结束而销毁(有特例)

    实参分类

    '''
    位置实参:必须按照顺序给形参传值
    关键字实参:按照关键字给形参传值,所以顺序可以颠倒

    # 注:位置实参、关键字实参共存是,关键字实参必须出现在所有位置实参之后
    '''

    形参的分类

    '''
    位置形参:位置实参与关键字实参都可以对其传参
    默认形参:位置实参与关键字实参都可以对其传参,也可以不被传参
    可变长位置形参:接收位置形参、默认形参没有接收完的所有位置实参
    无初值关键字形参:只能由关键字实参传值
    有初值关键字形参:只能由关键字实参传值,也可以不被传值
    可变长关键字参数:接收两种关键字形参没有接收完的所有关键字实参

    注:各种形参出现的先后
    位置形参 - 默认形参 - 可变长位置形参 - 无|有初值关键字形参 - 可变长关键字参数
    '''
    def fn(a, b, c=10, *args, d, e=20, f, **kwargs):
       pass
    # 位置形参:a、b
    # 默认形参:c
    # 可变长位置形参:args
    # 无初值关键字形参:d、f
    # 有初值关键字形参:e
    # 可变长关键字参数:kwargs

    可变参数的整体使用

    # 1、可变位置形参以元组方式接收值
    # 如:接收10,20,30三个数
    def fn1(*args):
       print(args)  # (10, 20, 30)
       
    # 第一种传参
    fn1(10, 20, 30)

    # 第二种传参
    t = (10, 20, 30)  # 单列集合及字符串类型都可以
    fn1(*t)


    # 2、可变位置实参以字典方式接收值
    def fn2(**kwargs):
       print(kwargs)  # {'name': 'Owen', 'age': 18}
    # 第一种传参
    fn2(name='Owen', age=18)

    # 第二种传参
    d = {'name': 'Owen', 'age': 18}  # 单列集合及字符串类型都可以
    fn2(**d)

     

    二、函数对象

    # 函数名就是存放了函数的内存地址,存放了内存地址的变量都是对象,即 函数名 就是 函数对象

    # 函数对象的应用场景
    # 1 可以直接被引用
    # 2 可以当作函数参数传递
    # 3 可以作为函数的返回值
    # 4 可以作为容器类型的元素

     

    三、名称空间与作用域

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

    # 三种名称空间
    # Built-in:内置名称空间;系统级,一个;随解释器执行而产生,解释器停止而销毁
    # Global:全局名称空间;文件级,多个;随所属文件加载而产生,文件运行完毕而销毁
    # Local:局部名称空间;函数级,多个;随所属函数执行而产生,函数执行完毕而销毁

    # 注:
    # del 名字:可以移除查找最近的名字与内存空间地址的对应关系
    # 加载顺序:Built-in > Global > Local
    # 作用域:名字起作用的范围
    # 作业:解决同名字可以共存问题

    # 四种作用域
    # Built-in:内置作用域
    # Global:全局作用域
    # Enclosing:嵌套作用域
    # Local:局部作用域

    # 注:
    # 不同作用域之间名字不冲突,以达到名字的重用
    # 查找顺序:Local > Enclosing > Global > Built-in

     

    四、函数的嵌套定义与闭包

    # closure:被包裹的函数,称之为闭包
    
    # 完整的闭包结构:1.将函数进行闭包处理;2.提升函数名的作用域
    
    # 案例:延迟加载
    def get_site(url): #url='https://www.baidu.com'
        def site():
            response=requests.get(url)
            if response.status_code == 200:
                print(response.text)
    
        return site
    baidu=get_site('https://www.baidu.com')
    python=get_site('https://www.python.org')
    

     

    五、装饰器

    '''
    装饰器:就是闭包(闭包的一个应用场景)
    	-- 把要被装饰的函数作为外层函数的参数通过闭包操作后返回一个替代版函数
    	
    优点:
    	-- 丰富了原有函数的功能
    	-- 提高了程序的可拓展性
    '''
    

    开放封闭原则

    '''
    1.不能修改被装饰对象(函数)的源代码
    2.不能更改被修饰对象(函数)的调用方式
    '''
    

    装饰器的简单实现

    def outer(func):
        def inner():
            print("新增功能1")
            func()
            print("新增功能2")
        return inner
    
    def func():
        print("原有功能")
        
    func = outer(func)
    

    装饰器语法糖

    def outer(func):
        def inner():
            print("新增功能1")
            func()
            print("新增功能2")
        return inner
    
    @outer
    def func():
        print("原有功能")
    
    

    装饰有参有返的函数

    def outer(func):
        def inner(*args, **kwargs):
            print("新增功能1")
            result = func(*args, **kwargs)
            print("新增功能2")
            return result
        return inner
    
    @outer
    def func(*args, **kwargs):
        print("原有功能")
        return "原有结果"
        
    
    

    有参装饰器

    def wrap(arg):
        def outer(func):
            def inner(*args, **kwargs):
                print("新增功能1")
                result = func(*args, **kwargs)
                print("新增功能2")
                return result
            return inner
    
    @wrap("装饰器参数")
    def func(*args, **kwargs):
        print("原有功能")
        return "原有结果"
    

    wraps修改函数文档注释

    from functools import wraps
    def outer(func):
        @wraps(func)
        def inner(*args, **kwargs):
            '''装饰器文档注释'''
            func(*args, **kwargs)
        return inner
    
    @outer
    def func(*args, **kwargs):
        '''原有文档注释'''
        print("原有功能")
    

     

    六、迭代器

    # 迭代器:循环反馈的容器(集合类型)
    #	-- 不同于索引取值,但也可以循环的从容器对象中从前往后逐个返回内部的值
    
    # 优点:不依赖索引,完成取值
    # 缺点:不能计算长度,不能指定位取值(只能从前往后逐一取值)
    

    可迭代对象

    # 可迭代对象:内置有__iter__()方法的对象,称之为可迭代对象
    
    # 可迭代对象:str | list | tuple | set | dict | range() | enumerate() | file | 生成器对象
    

    迭代器对象

    # 迭代器对象:内置有__next__()方法的对象,称之为迭代器对象,可以通过调用__next__()方法完成取值
    
    # 迭代器对象:enumerate() | file | 生成器对象
    
    # 注:迭代器对象调用__iter__()方法返回的还是迭代器对象
    

    迭代器

    # 迭代器:for循环
    # 工作原理:
    #	-- 1.获取可迭代对象的迭代器对象
    #	-- 2.循环__next__()取值
    #	-- 3.异常处理(StopIteration),停止迭代
    

    生成器

    # 生成器:一次生成一个值得容器(函数类型)
    #	-- 内部包含yield关键字的函数
    
    # 注:函数的调用不会执行函数,而是返回生成器对象
    

    生成器对象

    # 生成器对象:就是一个迭代器对象
    #	-- 可以通过__next__()方法取值,得到yield关键字的返回值
    #	-- 可以调用send()方法给yield关键字传值,内部接收yield关键字可以得到传入的值
    

    枚举对象

    # 枚举对象:通过enumerate()方法,可以为可迭代对象生成迭代索引,其本身也是一个迭代器对象
    

     

  • 相关阅读:
    Prince and princess「DP优化」
    Wooden Stricks——两个递增条件的线性DP
    死磕 java线程系列之线程池深入解析——构造方法
    死磕 java线程系列之线程池深入解析——体系结构
    死磕 java线程系列之自己动手写一个线程池(续)
    死磕 java线程系列之自己动手写一个线程池
    死磕 java线程系列之创建线程的8种方式
    死磕 java线程系列之线程模型
    死磕 java同步系列之终结篇
    死磕 java同步系列之redis分布式锁进化史
  • 原文地址:https://www.cnblogs.com/qiangyuzhou/p/10787330.html
Copyright © 2011-2022 走看看