• 四、global和nonlocal、函数名应用、格式化输出


    一、面试的常见坑

    • 形参角度,默认参数的坑

    如果你的默认参数指向的是可变的数据类型,那么你无论调用多少次这个默认参数,都是同一个

    # 默认参数的陷阱:
    def func(name,sex='男'):
       print(name)
       print(sex)
    func('alex')
    #结果:
    alex


    # 陷阱只针对于默认参数是可变的数据类型:
    def func(name,alist=[]):
       alist.append(name)
       return alist

    ret1 = func('alex')
    print(ret1,id(ret1))
    ret2 = func('太白')
    print(ret2,id(ret2))
    #结果:共用一个列表
    ['alex'] 2289216485128
    ['alex', '太白'] 2289216485128

    # 如果你的默认参数指向的是可变的数据类型,那么你无论调用多少次这个默认参数,都是同一个。
    def func(name,alist=[]):
       alist.append(name)
       return alist

    print(func(10,))  # [10,]
    print(func(20,[]))  # [20,]
    print(func(100,))  # [10,100]
    # print(func(20,[]))是name传值为10,alist传值为[],等同于新建立一个列表,不与print(func(10,))建立的列表用一个;print(func(100,)) 没有建立新列表,与print(func(10,))建立的列表用一个
    # 相当于以下代码:
    # l1 = []
    # l1.append(10)
    # print(l1)
    # l2 = []
    # l2.append(20)
    # print(l2)
    # l1.append(100)
    # print(l1)

    # 与上个代码对比:
    def func(a, list= []):
       list.append(a)
       return list
    ret1 = func(10,)    # ret1 = [10,]
    ret2 = func(20,[])  # ret2 = [20,]
    ret3 = func(100,)   # ret3 = [10,100]此时因为ret1用同一个列表,所以ret1也改变
    print(ret1)  # [10,100] (与上个代码对比,此处ret1没先打印,因而被ret3改变,然后打印)
    print(ret2)  # [20,]
    print(ret3)  # [10,100]
    • 局部作用域的坑

    在函数中,如果你定义了一个变量,但是在定义这个变量之前对其引用了,那么解释器认为:语法问题,你应该在使用之前先定义


    count = 1
    def func():
       print(count)
       count = 3
    func()
    # 代码在print(count)处飘红,local variable 'count' referenced before assignment,就是赋值前引用的本地变量“count”

     二、global nonlocal

      • global:

        1. 在局部作用域声明一个全局变量


          name = 'alex'
          def func():
             name = '太白'
             print(name)
          func()
          print(name)
          # 结果:
          太白
          alex


          # 还是以上代码加入global,
          name = 'alex'
          def func():
             global name
             name = '太白'
             print(name)
          func()
          print(name)
          # 结果:
          太白
          太白


          name = 'alex'
          def func():
             global name
             name = '太白金星'
          print(name)       #alex
          print(globals())  #获取全局变量
          func()
          print(name)       #太白金星
          print(globals()   #获取全局变量
        2. 修改一个全局变量


          count = 1
          def func():
             #print(count) ,报错,在全局声明之前使用名称“count”
             global count
             count += 1
          print(count)     # 1
          func()
          print(count)     # 2
          #将全局变量的count = 1,在函数中改成了2
      • nonlocal:

        1. 不能够操作全局变量


          count = 1
          def func():
             nonlocal count
             count += 1
          func()
          # 会报错,nonlocal是不能用来操作全局变量的
        2. 局部作用域:内层函数对外层函数的局部变量进行修改


          def wrapper():
             count = 1
             def inner():
                 nonlocal count
                 count += 1
             print(count)   # 1
             inner()
             print(count)   # 2
          wrapper()

      三、函数名的运用

    • 函数名指向的是函数的内存地址,函数名+()就可执行此函数

    • 函数名就是变量

    • 函数名可以作为容器类数据类型内容的元素

    • 函数名可以作为函数的参数

    • 函数名可以作为函数的返回值


    #1.函数名指向的是函数的内存地址
    def func():
       print(666)
    func()

    #2.函数名就是变量
    def func():
       print(666)
    f = func
    f1 = f
    f2 = f1
    f()      #相当于func() 666
    func()   #相当于func() 666
    f1()     #相当于func() 666    
    f2()     #相当于func() 666

    def func():
       print('in func')
    def func1():
       print('in func1')
    func1 = func
    func1()    #相当于func() in func

    #3.函数名可以作为容器类数据类型的元素
    def func1():
       print('in func1')

    def func2():
       print('in func2')

    def func3():
       print('in func3')
    l1 = [func1,func2,func3]
    for i in l1:
       i()
    #结果依次调用 func1,func2,func3
    in func1
    in func2
    in func3

    #4.函数名可以作为函数的参数
    def func(a):
       print(a)
       print('in func')
    b = 3
    func(b)
    print(func)


    def func():
       print('in func')

    def func1(x):
       x()
       print('in func1')

    func1(func)

    # #5.函数名可作为函数的返回值
    def func():
       print('in func')

    def func1(x):
       print('in func1')
       return x

    ret = func1(func)
    ret()

      四、新特性:格式化输出(3.6x以后的适用)

    可以加表达式,可以结合函数写:


    # %s format
    # name = '太白'
    # age = 18
    # msg = '我叫%s,今年%s' %(name,age)
    # msg1 = '我叫{},今年{}'.format(name,age)

    # 新特性:格式化输出
    name = '太白'
    age = 18
    msg = f'我叫{name},今年{age}'
    print(msg)

    count = 7
    print(f'最终结果:{count**2}')

    name = 'barry'
    msg = f'我的名字是{name.upper()}'
    print(msg)

    #加表达式:
    dic = {'name':'alex','age':13}
    msg = f'我叫{dic["name"]},今年{dic["age"]}'
    print(msg)

    #结合函数:
    def _sum(a,b):
       return a+b
    msg = f'最终结果为:{_sum(10,20)}'
    print(msg)

    # ! , : { } ;这些标点不能出现在{} 这里面。

    优点:

      1. 结构更加简化

        1. 可以结合表达式、函数进行使用

        2. 效率提升很多

  • 相关阅读:
    07.对称加密算法指令
    06.Openssl基本概念
    04.openssl背景
    03.openssl密码实现技术
    02.密钥学基本概念
    01.openssl-概述
    17行为型模式之命令模式
    16行为型模式之模板模式
    15结构型模式之享元模式
    14结构型模式之外观模式
  • 原文地址:https://www.cnblogs.com/yangzm/p/10921558.html
Copyright © 2020-2023  润新知