(3)什么是函数(函数的定义、形参、实参、默认形参、可变长函数args \ kwargs,私有地址)
什么是函数
函数是指将一组语句的集合通过一个名字(函数名)封装起来,想要执行这个函数,只需调用其函数名即可
1、减少重复代码
2、使程序变的可扩展
3、使程序变得易维护
定义函数的语法
形参
主要的作用就是接受实参参数的传递,并且被语法调用 #形参只有在被调用时才内存单元,在调用解释时,即可释放所分配的内存单元。因此,形参只在函数内部有效。函数调用结束返回主调用函数后则不能再使用该形参变量
实参
主要的作用就是讲参数传递给形参 #实参可以是常量、变量、表达式、函数等,无论实参是何种类型的量,在进行函数调用时,他们都必须有确定的值,以便把这些值传递给形参。因此应预先用赋值,输入等办法使参数获得确定值
定义函数的形参,并且将实参传递给形参再调用
def calc(x,y): #def关键词,calc函数名,括号里的是形参
res = x**y
return res
c = calc(a,b) #括号里的是实参,传递给x和y
print(c)
位置形参:指的是在函数定义阶段按照从左到右的顺序依次定义的形参
特点: 必须被传值,多一个不行,少一个也不行
位置实参: 指的是在函数调用阶段按照从左到右的顺序依次传入的值
特点: 与形参一一对应
形参也可以是默认值
def calc(x,y=2) #这样在函数调用时如果形参不指定参数,则在调用时就是默认值,如果指定,则是指定值
默认参数(默认形参): 在函数定义阶段,就已经为形参赋值
特点: 在定义阶段就已经有值,意味着在调用阶段可以不用传值
注意: 可以混用位置形参与默认形参,但是
1、位置形参必须放在默认形参前面
2、默认形参的值只在函数定义阶段赋值一次,定义之后的任何改动都无法影响默认形参的值
3、默认形参的值通常应该是不可变类型
关键字参数
一般情况下,给函数传参数时要按顺序,不想按顺序就可以用关键字参数,只需指定参数名即可(指定了参数名的参数就叫关键参数),关键字参数必须放在位置参数(以位置顺序确定对应关系的参数)之后
关键字参数(单指关键字实参): 在函数调用阶段,按照key=value形式为指定的形参赋值
特点: 可以完全打乱传值的顺序,但仍然能指名道姓地为指定的形参赋值
注意:可以混用位置实参与关键字实参,但是
1、不能为同一个形参重复赋值
2、位置实参必须放在关键字实参的前面
PS:位置参数就是形参没有指定任何值,所以函数调用时候是按照位置传递的,关键字参数就是指定形参的默认值,所以必须放在位置参数的后面,不然在函数传递时候按照位置传递,会出现错误
例:def stu_register(name,age,course='Py',counter='CN'): #def是函数定义关键字,结尾必须用:结束,这是强制语法
print("姓名:", name)
print("年龄:", age)
print("课程:" ,course)
print("国籍:", counter)
stu-register(‘Karl’,course='Py',age=22,counter='JP') #这里Karl没有用赋值去指定赋值给哪一个形参,所以必须对应形参的位置,而后面的实参都有赋值给指定的形参,所以可以不用按位置写
可变长参数
指的是在调用函数时传入的值(实参)个数不固定,而实参无非两种,一种位置实参
另外一种就是关键字实参,针对这两种形式实参的个数不固定,对应着形参也应该有两种解决方案来分别应对溢出位置实参(*)与关键字实参(**)
若在函数的定义时不确定用户想传入多少个参数,就可以使用非固定参数(*args和*kwargs)
*args会把多传入的参数变成一个元组形式
def stu(name,age,*args)
print(name,age,args)
stu('Karl',22)
输出的结果就是Karl 22() #输出结果后面有一个(),元组符号,这个就是args,因为没有传入任何值,所以就是一个空元组
**kwargs 会把多传入的参数变成一个dict的形式(字典形式)
def stu(name,age,*args,**kwargs)
print(name,age,args,kwargs)
stu('Karl',22,'cn','Py',sex='male')
输出结果就是Karl 22 ('cn', 'Py') {'sex': 'male'}
PS:*args前面是一个*,**kwargs前面是两个*,args传入时候单个字符串即可,而kwargs传入的时候必须以键值(key=value)传入(即字典形式传入)
PS:*args和**kwargs其实就是前面的*起作用,后面的名称只是约定俗成的名字(可以随便写),这样所有开发人员看到这个就知道一个是传入元祖,一个是将值变成关键字传入
PS:在实参中出现*: 但凡是在实参中出现*,都将*后的值打算成位置实参,然后再进行赋值
PS:在实参中出现**: 但凡是在实参中出现**,都将**后的值打算成关键字实参,然后再进行赋值
PS:在形参中出现**: **会将溢出的关键实参存成字典的形式,然后赋值紧跟其后的形参
PS:在形参中出现*: *会将溢出的位置实参存成元组的形式,然后赋值紧跟其后的形参
例:利用可变长参数在函数体内调用另一个函数的时候,将实参传递给被调用的函数
def index(a,b,c): #先定义一个有形参的函数
print('from index',a,b,c) #执行打印形参
def wrapper(*args,**kwargs): #这里再定义一个函数带有可变长参数
index(*args,**kwargs) #直接调用上面的函数并且形参定义成可变长参数
wrapper(1,c=3,b=2) #在调用的时候将wrpper实参传递给形参,然后在传递给调用的index的形参
PS:这里由于wapper函数执行的就是调用index
列:利用位置参数传递
def index(a,b,c):
print('from index',a,b,c) #这个函数执行的操作
def wrapper(name,age,sex): #这个带有形参的函数
index(name,age,sex) #这里调用函数时候的形参必须是这个函数块的形参
wrapper('karl','123','male') #这里按照位置实参传递
PS:这是一个相互传递的过程,调用时候将位置实参传递给形参,然后再传递给调用的函数
命名关键字形参:
在定义函数阶段,但凡是在*与**之间的参数称之为命名关键字形参
特点: 必须被传值, 并且在传值时必须按照key=value的形式传值
例:
def func(x,*args,m=111,n,**kwargs):
print(x)
print(m)
print(n)
print(kwargs)
print(args)
func(1,1,2,3,4,n=3,a=5,b=6)
PS:参数的顺序,先写位置,再写默认(*前面的才是默认参数),然后写*args,再跟关键字参数(*后面的是带有默认值(无默认值)的关键字参数),最后跟**kwargs
PS:位置实参必须按照位置传递,关键字和*和**之间则无需按照位置,位置后面如果没有传递默认参数的值,也没有传递关键字参数,则会将所有溢出字符传递给*,如果是k=v的值全部溢出的传递给**
返回值(return)
函数外部代码想要获取函数的执行结果,就可以在函数里用return语句把结果返回
def stu_register(name,age,course='Py',counter='CN'):
print("姓名:" ,name)
print("年龄:" ,age)
print("课程:" ,course)
print("国籍:" ,counter)
if age > 22: #这里输出一个判定条件,用return返回后面的结果
return False
else:
return True
s_reg = stu_register(‘Karl’,course='Py',age=22,counter='JP')
if s_reg:
print('注册成功')
else:
print('年龄太大‘’)
PS:函数在执行过程中只要遇到return语句,就会停止执行并返回结果,return语句就是函数运行结束的一个语句,如果未在函数中指定return,那么这个函数的返回值为None
全局变量和局部变量
1、在函数中定义的变量就是局部变量,在程序的一开始定义的变量称为全局变量
2、全局变量的作用域是整个程序,局部变量作用域就是定义该变量的函数
3、当全局变量与局部变量同名时,在定义局部变量的函数内,局部变量起作用,在其他地方则是全局变量起作用
name = 'Alex' #这个是全局变量
def change_name(name):
print('beore change:',name)
name ='Karl' #这个是局部变量,只给下面的调用
print('after change:',name)
change_name(name)
作用域
作用域(scope),通常来说,一段 程序代码中所用到的名字并不总是有效/可用的,而限定这个民资的可用性的代码范围就是这个名字的作用域
global 声明全局变量函数
def change_name(): #这里没有定义形参
global name #用global 函数声明name就是全局变量
name ='Karl'
print('after change:',name)
change_name() #用为用了global全局函数声明变量name,所以这里也可以打印name
嵌套函数
name = 'Alex'
def change_name():
name = 'Alex2'
def change_name2():
name = 'Alex3'
print('第三层:',name)
change_name2()
print('第二层:',name)
change_name()
print('最外层:',name)
PS:函数嵌套的调用也是需要按照层级来依次调用的,如果在第二层调用第三层则会报错,所以第二层只能调用第二层
匿名函数(lambda)
匿名函数就是不需要显式的指定函数名
通常函数的调用
def calc(x,y)
return x**y
print(clac(2,5))
采用匿名函数
calc = lambda x,y:x**y
print(calc(2,5))
匿名函数的主要用法是和其他函数搭配使用
res = map(lambda x:x**2,[1,5,7,4,8]) #这里直接定义了一个变量的计算过程,用lambda函数定义X,冒号后面传递一个计算的方式,最后用列表定义x的值
for i in res:
print(i)
闭包函数
什么是闭包函数=>函数嵌套+函数对象+名称空间与作用域
闭:定义在函数内的函数
包:该内部函数需要访问一个名字,该名字属于外层函数作用域的(强调:不是全局作用域)
为何要用闭包函数?
为了实现装饰器
如何用闭包函数
def outter(name): #定义函数
x = ‘456’
print(x) #
def inner(): #函数内嵌套一个函数
print(‘成功调取这一层值’)
return inner #返回函数,不能加括号,加了括号就是调取函数的功能,不加括号就是返回一个函数的内存空间地址
f = outter() #outter内最后将inner的内存地址做返回结果,所以outter函数的只会返回inner的内存空间地址,这里outter就是inner的内存空间地址并且赋值给f
f() #这里调用f就获得了inner局部变量的值
PS:首先定义一个函数,然后在下层定义一个函数,最后返回下层函数的内存空间地址给全局函数,然后将全局函数的赋值给一个变量
PS:由于全局函数outter下层有一个打印值,所以在打印f的时候也会打印出来
高阶函数
变量可以指向函数,函数的参数能接受变量,那么一个函数就可以接收另一个函数作为参数,这种函数就称之为高阶函数
只要满足以下任意一个条件,即是高阶函数
1、接受一个或多个函数作为输入
2、return返回另一个函数
def add(x,y,f):
return f(x) + f(y)
res = add(3,-6,abs)
print(res)
递归函数
在函数内部,可以调用其他函数。如果一个函数在内部调用自身本身,这个函数就是递归函数
特性:
1、必须有一个明确的结束条件
2、每次进入更深一层的递归时,问题规模相比上次递归都应有所减少
3、递归效率不高,栈就会增加一层栈帧,每当函数返回,栈就会减一层栈。由于栈的大小不是无限的,所以,递归调用的次数过多,会导致栈溢出
def calc(n):
print(n)
if int(n/2) == 0:
return n
return calc(int(n/2)
calc(10)
私有地址 # 只能在函数的内部获取
定义方式 __funcname # 私有地址的命名方式就是在函数名前加__
在函数的内部调用方式
class FTPHandler():
def handle(self):
print('1')
if hasattr(self,"_FTPHandler__get"): # hasattr判断如果有这个函数名,调用私有地址的方式是_父类名+私有地址的函数名将名字拼成私有地址的格式
print('yes')
else:
print('没有找到')
def __get(self):
'''私有地址函数'''
print('123')
ftp = FTPHandler()
ftp.handle()