2024-7.python函数操作

本文详细介绍了Python中的函数,包括其基本概念、内置函数与自定义函数的区别,以及如何通过函数重用代码、封装内部逻辑和处理参数传递。特别关注了默认参数、动态参数(*args和**kwargs)的使用方法和注意事项。
摘要由CSDN通过智能技术生成

函数

引言
  • 什么是函数?

    • 前面在讲解Python数据类型的时候,我们已经接触过函数了。我们说,所谓的函数其实就是Python语言中的一种工具,基于该工具可以完成不同的具体操作。函数可以分为两种,一种是内置函数,另一种是自定义函数,我们可以这么理解:
    • 内置函数:
      • 内置函数其实就是Python语言的开发者已经给我们设计好的工具,我们可以直接使用这些已经被设计好的工具或者函数完成相关的操作
    • 自定义函数:
      • 当然,Python语言的开发者们也无法将我们在不同需求中要使用的操作都设计成不同的函数或者工具,那么我们也可以向开发者那样自行设计定制我们专属功能的工具函数。
    • 案例:当你在野外露营的时候,如果想生火,如果你身上恰好带了打火机,则可以直接使用该工具自行完成生火操作,否则,你也可以自己利用现有环境下的资源自行制作取火工具。
  • 那么,为什么要使用函数呢?

    • 第一:
      • 函数的使用可以重用代码,省去重复性代码的编写,提高代码的重复利用率。如果程序中需要多次使用某种特定的功能,那么只需要编写一个合适的函数就可以了。程序可以在任何需要的地方调用该函数,并且同一个函数可以在不同的程序中调用,就像我们经常使用的print()和input()函数一样。
    • 第二:
      • 函数能封装内部实现,保护内部数据。很多时候,我们把函数看做“黑盒子”,即对应一定的输入会产生特定的结果或返回某个对象。往往函数的使用者并不是函数的编写者,函数的使用者对黑盒子的内部行为并不需要考虑,可以把精力投入到自身业务逻辑的设计而不是函数的实现细节。只有函数的设计者或者说编写者,才需要考虑函数内部实现的细节,如何暴露对外的接口,返回什么样的数据,也就是API的设计。
函数基础
  • 自定义函数的使用要经过两个过程

    • 函数的定义(制定)
    • 函数的调用(使用)
  • 函数定义语法

    • def 函数名(参数):
      	#内部代码
      	return 表达式
      
    • def myFunc(): #函数定义
          a = 1
          b = 2
          c = a + b
          print('a和b的计算结果为:',c)
      
  • 函数调用语法

    • 函数编写出来就是给人调用的。

    • 要调用一个函数,必须使用函数名后跟圆括号的方式才能调用函数。

    • 调用的同时要根据函数的定义体,提供相应个数和类型的参数,每个参数之间用逗号分隔,否则就会报错。

    • myFunc() #函数调用
      
  • 函数定义规范使用

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

  • 在定义函数的过程中,需要注意以下几点:
    • 函数代码块以def关键词开头,一个空格之后接函数标识符名称和圆括号(),再接个冒号。
    • 任何传入的参数必须放在圆括号中间。
    • 函数的第一行语句后可以选择性地使用文档字符串—用于存放函数说明。
    • 函数内容以冒号起始,并且缩进。
    • 使用return结束函数。默认返回None。
    • return语句依然在函数体内部,不能回退缩进。直到函数的所有代码写完,才回退缩进,表示函数体结束。
返回值
  • return语句
    • 当一个函数被调用结束后,该函数势必已经将一组操作执行结束了,如果在操作执行结束后,想要将一个结果返回给调用者,则就可以使用return语句实现。
#返回一个表达式
def myAdd():
    num1 = input('enter a num1:')
    num2 = input('enter a num 2:')
    num1 = int(num1)
    num2 = int(num2)
    #return就会将后面的内容返回给函数的调用
    return num1 + num2
