08--函数基本使用与参数详解

一、函数介绍

1.什么是函数

函数就是相当于具备某一功能的工具

2.为何要用函数

  • 程序的组织结构不清晰,可读性差
  • 代码冗余
  • 可维护性、拓展性差

3.如何用函数

1.先定义:

    三种定义方式

2.后调用

    三种调用方式

3.返回值

    三种返回值的形式

二、函数的基本使用

1.定义函数

# 语法:
def 函数名(参数1,参数2,...):
    """文档描述"""
    函数体
    return# tips:函数名尽量是描述该功能的动词形式(英文)

# 定义函数时,发生的事情:
  1.申请内存空间,保存函数体代码
  2.将上诉内存地址绑定给函数名
  3.定义函数不会执行函数体代码,但会检测函数体代码语法

# 形式一:无参函数
def func1():
    print('哈哈哈哈')

# 形式二:有参函数
def func2(x, y):
    print(x, y)

# 形式三:空函数
def func3():
    pass
    # 或者 ...

2.调用函数

# 调用函数时,发生的事情:
  1.通过函数名找到其函数的内存地址
  2.然后加括号(类的实例化)就是触发函数体代码的执行

# 形式一:语句的形式:只加括号调用
func1()

# 形式二:表达式的形式
res = func2(2, 3)  # 赋值表达式
print(res)
res = func2(2, 3) * 10  # 运算表达式
print(res)

# 形式三:函数调用可以当做参数
res = func2(func2(2, 3), 5)
print(res)

3.函数返回值

# return 是函数结束的标志,
  即函数体代码一旦运行到return就会立刻终止函数的运行,
  并且将return后的值作为返回结果返回。

# 形式一:返回None
# 无return 或者  return 后无值
def func():
    return
    # pass

print(func())  # None

# 形式二:返回一个值
def func():
    return 20

print(func())  # 20

# 形式三:返回多个值(用逗号分隔开多个值,返回成元祖)
def func():
    return 20, 'abc', [5,6]

print(func())  # (20, 'abc', [5, 6])

三、函数参数详解

1.形参与实参的介绍

# 形参:在定义函数阶段定义的参数称之为形式参数,简称形参,相当于变量名

# 实参:在调用函数阶段传入的值称之为实际参数,简称实参,相当于变量值

# 形参与实参的关系:
  1.在调用阶段,实参(变量值)会绑定给形参(变量名)
  2.这种绑定关系只能在函数体内使用
  3.实参与形参的绑定关系在函数调用时生效,函数调用结束后解除绑定关系

2.形参与实参的具体使用

2.1位置参数-形实

# 位置参数:按照从左到右的顺序依次定义的参数称之为位置参数

位置形参:在函数定义阶段,按照从左到右的顺序直接定义的"变量名"
    特点:必须被传值,多一个不行少一个也不行
    
def func(x,y):
    print(x,y)
    
func(1,2,3)  # 报错
func(1,)  # 报错

位置实参:在函数调用阶段, 按照从左到有的顺序依次传入的值
    特点:按照顺序与形参一一对应
    
func(1,2)
func(2,1)

2.2关键字参数 -实

关键字实参:在函数调用阶段,按照key=value的形式传入的值
    特点:指名道姓给某个形参传值,可以完全不参照顺序
    
def func(x,y):
    print(x,y)

func(y=2,x=1)  # 关键字实参
func(1,2)  # 位置实参

关键字实参与位置实参的混合使用,强调:
# 1.位置实参必须放在关键字实参前
func(1,y=2)
func(y=2,1)  # 报错

# 2.不能能为同一个形参重复传值
func(1,y=2,x=3)
 func(1,2,x=3,y=4)

2.3默认参数-形

默认形参:在定义函数阶段,就已经被赋值的形参,称之为默认参数
      特点:在定义阶段就已经被赋值,意味着在调用阶段可以不用为其赋值
    
def func(x,y=3):
    print(x,y)

func(x=1)
func(x=1,y=44444)


def register(name,age,gender='男'):
    print(name,age,gender)

register('三炮',18)
register('二炮',19)
register('大炮',19)
register('没炮',19,'女')


# 位置形参与默认形参混用,强调:
# 1.位置形参必须在默认形参的前边
def func(y=2,x):  # 报错
    pass 

# 2.默认参数的值是在函数定义阶段被赋值的,准确地说被赋予的是值的内存地址
python的函数 参数传递是值传递还是引用传递?
  既不是值传递,也不是引用传递,而是赋值传递,或者是叫对象的引用传递。
    针对可变类型,是对象的引用传递,传递的是参数对象的内存地址
    针对不可变类型,是对象被copy一份,传递的是参数对象值拷贝后的内存地址


# 示范1:
m=2
def func(x,y=m):  # y=>2的内存地址
    print(x,y)
m=3333333333333333333
func(1)  # 1, 2

# 示范2:
m = [111111, ]
def func(x, y=m):  # y=>[111111, ]的内存地址
   print(x, y)
m.append(3333333)
func(1)  # 1, [111111, 3333333]

# 3.虽然默认值可以被指定为任意数据类型,但是不推荐使用可变类型
  函数最理想的状态:函数的调用只跟函数本身有关系,不外界代码的影响
          
m = [111111, ]
def func(x, y=m):
   print(x, y)
m.append(3333333)
func(1)

