• 初识函数定义与调用 * 和 **


    在了解函数之前,我们先想象一个这样的场景:

      如果现在len方法不能用了,我们要如何用代码计算‘kilobitten’这个字符串的长度?

    str1 = 'kilobitten'
    length = 0
    for i in str1:
        length += 1
    print(length)  # 10

      此时又有一个需求,要计算另一个字符串的长,“New Centry”,我们该怎么做呢?如法炮制

    str1 = 'kilobitten'
    length = 0
    for i in str1:
        length += 1
    print(length)  # 10

    str2 = 'The New Centry' length2 = 0 for j in str2: length2 += 1 print(length2) # 14

      不难发现,虽然这样可以满足需求,但是造成了代码冗余

      并且对比len方法来说,这种实现方式的代码可读性较差

    str1 = 'kilobitten'
    str2 = 'The New Centry'
    print(len(str1))  # 10
    print(len(str2))  # 14

      这时候,我们就希望能有个方法能像len那样,可以直接使用,来计算长度。

    于是,就引入了函数的概念。

    函数

      函数,就是具有某个具体功能的工具

    为什么要有函数?

      提供开发效率
      减少代码冗余
      提高程序的扩展性

    定义一个函数

      'def' 是定义函数的关键字

      函数名:函数名的命名规则和变量名一致

        1.不能以关键字命名(**重要**)

        2.函数名也应该做到见名知意

      需要注意的是:

        函数在定义的时候只检测函数体语法 不执行函数体代码

    1 s = 'kilobitten'
    2 
    3 def my_len():  # 自定义函数 # 在定义函数时,以下4行代码并不执行
    4     n = 0
    5     for i in s:
    6         n += 1
    7     print(n)  # 10
    res = my_len()
    print(res)  # None

        调用函数的固定格式

          函数名+括号
          函数名只要遇到括号会立即执行函数体代码
          代码中遇到函数名加括号 优先级最高
          先去执行函数 再看下面的代码

        我们自己写的函数现在的问题

          1.没有返回值 只能固定的执行打印操作
          2.只能够固定的统计某一个容器类型的长度

    返回值

      上面我们自己写的代码,虽然可以正确运行但不能像len方法那样有返回值。

      函数内要想返回给调用者值 必须用关键字return

      

      关于return有五种情况

      1.不写return

    # 不写return:函数默认返回None
    def func():
        print('hahaha')  # hahaha
    res = func()
    print(res)  # None

      2.只写return

    # 只写return:return除了可以返回值之外 还可以直接结束整个函数的运行
    # 只写return 返回的也是None(None就表示什么都没有)
    def func():
        l = ['jason','egon','tank']
        while True:
            for i in l:
                if i == 'egon':  # 当i为egon的时候 直接结束函数运行
                    # 下面的return相当于break
                    return
                    # print('asdasdkljlsjadl')  # 这一行代码拥有都不会运行
                print(i)  # jason
    res = func()
    print(res)  # None

      3.写return None

    # 写return None:跟上面的只写return是一样的
    def func():
        return None
    res = func()
    print(res)  # None

      4.写return返回一个值

    # 写return返回一个值:这个值可以是python任意数据类型
    def func():
        return '123'  # 123
    def func1():
        return [1,2,3]  # [1,2,3]
    def func2():
        return {'name':'jason'}  # {'name': 'jason'}
    def func3():
        return (1,)  # (1,)
    def func4():
        return {1,2,3,4,5}  # {1, 2, 3, 4, 5}
    def func5():
        return True  # True
    print(func(),func1(),func2(),func3(),func4(),func5())
    # 123 [1, 2, 3] {'name': 'jason'} (1,) {1, 2, 3, 4, 5} True

      5.写return返回多个值

    # 写return返回多个值:return会自动将多个值以元组的形式返回给调用者
    '''
    1.为什么组织成元组返回
        函数不希望自己处理的结果被修改
    2.如何不返回元祖
    '''
    
    
    def func():
        return 1, 2, 3, 4
    res0 = func()
    print(res0)  # 返回的是(1, 2, 3, 4)
    
    
    def func1():
        return 'a', 'b', 'c'
    res = func1()
    print(res)  # 返回的是 ('a', 'b', 'c')
    
    
    def func2():
        return [1, 2, 3], [1, 2, 3], [1, 2, 3]  # ([1, 2, 3], [1, 2, 3], [1, 2, 3])
    
    
    res1 = func2()
    print(res1)
    
    
    def func3():
        return {'name': 'jason'}, {'username': 'tank'}, {
            'user_name': 'egon'}  # ({'name': 'jason'}, {'username': 'tank'}, {'user_name': 'egon'})
    
    
    res2 = func3()
    print(res2)
    
    
    # 返回多个值 并且不想让return帮你做处理  自己手动加上你想返回的数据类型符号
    def func4():
        return [[1, 2, 3, 4], [1, 2, 3, 4], [1, 2, 34]]
    
    
    res = func4()
    print(res)

      # 1.所有的函数都有返回值,无论你写不写return
        python中所有的函数都有返回值 不写的情况下默认返回None

      # 2.光写return 或者return None并不是为了考虑返回值 而是为了结束函数的运行

    函数的参数概要

    函数参数的两大类型

      形参:在函数的定义阶段 括号内写的变量名 叫做该函数的形式参数 简称 形参

      实参:在函数的调用阶段 括号内实际传入的值 叫做实际参数 简称 实参

      形参和实参的关系

        形参就相当于变量名,而实参就相当于变量的值

        函数调用传参的过程,就是给形参变量名赋值的过程

      注:形参和实参的绑定关系只在函数的调用阶段有效,函数运行结束关系自动解除

        只在函数内部有效,函数外部无影响

      函数的简易结构

        def 函数名(形参1,形参2...)
          '''
          函数的注释
          描述该函数的作用
          以及各个形参的类型       
    '''       函数体代码1       函数体代码2       .........       return 函数的返回值
        def func(x,y):
            """
            该函数的作用
            :param x: 对形参x的解释
            :param y: 对形参y的解释
            :return: 对函数返回值的解释
            """
            print('hahaha')
            return 'heihei'
    
          # print(help(func))
          # print(help(len))
          # 通过这种形式 去查看相关函数的注释

    位置参数

      位置参数:在函数定义阶段按照位置从左往右依次书写的变量名 叫做函数位置形参

      位置形参在调用的时候 必须为其传值

    def my_max(x,y):
        print(x,y)
        if x > y:
            return x
        else:
            return y
    
    # res = my_max(1)  # 在调用函数的时候 少一个实参不行
    # res = my_max(1,2,3)  # 在调用函数的时候 多一个实参也不行
    
    # res = my_max(20,10)
    # 这样才可以
    
    # 位置实参:在函数的调用阶段 传入的参数会按照位置一一对应给形参
    # print(res)
      传参方法

      # 第一种直接按照位置传 一一对应

      # 第二种指名道姓的传 >>>:关键字传参

    # my_max(y=20,x=10)
    # my_max(10,y=20)  # 位置和关键字混合使用

      #注意:在函数的调用阶段 位置参数和关键字参数可以混合使用

        但是必须保证

        1.位置参数必须在关键字参数的前面(越短的越靠前,越长的越复杂的越靠后)
        2.同一个形参不能被多次赋值

    默认值参数

      默认值参数:在函数的定义阶段,形参(变量名)就已经被赋值了

      在调用的时候可以不为默认值形参传值,默认使用定义阶段就已经绑定的值

      在调用的时候如果可以给默认值形参传值 传了那么就使用你传的值

      在定义阶段 默认值形参必须放在位置形参的后面

    def my_max(x,y=100):
        if x > y:
            return x
        return y
    # res = my_max(200)
    res1 = my_max(200,1000)
    res2 = my_max(y=200,x=1000)
    print(res2)
    def register(username,age,gender='male'):
        print(username,age,gender)
    register('jason',18)
    register('tank',28)
    register('egon',84)
    register('kevin',58)
    register('xiaohou',17,'female')
    # 默认值参数的应用场景
    # 当形参接收的到值比较单一的情况下 通常可以考虑用默认值形参

    重要坑点1

    def info(username,hobby,l=None):
        if l == None:
            l = []  # 因为这里给的是空列表,所以有传入的,就会添加进去
        l.append(hobby)
        print('%s 的爱好是 %s'%(username,l))
    info('jason','study')
    info('tank','生蚝')
    info('kevin','喝腰子汤')
    info('egon','女教练')
    
    # 解决方法1
    info('jason','study',[])
    info('tank','生蚝',[])
    info('kevin','喝腰子汤',[])
    info('egon','女教练',[])
    
    # 解决方法2
    info('jason','study')
    info('tank','生蚝')
    info('kevin','喝腰子汤')
    info('egon','女教练')

    重要坑点2

    m = 100
    def my_max(x,y=m):
        print(x,y)
    m = 222
    my_max(111)  # 111 100
    
    # 因为一开始 m已经指向了100
    # 然后m=100 赋值给y
    # 再把m指向222 已经和y无关了
    # 函数在定义阶段 内部所使用的变量都已经初始化完毕了
    # 不会因为调用的位置的变化 而影响到内部的值(暂时可忽略)
    
    # 函数无论在什么地方被调用
    # 都会跑到函数定义阶段去执行代码
    # 形参中用到的值都是往函数定义阶段代码往上找

    可变长实参

      站在调用函数传递实参的角度 实参的个数不固定的情况
      也就意味形参也不固定
      站在形参的角度 可以用*和**来接收多余的(溢出的)位置参数和关键字参数

    *

      站在形参的角度 看 *
      形参中的*会将多余的(溢出的)位置实参 统一用元组的形式处理 传递给*后面的形参名

    def func(x,y,*z):
        print(x,y,z)  # z = (3, 4, 5, 6, 7, 8, 54, 43, 4, 5, 6, 6, 7, 8)
    func(1,2,3,4,5,6,7,8,54,43,4,5,6,6,7,8,)

      站在实参的角度 看 *

    def func(x,y,z):
        print(x,y,z)
    # l = [1,2,3]
    # a,b,c = l
    # func(a,b,c)
    # func(*[1,2,3,4,5,6])  # *会将列表打散成位置实参一一传入等价于func(1,2,3,4,5,6)
    func(*(1,2,3))  # 等价于func(1,2,3)
    def func(x,*z):
        print(x,z)
    func(1,*{1,2,3})  # *在形参中只能接收多余的位置实参 不能接收关键字实参

      # *只能将列表 元组 集合 字符串
      # *的内部你可以看成是for循环

    **

      站在形参的角度看 **

    def func(x,y,**z):
        print(x,y,z)  # z = {'z': 1, 'a': 1, 'b': 2, 'c': 3}
    func(x=1,y=2,z=1,a=1,b=2,c=3)

      # **会接收所有多余的关键字参数 并将关键字参数 转换成字典的形式

        字典的key就是关键字的名字

      # 字典的value就是关键字的名字指向的值 将字典交给**后面的变量名

      

      站在实参的角度看 **

    def func(x,y,z):
        print(x,y,z)
    func(12,3,4)
    func(x=1,y=2,z=3)
    d = {'x':1,'y':2,'z':333}
    func(x=1,y=2,z=3)
    func(**d)  # 等价于func(x=1,y=2,z=333)

      # **会将字典拆封成 key = value的形式

    总结

      * 与 **
      

      *在形参中能够接受多余的位置参数 组织成一个元祖赋值给*后面的变量名
      **在形参中能够接受多余的关键字参数 组织成一个字典赋值给**后面的变量名


      *:在实参中 *能够将列表 元祖 集合 字符串 打散成位置实参的形式传递给函数
      (*就看成是for循环取值)
      **:在实参中 能将字典打散成key = value的形式 按照关键字参数传递给函数

      

    现提出一个需求:

      你写的函数 无论调用者按照正确传参的方式无论怎么传

      你的函数都能够正常执行

    def func1(*x,**y):
        print(x,y)
    func1(1,2,3,4,5,6,x=1,y=2,z = 3)
    # (1, 2, 3, 4, 5, 6) {'x': 1, 'y': 2, 'z': 3}

      # 注意python推荐形参*和**通用的写法

    def func2(*args,**kwargs):
        print(args,kwargs)
    func2(1,2,3,4,5,6,x=1,y=2,z = 3)
     补充: 

      # 命名关键字参数:在函数定义阶段 写在*与**可变长参数之间的形参
      # 在给命名关键字参数传值的时候 只能用关键字为其传值
      # 在定义阶段 给你感觉好像z是默认值参数 放在了m这个位置参数的前面了 应该报错
      # 其实z和m都是命名关键字参数 不是默认值参数和位置参数 

    def func(x,y=1,*args,z=3,m,**kwargs):
    print(x,y)
    print(args)
    print(z,m)
    print(kwargs)
    func(1,2,1,2,3,4,5,6,7,78,8,9,0,z=69,m=999,o=666999,l = 999666)
  • 相关阅读:
    解决在Apple Silicon (M1)安装php MongoDB扩展失败
    dyld: Library not loaded: /usr/local/opt/libpng/lib/libpng16.16.dylib
    docker自建bitwarden_rs服务器更新支持send功能
    centos安装puppeteer遇到的报错及解决方案
    Centos宝塔安装NextCloud
    苹果设备型号代码(更新至iPhone12)
    electron内使用vue-slider-component组件报“$attrs is readonly”错误
    ZSH隐藏命令行前面的用户名和主机名
    Android9.0配置charles的https抓包
    记一次discuz修改首页图片路径问题
  • 原文地址:https://www.cnblogs.com/PowerTips/p/11159459.html
Copyright © 2020-2023  润新知