result = myAdd()
print(result)
#不写return默认返回None
def myAdd():
    num1 = input('enter a num1:')
    num2 = input('enter a num 2:')
    num1 = int(num1)
    num2 = int(num2)
    #注意:如果函数定义中没有写return语句,则该函数默认会返回一个None
    #解释器会自动给函数定义的末尾加上一句代码:return None
result = myAdd()
print(result)
#返回多个结果
def func():
    a = 1
    b = ['bobo','100']
    c = 'jay'

    return a,b,c
#python的函数可以返回多个结果
r1,r2,r3 = func()
print(r1,r2,r3)
#return后面的代码无意义
def func():
    print('函数开始执行')
    print('函数正在执行')
    #当一个函数执行到return语句的时候,说明函数调用结束了(return语句后面的代码不会被执行)
    return 'over'
    print('函数执行结束')

result = func()
print(result)
#内部函数
def outer():
    print('正在执行outer函数')
    def inner():#内部函数
        print('正在执行inner函数')
    print('outer函数执行结束')

outer()
#执行结果:发现inner函数内部的语句没有被执行
	#正在执行outer函数
	#outer函数执行结束
#如何调用inner这个内部函数呢?
def outer():
    print('正在执行outer函数')
    def inner():
        print('正在执行inner函数')
    print('outer函数执行结束')
    return inner #outer将内部函数inner的函数名给返回了
#result == inner
result = outer() #result就是inner(内部函数的函数名),至此result也成为了内部函数的函数名
result() #调用了内部函数
#返回函数调用
def outer():
    print('正在执行outer函数')
    def inner():
        print('正在执行inner函数')
        return 123
    print('outer函数执行结束')
    return inner() #inner() == 123
    #return 123

result = outer() #outer() == 123
#result = 123
print(result)
  • return可以返回什么
    • 什么都不返回,仅仅return
    • 数字/字符串/任意数据类型: return ‘hello’
    • 一个表达式:return 1+2
    • 一个判断语句:return 100 > 99
    • 一个变量:return a
    • 一个函数调用:return func()
    • 多个返回值,以逗号分隔:return a, 1+2, “hello”
    • 简而言之,函数可以return几乎任意Python对象。
  • 注意:
    • 一旦函数执行过程遇到return语句,那么之后函数体内的所有代码都会被忽略,直接跳出函数体。那怕你现在正在一个循环内。
基本参数传递
  • 参数传递

    • 函数通常都有参数,用于将外部的实际数据传入函数内部进行处理。但是,在处理不同数据类型的参数时,会有不同的情况发生。这一切都是因为以下两点。

      • Python的函数参数传递的是实际数据的内存地址。
      • Python的数据类型分可变数据类型和不可变数据类型。
    • 例子1:不可变类型参数

      • a = 1 #a表示的是1这个数据对应内存空间的地址,a是不可变类型数据
        
        def func(b):#形参,实参给形参的传值(地址):b = a
            print('在函数内部修改之前,形参b的内存地址为:%s'%id(b))
            b = 2 #将形参b的指向发生了改变
            print('在函数内部修改之后,形参b的内存地址为:%s'%id(b))
            print('函数内部的b为:%s'%b)
            
        print('调用函数之前,变量a的内存地址为:',id(a))
        func(a) #a作为实参
        print('函数调动后,函数外部的变量a为:',a)
        
    • 例子2:上面例子说的是不可变类型参数,如果是可变类型的,比如列表呢?

      • a = [1,2,3] #可变类型的数据结构
        def func(b): #形参:b = a
            print('在函数内部修改之前,变量b的内存地址为:',id(b))
            b.append(4)
            print('在函数内部修改之后,变量b的内存地址为:',id(b))
            print('函数内部的b为:',b)
        
        print('调用函数之前,变量a的内存地址为:',id(a))
        func(a) #实参
        print('函数外部的变量a为:',a)
        
    • 调用函数时将列表对象a的地址传递给了函数内部的变量b。b.append(4)的时候,根据传进来的内存地址,找到[1,2,3]这个列表对象,在它的后面添加了4。

    • 可以看出,此时的a和b实际指向了同一个对象。为什么会这样?因为最关键的b.append(4)这句代码,它不同于“=”赋值语句,不会创建新的变量,而列表作为可变类型,具有append方法,这个方法只是对列表的一种调用而已。因此,a和b实际还是同一个对象。

