zoukankan      html  css  js  c++  java
  • python 函数、参数及参数解构

    函数

    • 数学定义
      • y=f(x), y是x函数,x是自变量。y=f(x0,x1...xn)
    • Python函数
      • 由若干语句组成的语句块,函数名称,参数列表构成,它是组织代码的最小单位
      • 完成一定的功能
    • 函数作用
      • 结构化编程对代码的最基本的封装,一般按照功能组织一段代码
      • 封装的目的为了复用,减少冗余代码
      • 代码更加简洁美观,可读易懂
    • 函数分类
      • 内建函数,如max(),reversed()等
      • 库函数,如math.ceil()等

    函数定义、调用

    def语句定义函数

    def 函数名(参数列表):

      函数体(代码块)

      [return 返回值]

    • 函数名就是标识符,命名要求一致
    • 语句块必须缩进,约定4个空格
    • Python的函数没有return语句,隐式会返回一个None值
    • 定义中的参数列表成为形式参数,只是一种符号表达,简称形参

    调用:

    • 函数定义,只是声明了一个函数,它不会被执行,需要调用
    • 调用的方式,就是函数名加上小括号,括号内写上参数
    • 调用时写的参数就是实际参数,是实实在在传入的值,简称实参

    举例:

    #定义函数
    def add(x,y):
        result = x + y
        return result
    #调用函数
    out = add(4,5)
    print(out)
    • 上面只是一个函数的定义,有一个函数叫做add,接收2个参数
    • 计算的结果,通过返回值返回
    • 调用通过函数名add加2个参数,返回值可使用变量接收
    • 定义需要在调用前,也就是说调用时,已经被定义过了,否则抛NameError异常
    • 函数是可调用的对象,callable()

    函数参数

    参数调用时传入的参数要和定义的个数相匹配(可变参数例外)

    位置参数

    • def f(x,y,z) 调用使用f(1,3,5)
    • 按照参数定义顺序传入实参

    关键字参数

    • def f(x,y,z) 调用使用f(x=1,y=3,z=5)
    • 使用形参的名字来传入实参的方式,如果使用了形参的名字,那么传参顺序就可和定义顺序不同

    传参

    • f(z=None,y=10,x=[1])
    • f((1,),z=6,y=4.1)
    • f(y=5,z=6,2) #报错,要求位置参数必须在关键字参数之前传入,位置参数是按位置对应的

    函数参数默认值

    参数默认值(缺省值)

    • 定义时,在形参后面跟上一个值
    def add(x=4,y=5):
        result = x + y
        return result

    调用测试

    add(6, 10),add(6, y=7),add(x=5),add(),add(y=7),add(x=5, y=6),add(y=5, x=6)#add(x=5,6),add(y=8,4)要求位置参数必须在关键字参数之前传入

    作用:

    • 参数的默认值可以在未传入足够的实参的时候,对没有给定的参数赋值为默认值
    • 参数非常多的时候,并不需要用户每次都输入所有的参数,简化函数调用

    举例:

    • 定义一个函数login,参数名称为host、port、username、password
    def login(host='127.0.0.1',port='8080',username='jaxzhai',password='123'):
        print('{}:{}@{}/{}'.format(host,port,username,password))
    login()
    login('127.0.0.1', 80, 'jax', 'zhai')
    login('127.0.0.1', username='root') 
    login('localhost', port=80,password='com') 
    login(port=80, password='123', host='www')

     可变参数

    可变参数

    • 一个形参可以匹配任意个参数

    位置参数的可变参数

    • 有多个数,需要累加求和
    #构造函数
    def add(*nums):
        sum = 0
        print(nums)
        for a in nums:
            sum +=a
        return sum
    #调用
    add(1,4,5,1,2,5)
    • 在形参前使用*表示该形参是可变参数,可以接收多个实参
    • 收集多个实参为一个tuple

    关键字参数的可变参数

    • 配置信息打印
    #关键字参数的可变参数
    def showconfig(**kwargs): 
        for k,v in kwargs.items():
            print('{} = {}'.format(k, v))
    #调用
    showconfig(host='127.0.0.1',port='80',username='jax',password='123')
    • 形参前使用**符合,表示可以接收多个关键字参数
    • 收集的实参名称和值组成一个字典

    可变参数混合使用

    • 配置信息打印
    def showconfig(username, password, **kwargs)
    def showconfig(username, *args, **kwargs)
    def showconfig(username, password, **kwargs, *args)

    总结:

    • 有位置可变参数和关键字可变参数
    • 位置可变参数在形参前使用一个星号*
    • 关键字可变参数在形参前使用两个星号**
    • 位置可变参数和关键字可变参数都可以收集若干个实参,位置可变参数收集形成一个tuple,关键字可变参数收集形成一个dict
    • 混合使用参数的时候,可变参数要放到参数列表的最后,普通参数需要放到参数列表前面,位置可变参数需要在关键字可变参数之前

    举例:

    #定义函数
    def fn(x, y, *args, **kwargs):
        print(x) 
        print(y) 
        print(args) 
        print(kwargs)
    #调用
    fn(3,5,7,9,10,a=1,b='python')
    fn(3,5)
    fn(3,5,7)
    fn(3,5,a=1,b='python')
    fn(7,9,y=5,x=3,a=1,b='python') # 错误,7和9分别赋给了x,y,又y=5、x=3,重复

     keyword-only参数

    keyword-only参数:

    • 如果在一个星号参数后,或者在一个位置可变参数后,出现的普通参数,实际上已经不是普通的参数了,而是keyword-only参数
    def fn(*args, x):
        print(x) 
        print(args)
    
    #fn(3,5)  #报错,3和5被args接受,而x是keyword-only参数,必须给出x=***
    #fn(3,5,7) #报错,3、5和7被args接受,而x是keyword-only参数,必须给出x=***
    fn(3,5,x=7) 

    args可以看做已经截获了所有的位置参数,x不使用关键字参数就不可能拿到实参

    #错误案例   **kwargs语法就应该在最后面
    def(**kwargs, x):
        print(x)
        print(kwargs)

    可以理解为kwargs为关键字参数,就算写了x=5,x也永远拿不到值,所以是语法错误

    keyword-only参数另一种形式

    def fn(*, x,y):
        print(x,y)
    fn(x=5,y=6)

    *号之后,普通形参都变成了必须给出的keyword-only参数

    可变参数和参数默认值

    def fn(*args, x=5):
        print(x)
        print(args)
    fn() # 等价于fn(x=5)  args可以没有实参给他
    fn(5) # args = 5 x = 5
    fn(x=6) # args = ()  x=6
    fn(1,2,3,x=10) # args = (1,2,3) x=10
    def fn(y, *args, x=5):
        print('x={}, y={}'.format(x, y))
        print(args) 
    fn()   # 报错,因为y没有获得实参
    fn(5)  #正确  y获取位置参数5  args为空 x=5
    fn(x=6) # 报错,因为y没有获得实参
    fn(1,2,3,x=10) #正确  y =1 args=(2,3) x=10
    fn(y=17,2,3,x=10) # 报错
    fn(1,2,y=3,x=10) #x 是 keyword-only参数  y被重复定义
    def fn(x=5, **kwargs):
        print('x={}'.format(x))
        print(kwargs) 
    fn() #x= 5 kwargs={}
    fn(5) #x = 5  kwargs={}
    fn(x=6) # x =6  kwargs={}
    fn(y=3,x=10) #x =10 kwargs={'y': 3}
    fn(3,y=10) #x =3 kwargs={'y': 10}

    函数参数

    参数规则:

    • 参数列表参数一般顺序是,普通参数-缺省参数-可变位置参数-keyword-only参数(可带缺省值)、可变关键字参数
    def fn(x, y, z=3, *arg, m=4, n, **kwargs):
        print(x,y,z,m,n) 
        print(args) 
        print(kwargs)

    注意:

    • 代码应该易读懂,而不是难为别人
    • 请按照书写习惯定义函数参数

    参数规则举例:

    def connect(host='localhost', port='3306', user='admin', password='admin', **kwargs):
        print(host, port) 
        print(user, password) 
        print(kwargs)
    connect(db='cmdb')
    connect(host='192.168.1.123', db='cmdb') 
    connect(host='192.168.1.123', db='cmdb', password='mysql')

    参数解构

    举例:

    • 加法函数
    def add(x, y): 
        return x+y
    add(4, 5)
    add((4,5)) #报错 
    add((4, 5)[0], (4, 5)[1])
    add(*(4, 5))  #或 列表 add(*[4,5])  集合 add(*{4,6})
    add(*range(1,3)) # 切忌一订要加*
    • 给函数提供实参的时候,可以在集合类型前使用*或者**,把集合类型的结构解开,提取出所有元素作为函数的实参
    • 非字典类型使用*解构成位置参数
    • 字典类型使用**解构成关键字参数
    • 提取出来的元素数目要和参数的要求匹配,也要和参数的类型匹配
    def add(x, y):
     return x+y 
    add(*(4,5)) # 解构成4,5 位置参数传进去
    d = {'x': 5, 'y': 6}  
    add(**d) #解构成x=5,y=6 关键字参数传进去
    #add(**{'a': 5, 'b': 6}) #解构成 a=5,b=6 关键字参数进去,没找到形参,报错
    add(*[4,5])  #解构成4,5 位置参数传进去
    add(*{4,6}) #解构set
    add(*{'a': 5, 'b': 6}) #一个* 解构key  不报错,输入是ab
  • 相关阅读:
    浅谈VUE中的深拷贝与浅拷贝Object.assign()方法
    echarts详细
    实现HTML调用打开本地软件文件
    el-select 中放置el-tree
    关于前端平时需要用到的方法(未完成)
    10个Vue开发技巧助力成为更好的工程师
    element ui 弹出组件的遮罩层以及多层遮罩解决办法。。
    vue样式穿透 v-deep与deep的具体使用
    # Ansible安装指南
    Linux笔记—网络管理
  • 原文地址:https://www.cnblogs.com/xzkzzz/p/11225858.html
Copyright © 2011-2022 走看看