# 若是在有需求,需要定为可变类型,可以如下操作:默认为None,加判断,初始化
def func(x,y,z,l=None):
    if l is None:
        l=[]
    l.append(x)
    l.append(y)
    l.append(z)
    print(l)

func(1,2,3)  # [1, 2, 3]
func(4,5,6)  # [4, 5, 6]

new_l=[111,222]
func(1,2,3,new_l)  # [111, 222, 1, 2, 3]

2.4 可变长度的参数(*与**的用法)

# 可变长度指的是在调用函数时,传入的值(实参)的个数不固定
  而实参是用来为形参赋值的,所以对应着,针对溢出的实参必须有对应的形参来接收
2.4.1 可变长度的位置参数
# I:*形参名:用来接收溢出的位置实参,溢出的位置实参会被*保存成元组的格式,然后赋值紧跟其后的形参名
     *后跟的可以是任意名字,但是约定俗成应该是args

def func(x,y,*z):  # z =(3,4,5,6)
    print(x,y,z)

func(1,2,3,4,5,6)  # 1,2,(3,4,5,6)

def my_sum(*args):
    res=0
    for item in args:
        res+=item
    return res

res=my_sum(1,2,3,4,)
print(res)  # 10


# II: *可以用在实参中,实参中带*,先*后的值打散成位置实参
def func(x,y,z):
    print(x,y,z)

func(*[11,22,33])  # func(11,22,33)
func(*[11,22])  # func(11,22)

l=[11,22,33]
func(*l)

# III: 形参与实参中都带*
def func(x,y,*args):  # args=(3,4,5,6)
    print(x,y,args)

func(1,2,[3,4,5,6])
func(1,2,*[3,4,5,6])  # func(1,2,3,4,5,6)
func(*'hello')  # func('h','e','l','l','o')
2.4.2 可变长度的关键字参数
# I:**形参名:用来接收溢出的关键字实参,**会将溢出的关键字实参保存成字典格式,然后赋值给紧跟其后的形参名
     **后跟的可以是任意名字,但是约定俗成应该是kwargs
    
def func(x,y,**kwargs):
    print(x,y,kwargs)

func(1,y=2,a=1,b=2,c=3)

# II: **可以用在实参中(**后跟的只能是字典),实参中带**,先**后的值打散成关键字实参
def func(x,y,z):
    print(x,y,z)

func(*{'x':1,'y':2,'z':3})  # func('x','y','z')
func(**{'x':1,'y':2,'z':3})  # func(x=1,y=2,z=3)

func(**{'x':1,'y':2,})  # func(x=1,y=2)  # 错误
func(**{'x':1,'a':2,'z':3})  # func(x=1,a=2,z=3)


# III: 形参与实参中都带**
def func(x,y,**kwargs):
    print(x,y,kwargs)

func(y=222,x=111,a=333,b=444)
func(**{'y':222,'x':111,'a':333,'b':4444})
2.4.3 可变形参的混用使用:*与**
# 混用*与**:*args必须在**kwargs之前(就是位置参数必须在关键字参数之前)
def func(x,*args,**kwargs):
    print(args)
    print(kwargs)

func(1,2,3,4,5,6,7,8,x=1,y=2,z=3)

def index(x,y,z):
    print('index=>>> ',x,y,z)

def wrapper(*args,**kwargs):  # args=(1,) kwargs={'z':3,'y':2}
    index(*args,**kwargs)
    # index(*(1,),**{'z':3,'y':2})
    # index(1,z=3,y=2)

wrapper(1,z=3,y=2) # 为wrapper传递的参数是给index用的
# 原格式---》汇总-----》打回原形

2.5 命名关键字参数(了解)

# 命名关键字参数:在定义函数时,*后定义的其他参数,如下所示,称之为命名关键字参数

特点:
# 1.命名关键字实参必须按照key=value的形式为其传值
def func(x,y,*,a,b):  # 其中,a和b称之为命名关键字参数
    print(x,y)
    print(a,b)

func(1,2,b=222,a=111)

# 2.命名关键字形参可以为其添加默认值
  强调:命令关键字参数添加默认值时,没有顺序要求(并不是默认值参数)
def func(x,y,*,a=11111,b):  # 不会报错:默认参数必须在位置参数之后
    print(x,y)
    print(a,b)

func(1,2,b=22222)

2.6 组合使用(了解)

# 1.形参混用的顺序:位置形参,默认形参,*args,命名关键字形参,**kwargs
def func(x,y=111,*args,z,**kwargs):
    print(x)
    print(y)
    print(args)
    print(z)
    print(kwargs)

    
# 2.实参混用的顺序:(本质就是位置参数在 关键字参数 之前)
def func(x,y,z,a,b,c):
    print(x)
    print(y)
    print(z)
    print(a)
    print(b)
    print(c)

func(111,y=222,*[333,444],**{'b':555,'c':666})
func(111,y=222,333,444,b=555,c=666)  # 报错

func(111,*[333,444],a=222,**{'b':555,'c':666})
func(111,333,444,a=222,b=555,c=66)

func(111,*[333,444],**{'b':555,'c':666},a=222,)
func(111,3333,4444,b=555,c=666,a=222)

# 实际使用时,实参形式:
func(1)
func(x=1)
func(1,x=1)
func(*'hello')
func(**{})
func(*'hell',**{})
posted @   Edmond辉仔  阅读(133)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?
点击右上角即可分享
微信分享提示