参数类型
  • 绝大多数函数在定义的时候需要接收一定数量的参数,然后根据实际调用时提供的参数的不同,输出不同的结果。注意将函数内部的参数名字,定义得和外部变量的名字一样是一种不好的习惯,它容易混淆思维,甚至发生错误。

  • 参数的两种称谓

    • 形参
      • 函数定义时,制定的参数叫做形参
    • 实参
      • 函数调用时,传递的参数叫做实参
    • 而我们通常讨论的参数,指的都是形参
  • 参数的不同种类

    • 定义函数时,参数的名字和位置确定下来,函数的接口就固定了。对于函数的调用者来说,只需要知道如何传递正确的参数,以及函数将返回什么样的值就够了,函数内部的复杂逻辑被封装起来,调用者无需了解。Python函数的参数定义灵活度非常大。除了正常定义的位置参数外,还可以使用:
      • 位置参数
      • 默认参数
      • 动态参数
位置参数
  • 也叫必传参数或者顺序参数,是最重要的、也是必须在调用函数时明确提供的参数!位置参数必须按先后顺序,一一对应,个数不多不少的传递!

  • def add(a,b,c):
        return a+b+c
    
    x = y = 5
    r1 = add(x,y,x)
    r2 = add(4,5,6)
    print(r1,r2)
    
  • 上面例子中的a,b,c就是位置参数,我们在使用add(4, 5, 6)调用时,就是将4的地址传给a,5的传给b,6的传给c的一一对应传递。

    • 类似add(4, 5, 6, 7)、add(4)这种“画蛇添足”、“缺胳膊少腿”和“嫁错郎”类型的调用都是错误的。
  • 注意: Python在做函数参数传递的时候不会对数据类型进行检查,理论上你传什么类型都可以!

  • def add(a,b,c):
        return a+b+c
    
    add(1,2,'haha')
    
  • 但是,上面的add函数,如果你传递了一个字符串和两个数字,结果是弹出异常,因为字符串无法和数字相加。

    • 这就是Python的弱数据类型和动态语言的特点。在简单、方便的时候,需要你自己去实现数据类型检查。
默认参数
  • 在函数定义时,如果给某个参数提供一个默认值,这个参数就变成了默认参数,不再是位置参数了。在调用函数的时候,我们可以给默认参数传递一个自定义的值,也可以使用默认值。

  • def power(x,n=2): #x是位置参数,n是默认参数
        return x * n
    
    power(10)
    power(10,4)
    
  • 上面例子中的n就是个默认参数。默认参数可以简化函数的调用,在为最常用的情况提供简便调用的同时,还可以在特殊情况时传递新的值。

  • 默认参数的注意事项:

    • 默认参数必须在位置参数后面!
    • 使用参数名传递参数
    • 使用不可变的数据类型作为默认值!
  • 默认参数必须在位置参数后面

    • 如果你违反了这点,在语法层面直接是通不过的。
    • 当有多个默认参数的时候,通常将更常用的放在前面,变化较少的放后面。
    • 在调用函数的时候,尽量给实际参数提供默认参数名。
  • def student(name,sex,age,classroom='101',tel='1323333333',address='...'):
        pass
    
    student('jack','male',17) #可以
    student('tom','male',18,'102','666','Beijing') #可以
    student('marry','female',18,'102',address='SH') #可以
    
    
  • student('mary','female',address='Bj',18)   #  这是错误的参数传递方式
    
  • 使用参数名传递参数

    • 通常我们在调用函数时,位置参数都是按顺序先后传入,而且必须在默认参数前面。但如果在位置参数传递时,给实参指定位置参数的参数名,那么位置参数也可以不按顺序调用
  • student(name='mary',sex='female',address='Bj',age=18)   #  这是错误的参数传递方式
    
  • 默认参数尽量指向不变的对象

    • 首先先来看一道面试真题:输出程序运行后的结果

    • def func(a=[]): #将可变类型数据作为了默认参数的默认值
          a.append('A')
          return a
      
      print(func()) 
      print(func()) 
      print(func()) 
      
    • 解释:

      • 因为Python函数体在被读入内存的时候,默认参数a指向的空列表对象就会被创建,并放在内存里了。因为默认参数a本身也是一个变量,保存了指向对象[]的地址。每次调用该函数,往a指向的列表里添加一个A。a没有变,始终保存的是指向列表的地址,变的是列表内的数据!
