• 7.9函数(一)


    函数

    一、函数简介

    1.内置函数:python提前给你的写好了的函数,你直接调用即可

    s = 'hello'
    print(len(s))  # len() 内置函数:python提前给你的写好了的函数  你直接调用即可

    2.自定义函数:函数必须先定义后调用,定义了的函数可以在任意位置调用

    s = 'hello'
    def my_len():  # 自定义函数
        n = 0
        for i in s:
            n += 1
        print(n)
    my_len()  # 函数必须先定义后调用    定义了的函数可以在任意位置调用

    此自定义函数my_len()的问题所在:

    (1)没有返回值,只能固定的执行打印操作

    (2)只能够固定的统计某一个容器类型的长度

    二、函数的返回值

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

    def func():
        return '返回值'  # 函数内要想返回给调用者值  必须用关键字return
    res = func()
    print(res)

    1.不写return

    不写return ,函数默认返回None

    def func():
        print('hahaha')
    res = func()
    print(res)  # 默认返回None

    2.只写return

    return可以返回值之外,还可以直接结束整个函数的运行

    def func():
        l = ['jason','egon','tank']
        while True:
            for i in l:
                if i == 'egon':  # 当i为egon的时候,直接结束函数运行
                    return  # 直接结束了for和while循环,结束整个函数运行
                    print('asdasdkljlsjadl')  # 这一行代码永远都不会运行
                print(i)
    res = func()
    print(res)  # 只写return 返回的也是None(None就表示什么都没有)
    View Code

    3.写return None

    跟上面的只写return是一样的

    def func():
        return None
    res = func()
    print(res)  # 返回None

    4.写return加上返回一个值

    这个值可以是python任意数据类型

    def func():
        return '123'
    def func1():
        return [1,2,3]
    def func2():
        return {'name':'jason'}
    def func3():
        return (1,)
    def func4():
        return {1,2,3,4,5}
    def func5():
        return True
    print(func(),func1(),func2(),func3(),func4(),func5())  # 省略res = func()
    View Code

    5.写return加上返回多个值

    return会自动将多个值以元组的形式返回给调用者

    def func():
        return 1,2,3,4  # 返回的是(1, 2, 3, 4)
    res = func()
    print(res)
    
    def func1():
        return 'a','b','c'  # ('a', 'b', 'c')
    res1 = func1()
    print(res1)
    
    def func2():
        return [1,2,3],[1,2,3],[1,2,3]  # ([1, 2, 3], [1, 2, 3], [1, 2, 3])
    res2 = func2()
    print(res2)
    
    def func3():
        return {'name':'jason'},{'username':'tank'},{'user_name':'egon'}  # ({'name': 'jason'}, {'username': 'tank'}, {'user_name': 'egon'})
    res3 = func3()
    print(res3)
    View Code

    (1)为什么组织成元组返回

    因为函数不希望自己处理的结果被修改(元组不可变)

    (2)怎样不返回元组形式

    返回多个值,并且不想让return帮你做处理(指返回元组形式),只需要自己手动加上你想返回的数据类型符号

    def func():
        return [1,2,3,4]  # 返回的是[1, 2, 3, 4]
    res = func()
    print(res)
    
    def func1():
        return ['a','b','c']  # ['a', 'b', 'c']
    res1 = func1()
    print(res1)
    
    def func2():
        return [[1,2,3],[1,2,3],[1,2,3]]  # [[1, 2, 3], [1, 2, 3], [1, 2, 3]]
    res2 = func2()
    print(res2)
    
    def func3():
        return [{'name':'jason'},{'username':'tank'},{'user_name':'egon'}] # [{'name': 'jason'}, {'username': 'tank'}, {'user_name': 'egon'}]
    res3 = func3()
    print(res3)
    View Code

    6.总结:

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

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

    三、函数的参数概要

    s = 'hello'
    l = [1,2,3,4]
    print(len(s))  # 需要1个参数
    print(len(l))
    
    def my_len():  # 需要0个参数
        n = 0
        for i in s:
            n += 1
        return n
    print(my_len())

    如何让自定义函数实现内置函数添加参数的功能?

    s = 'hello'
    l = [1,2,3,4]
    def my_len(args):  # 给my_len()传一个参数
        print(args)
        n = 0
        for i in args:  # 对象也该为该参数
            n += 1
        return n
    print(my_len(s))
    print(my_len(l))
    print(my_len('hello world'))  # 实现了len()可以添加参数的功能

    1.函数参数的两大类型

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

    (2)实参:在函数的调用阶段,括号内实际传入的值,叫做该函数的实际参数。简称:实参

    2.形参与实参的关系

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

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

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

         (2)只在函数内部有效,函数外部无任何影响

    3.函数的简易结构:

    def 函数名(形参1,形参2...):
      '''
      函数的注释,用来描述该函数的作用以及各个形参的类型
      '''
      函数体代码1
      函数体代码2
      ...
      return 函数的返回值

    4.函数的注释

    怎么写以及怎么查看

    def func(x,y):
        '''
      注释第一行:写该函数的作用
      :param x: 对形参x的解释
      :param y: 对形参y的解释
      :return: 对函数返回值的解释
      '''
        print('hahaha')
        return 'heihei'
    print(help(func))  # 调出该函数的注释
    print(help(len))  # 调出该函数的注释

    四、位置参数

    1.位置参数:

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

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

    (3)在函数的调用阶段,传入的参数会按照位置一一对应给形参

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

    def my_max(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)

    第二种:指名道姓的传(关键字传参)

    def my_max(x,y):
        if x > y:
            return x
        else:
            return y
    my_max(y=20,x=10)  # 关键字传参
    print(my_max(y=20,x=10))
    my_max(10,y=20)  # 位置传和关键字传参混合使用
    print(my_max(10,y=20))
    my_max(x=10,20)  # 报错,位置参数必须在关键字参数的前面
    my_max(20,y=40,x=30)  # 报错,同一个形参不能被多次赋值

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

    但是必须保证:

    (1)位置参数必须在关键字参数的前面(越短的越靠前,越长的越复杂的越靠后)

    (2)同一个形参不能被多次赋值

    五、默认值参数

    1.默认值参数:

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

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

    def my_max(x,y=100):
        print(x,y)
        if x > y:
            return x
        return y
    res = my_max(200)
    print(res)

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

    def my_max(x,y=100):
        print(x,y)
        if x > y:
            return x
        return y
    res = my_max(200,1000)
    print(res)

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

    2.默认值参数的应用场景

    当形参接收的到值比较单一的情况下 通常可以考虑用默认值形参

    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')

    3.面试题:统计每个人的爱好

    def info(username,hobby,l=[]):
        l.append(hobby)
        print('%s 的爱好是 %s'%(username,l))
    info('jason','study')
    info('tank','read')
    info('egon','write')
    # jason 的爱好是 ['study']
    # tank 的爱好是 ['study', 'read']
    # egon 的爱好是 ['study', 'read', 'write']

    如何解决这个问题

    (1)第一种

    不使用默认值参数,自己传参数

    def info(username,hobby,l=[]):
        l.append(hobby)
        print('%s 的爱好是 %s'%(username,l))
    info('jason','study',[])
    info('tank','read',[])
    info('egon','write',[])
    # jason 的爱好是 ['study']
    # tank 的爱好是 ['read']
    # egon 的爱好是 ['write']

    (2)第二种

    调用阶段重新赋值

    def info(username,hobby,l=None):
        if l == None:
            l = []
        l.append(hobby)
        print('%s 的爱好是 %s'%(username,l))
    info('jason','study')
    info('tank','read')
    info('egon','write')
    # jason 的爱好是 ['study']
    # tank 的爱好是 ['read']
    # egon 的爱好是 ['write']

    4.函数在定义阶段,内部所使用的变量都已经初始化完毕了,不会因为调用的位置的变化 而影响到内部的值(暂时可忽略)

    m = 100
    def my_max(x,y=m):
        print(x,y)
    m = 222
    my_max(111)

    函数无论在什么地方被调用,都会跑到函数定义阶段去执行代码,形参中用到的值都是往函数定义阶段代码往上找

    六、可变长参数

    站在调用函数传递实参的角度,实参的个数不固定的情况,也就意味形参也不固定,如何解决?

    解决办法:站在形参的角度,可以用可以用*和**来接收多余的(溢出的)位置参数和关键字参数

    1. * 

    (1)站在形参的角度,看 *

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

    def func(x,y,*z):
        print(x,y,z)  # z = (3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
    func(1,2,3,4,5,6,7,8,9,10,11,12)

    (2)站在实参的角度,看 *

    * 的内部你可以看成是for循环,只能将列表,元组,集合,字符串打散并一一传入

    def func(x,y,z):
        print(x,y,z)
    func(*[1,2,3])  # *会将列表打散成位置实参一一传入,等价于func(1,2,3)

    (3)配合使用

    def func(x,*z):
        print(x,z)
    func(1,*{1,2,3})  # *在形参中只能接收多余的位置实参 不能接收关键字实参

    2. ** 

    (1)站在形参的角度,看 **

    ** 会接收所有多余的关键字参数,并将关键字参数转换成字典的形式,字典的key就是关键字的名字,字典的value就是关键字的名字指向的值,将字典交给 ** 后面的变量名

    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)

    (2)站在实参的角度,看 ** 

    ** 会将字典拆封成 key = value的形式,以关键字参数的形式传递给函数

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

    3.总结:

    (1)形参中:

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

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

    (2)实参中:

      * 在实参中能够将列表、元祖、集合、字符串,打散成位置实参的形式传递给函数( * 就看成是for循环取值)

      ** 在实参中能将字典打散成key = value的形式,以关键字参数形式传递给函数

    4.如何使你写的函数无论调用者按照正确传参的方式无论怎么传,你的函数都能够正常执行?

    def func1(*x,**y):
        print(x,y)  # x = (1, 2, 3, 4, 5, 6)  y = {'x': 1, 'y': 2, 'z': 3}
    func1(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)
  • 相关阅读:
    主效应|处理误差 |组间误差|处理效应|随机误差|组内误差|误差|效应分析|方差齐性检验|SSE|SSA|SST|MSE|MSA|F检验|关系系数|完全随机化设计|区组设计|析因分析
    第二类错误|检验统计量|左偏|右偏|P值
    估计量|估计值|置信度|置信水平|非正态的小样本|t分布|大样本抽样分布|总体方差|
    参数|统计量|抽样分布|估计标准误差|标准误差|标准误|标准差|二项分布|泊松分布|中心极限定理|样本方差|
    ruby 分析日志,提取特定记录
    find 找出大文件
    momentjs 求小时差异
    linux下对date和timestamp的互转
    golang protobuf SetExtension
    对文本中的某两列求和,请统计重复出现次数
  • 原文地址:https://www.cnblogs.com/francis1/p/11158881.html
Copyright © 2020-2023  润新知