一、函数介绍
1、函数是什么?
函数一词来源于数学,编程中的"函数"与数学中的函数有很大的不同。
(BASIC中叫subroutine,C中叫function,java中叫method)
定义:函数是指将一组语句的集合通过一个名字(函数名)封装起来,想执行这个函数,只需调用其函数名即可。
2、为什么要使用函数?
(1)减少重复代码:否则遇到重复的功能只能重复编写实现代码,代码冗余
(2)使程序变得可扩展:否则功能需要扩展时,需要找出所有实现该功能的地方修改之,无法统一管理且维护难度极大
(3)使得程序变得易维护:否则代码的组织结构不清晰,可读性差
3、函数分类
(1)内置函数:针对一些简单的功能,python解释器已经为我们定义好了的函数即内置函数。对于内置函数,我们可以拿来就用而无需事先定义,如len(),sum(),max()。
(2)自定义函数:很明显内置函数所能提供的功能是有限的,这就需要我们自己根据需求,事先定制好我们自己的函数来实现某种功能,以后,在遇到应用场景时,调用自定义的函数即可
二、函数的参数
参数的作用:可以让函数更灵活,不只能做死动作,还可以根据调用时传参的不同来决定函数内部的执行流程。
# 无参函数
def sayhi(): # 函数名(小写即可)
print("Hello,I'm nobody!")
sayhi() # 调用函数,函数名指向上述代码指向的内存位置,加上括号才是执行代码
print(sayhi) # 函数sayhi指向的内存地址
# 单个参数的函数
def sayhi(name):
print("hello",name)
print("my name is black girl....", name)
sayhi("tracy")
# 多个参数函数
def calc(x,y): # 定义算数函数
res = x**y
return res # 返回函数执行结果
a,b = 5,8
c = calc(a,b) # 结果赋值给变量c
print(c)
calc(2,10) # 直接写入参数
1、形参
只有在被调用时才分配内存单元,在调用结束时,即刻释放所分配的内存单元。因此,形参只在函数内部有效。函数调用结束返回主调用函数后则不能再使用该形参变量
2、实参
可以是常量、变量、表达式、函数等,无论实参是何种类型的量,在进行函数调用时,它们必须有确定的值,以便把这些值传送给形参。因此应预先用赋值,输入等办法使参数获得确定值
def calc(x,y): # 形参
res = x**y
return res
a,b = 3,5
c = calc(a,b) # 实参
print(c)
d = calc(2,3) # 实参
3、默认参数
def stu_register(name,age,country,course):
print("注册学生信息".center(50,'-'))
print("姓名:",name)
print("age:" ,age)
print("国籍",country)
print("课程",course)
stu_register("山炮",22,"CN","python_devops")
stu_register("丰收",23,"CN","linux")
'''
由于很多人国籍都是中国,可以将country设置为默认参数
默认参数:
'''
def stu_register(name,age,course,country="CN"): # 非默认参数不能跟在默认参数后面
print("registration info".center(50,'-'))
print(name,age,country,course)
stu_register("jack",22,'c++') # 实参和形参按顺序一一对应
stu_register("rain",32,'dance','Korean')
stu_register("Alfa",21,'python')
4、关键字参数
应用场景:正常情况下,给函数传参数要按顺序,不想按顺序可以用关键参数
定义:指定了参数名的参数就叫关键参数
关键参数必须放在位置参数(以位置顺序确定对应关系的参数)之后
def stu_register(name,age,course,country='CN'):
print("注册学生信息".center(50,'-'))
print("姓名:",name)
print("age:" ,age)
print("国籍",country)
print("课程",course)
stu_register("绮珊",course='Python',age=22,country='JP') # 后三个均为为关键参数置于name位置参数之后
# stu_register("杉树",course='Python',22,country='JP') # 22为位置参数不能放在关键参数course之后
# stu_register("曼玉",22,age=25,country='JP') # age获得多个赋
5、非固定参数(动态参数)
形式参数中出现*,传递的参数就可以不再是固定个数,会将传过来的所有参数打包为元组。
def send_alert(msg,*users):
for u in users:
print('报警发送给',u)
# 方式一:
# 报警,十个运维人员
send_alert('注意系统资源别浪了','Alex','jack','tracy','wood')
# 方式二:
# send_alert('注意内存紧张',['alex','jack','tracy','wood']) # 传入的参数是数据
send_alert('注意内存紧张',*['alex','jack','tracy','wood']) # 传入的参数是数组内的元素
# 传入的参数由(['alex','jack','tracy','wood']) ————>('alex','jack','tracy','wood')
如果动态参数后还有参数
# 方法一:
def send_redalert(msg,*users,age):
for u in users:
print('CPU紧张',u)
# send_redalert("alex","rain",22) # 22也会传递给*users
send_redalert("alex","rain",age=22)
# 方法二:
def func(name,*args,**kwargs):
print(name,args,kwargs)
func('Alex',22,'tesla','500w',addr='湖北',num=123332313)
# Alex (22, 'tesla', '500w') {'addr': '湖北', 'num': 123332313}
d = {'degree':'primary school'}
func('Peiqi',**d)
- *args代表位置参数,它会接收任意多个参数并把这些参数作为元组传递给函数。
- **kwargs代表的关键字参数,允许你使用没有事先定义的参数名。
- 位置参数一定要放在关键字参数的前面。
优点:使用*args和**kwargs可以非常方便的定义函数,同时可以加强扩展性,以便日后的代码维护。
三、函数的返回值
返回值:函数外部的代码要想获取函数的执行结果,可以在函数里用return语句把结果返回。
注意:
1、函数在执行过程中只要遇到return语句,就会停止执行并返回结果,可以理解return语句代表着函数的结束
2、如果未在函数中指定return,那这个函数的返回值为None
def stu_register(name,age,course='PY',country='CN'):
print("注册学生信息".center(50,'-'))
print("姓名:", name)
print("age:", age)
print("国籍", country)
print("课程", course)
if age > 22:
return False
else:
return True
registration_status = stu_register('阿斯顿',22,course="全栈开发",country='JP')
if registration_status:
print("注册成功".center(50,'-'))
else:
print("too old to be a student.")
'''
执行到return语句后,停止执行函数并返回结果
'''
def stu_register(name,age,course):
print(name,age,course)
#
# if age > 22:
# return 'sdfsf' # 返回值可以是任意值
# else:
# return True
#
# return None # 到return语句后,停止执行函数并返回结果
# print('hahah')
# return 1
return [name,age]
status = stu_register('Peiqi',29,'安保')
print(status)
四、全局和局部变量
局部变量:函数内定义的变量,只能在局部生效
全局变量:定义在函数外部一级代码的变量,整个程序可用(由上到下可用)
在函数内部可以引用全局变量。
变量查找顺序:如果全局和局部都有一个变量,函数查找变量的顺序是由内而外的。两个函数间互不可见
name = "Black girl" # 全局变量
def change_name():
# global name # (不建议使用global)在函数内修改全局变量,不能放在函数内局部变量后面
name = "黑色的姑娘" # 局部变量
print("在",name,"里面...",id(name))
change_name()
print(name,id(name)) # 与函数内的变量内容完全不同
'''
输出结果:
在 黑色的姑娘 里面... 4302993904
Black girl 4316351984
'''
函数内可以修改字典、列表、集合、对象、类、元组内的列表
names = ['Alex','Black Girl','Peiqi']
def change_name():
names = ['Alex']
del names[2] # names整体的内存地址不能修改只能引用,但其内部的元素是可以修改的
names[1] = "黑姑娘"
print(names)
change_name()
print(names)
'''
输出结果:
['Alex', '黑姑娘']
['Alex', '黑姑娘']
'''
- 在函数中定义的变量称为局部变量,在程序的一开始定义的变量称为全局变量。
- 全局变量作用域是整个程序,局部变量作用域是定义该变量的函数。
- 当全局变量与局部变量同名时,在定义局部变量的函数内,局部变量起作用;在其它地方全局变量起作用。
五、作用域(scope)
作用域定义:一段程序代码中所用到的名字并不总是有效/可用的,而限定这个名字的可用性的代码范围就是这个名字的作用域。
- python中一个函数就是一个作用域,局部变量放置在其作用域中
- C# Java中作用域{}
- 代码定义完成后,作用域已经完成,作用域链向上查找
age = 18
def func1():
age = 73
def func2():
age = 84
print(age)
return 666
val = func1()
print(val)
'''
输出:666
'''
# 函数名可以当作返回值
age = 18
def func1():
age = 73
def func2():...
return func2 # 返回一个函数名# val = func1()
print(val)
'''
输出:<function func1.<locals>.func2 at 0x101462598>
'''
# 代码写完之后作用域已经生成,不管函数名传到哪里,只要执行都回回定义的地方往上找
age = 18
def func1():
age = 73
def func2():
print(age)
return func2 # 返回一个函数名不带括号
val = func1()
val()
'''
输出结果:73
'''
六、嵌套函数
嵌套函数,就是指在某些情况下,您可能需要将某函数作为另一函数的参数使用。
1、函数定义完成之后,没有通过名字调用,内部代码永远不会执行
def func1():
print('alex')
def func2():
print('eric')
func1()
'''
输出:alex
'''
def func1():
print('alex')
def func2():
print('eric')
func2()
func1()
'''
输出:
alex
eric
'''
总结:
- 函数内部可以再次定义函数
- 执行函数需要被调用
2、嵌套函数寻找变量,优先自己函数,再找父级、爷爷级等,没有就找全局变量(一层一层往上找)
age = 19
def func1():
age = 73
print(age) # 寻找变量,先找自己的函数内,没有就找全局变量(一层一层往上找)
def func2():
age = 84
print(age) # 寻找变量,优先自己函数、再找父级、爷爷级,最后找全局变量
func2()
func1() # 输出:73 84
# 测试二:
age1 = 19
def func1():
age1 = 73
def func2():
print(age1) # 按顺序往上找,找到父级的age=73
func2()
func1() # 输出:73
3、局部变量位置调整
age = 19
def func1():
def func2():
print(age)
age = 73
func2()
func1()
"""
输出:
73
"""
age = 19
def func1():
def func2():
print(age)
func2()
age = 73
func1()
'''
执行报错:free variable 'age' referenced before assignment in enclosing scope
73在func2后面,func2不知道该取哪个参数
'''
为了避免这种情况,声明变量尽量写在前面。
4、全局变量设置
age = 19
def func1():
global age # 此时已经拿到age=19,age=73还没有执行
def func2():
print(age)
func2()
age = 73 # 此时修改全局age=73
func1()
print(age)
"""
输出:
19
73
"""
age = 19
def func1():
global age
def func2():
print(age)
age = 73 # 将全局变量age改为73,再执行func2,函数没有在func2\func1中找到age,继续到全局找age,但全局已经修改为73
func2()
func1()
print(age)
"""
输出:
73
73
"""
第一个程序,在global age时,函数已经拿到age=19,但此时age=73还没有执行,此时执行func2函数,打印age=19。随后修改全局变量age=73,直接打印age,因此输出73.
第二个程序,在global age时,函数拿到age=19,随后修改全局变量age=73,接着执行func2函数,打印age=73,打印此时的全局变量age,也同样输出73.