动态参数
  • 顾名思义,动态参数就是传入的参数的个数是动态的,可以是1个、2个到任意个,还可以是0个。在不需要的时候,你完全可以忽略动态函数,不用给它传递任何值。

  • Python的动态参数有两种,分别是:

    • *args
    • **kwargs
    • 这里面的关键是一个和两个星号的区别,而不是args和kwargs在名字上的区别
  • 注意:

    • 动态参数,必须放在所有的位置参数和默认参数后面!
  • *args

  • 一个星号表示接收任意个参数。调用时,会将实际参数打包成一个元组传入形式参数。如果参数是个列表,会将整个列表当做一个参数传入。例如:

  • def func(*args):#动态参数
        print(args)
        for param in args:
            print(param) #param就是接收到的每一个参数
    func(1,2,3)
    
    • 问题:

      • 通过循环args,我们可以获得传递的每个参数。但是li这个列表,我们本意是让它内部的1,2,3分别当做参数传递进去,但实际情况是列表本身被当做一个整体给传递进去了。怎么办呢?

      • def func(*args):
            print(args)
            for param in args:
                print(param)
        
        li = [1,2,3]
        func(li)
        
        • 解决:

          • 使用一个星号!调用函数,传递实参时,在列表前面添加一个星号就可以达到目的了。实际情况是,不光列表,任何序列类型数据对象,比如字符串、元组都可以通过这种方式将内部元素逐一作为参数,传递给函数。而字典,则会将所有的key逐一传递进去。

          • def func(*args):
                print(args)
                for param in args:
                    print(param)
            
            li = [1,2,3]
            func(*li)
            
  • **kwargs

    • 两个星表示接受键值对的动态参数,数量任意。调用的时候会将实际参数打包成字典。例如:

    • def func(**kwargs):
          print(kwargs)
      
      func(k1=1,age=2,k3=2)
      
  • 问题:

    • 而如果我们这样传递一个字典dic呢?我们希望字典内的键值对能够像上面一样被逐一传入。

    •   def func(**kwargs):
            print(kwargs)
        
        dic = {
            'k1':1,
            'age':20,
            'k3':3
        }
        func(d = dic)
      
      • 上述程序实际结果却是弹出错误,为什么?

      • 解释:

        • 因为这时候,我们其实是把dic当做一个位置参数传递给了func函数。而func函数并不接收任何位置函数。那怎么办呢?使用两个星号!
      • def func(**kwargs):
            print(kwargs)
        
        dic = {
            'k1':1,
            'age':20,
            'k3':3
        }
        func(**dic)
        
      • 有了前面一个星号的基础,这里我们应该很好理解了。两个星号能将字典内部的键值对逐一传入**kwargs。

    • 万能参数

      • 当*args和kwargs组合起来使用,理论上能接受任何形式和任意数量的参数,在很多代码中我们都能见到这种定义方式。需要注意的是,*args必须出现在kwargs之前。
  • 4
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值