一、函数定义
1 def name( parameters): #没有参数括号内可以为空 2 "函数描述" #其实就是注释 3 <代码块> 4 return [expression] #没有返回值可以不加[]内容,也可以省略return
def是定义函数的关键字,name是函数名,parameters是形参
函数描述可以省略,但建议要有
expression是返回值,可以没有返回值,也可以没有retrun。
函数在return处结束。
二、传参过程
这里需要了解几个名词:位置参数、关键参数、默认参数
1.位置参数:按照函数形式参数的顺序传递参数
def func(x,y): print("x={0} y={1}".format(x,y)) func(3,4) func(4,3) #输出: x=3 y=4 x=4 y=3
可以发现,位置参数的使用相当于C语言当中的函数调用方式。
2.关键参数:指的是在传递参数时,不必考虑形式参数的具体先后顺序,以“形式参数=实际参数”的形式传参。
def func(x,y): print("x={0} y={1}".format(x,y)) func(y=3,x=4) #输出结果: x=4 y=3
3.默认参数:如果函数有默认参数,允许调用函数时不对默认参数传参(这时形参为默认值)
def text2(x,y=2):#y为默认参数,非必须传递 print(x,y) text2(1) #可以这样调用 text2(1,3) #也可以这样调用 #输出结果: 1 2 1 3
4.参数组
如何向函数传递一个元组呢?
需要参数组。
def text3(*args): print(args) text3(1,2,3) #当使用参数组时,调用函数可以不传参 #输出: (1,2,3)
形参中*args是参数组,可以接收数量不固定的参数,也可以接收0个参数,并将这些参数组成一个元组在函数中使用。
参数组可以在参数个数不确定的情况下使用。
5.字典的传递
def text5(**kwargs):#向函数传递字典 print(kwargs) text5() #调用函数时可以不传参 text5(name="liuwei",sex="man",index=3) #标准的传参传字典过程 #输出结果: {“name”:"liuwei","sex":"man","index":3} #3是int型
这种传参方式很容易和关键参数混淆。
6.多种类型参数传递和优先级问题
def text6(a,b=2,*args,**kwargs): print(a) print(b) print(args) print(kwargs) text6("woo",3,sex="m",hobby="aa") #输出结果: woo 3 () {'sex': 'm', 'hobby': 'aa'}
在定义函数时,形参应当遵循 普通参数-->默认参数-->参数组-->字典参数 的顺序
在调用函数时,实参应当遵循 位置参数-->关键参数-->参数组-->字典参数 的顺序
这样做为的是避免混淆。
7.如何传递列表
(1)位置参数
def func(names): print(names) names[1] = 'jujinyi' names = ['aaron','jim'] func(names) print(names) #输出: ['aaron', 'jim'] ['aaron', 'jujinyi']
永久性改变了列表的值。如果想要对列表附件操作,可以用切片的方式
def func(names): print(names) names[1] = 'jujinyi' names = ['aaron','jim'] func(names) print(names) #输出: ['aaron', 'jim'] ['aaron', 'jim']
(2)关键参数
本质上和位置参数是一样的,没有什么区别
(3)元组传递
将传进去的元组使用list函数
def func(*toppings): toppings = list(toppings) print(toppings) func('mushrooms','extra','cheese') func('mushrooms','extra') #输出: ['mushrooms', 'extra', 'cheese'] ['mushrooms', 'extra']
三、返回值
可以返回任何类型。基本数据类型(字符、整型等),列表(元组),字典,集合,甚至函数名也可以被返回
如果返回多个值,所有的返回值会被封装成一个元组返回
四、局部变量和全局变量
全局变量是定义在函数体外面的变量,局部变量是定义在函数体内的变量
全局变量的作用域和生命周期从被定义开始直到程序结束,而局部变量则是从被定义到所在的函数结束。
2.全局变量的使用
全局变量当然可以在函数体外随意使用,在函数体内还会一样吗?
a=6 c=[1,2,3,4,5] def text(): a=5 b=10 c[1]=0 print("a={0},b={1},c={2}".format(a,b,c)) text() print("a={0},b={1},c={2}".format(a,b,c))
运行结果报错,问题出在最后一行:NameError: name 'b' is not defined。因为变量b是局部变量,只存在于函数内,在函数外已经不存在了。
改一下:
print("a={0},c={1}".format(a,c)) #输出结果: a=5,b=10,c=[1, 0, 3, 4, 5] a=6,c=[1, 0, 3, 4, 5]
发现了什么问题?a和c同为全局变量,都在函数内经过了修改,但是c被永久性的修改,a只是在函数内被修改,出来后被复原了。
这是因为列表c在函数内外都是原件,而整型a在函数内修改,修改的是函数拷贝的临时变量,保证了全局变量的安全性。
因此,可以用函数直接对列表、字典、集合等复杂类型的全局变量进行操作,但是无法在函数内对基本类型(如int,str,float等)的全局变量进行修改。
3.如何在函数内修改基本类型的全局变量?
Python提供了global关键字来进行声明
a=6 def text(): global a #声明要在函数内修改全局变量 a=10 print(a) text() print(a) #运行结果: 10 10
这样,全局变量就被修改了。
虽然有这种方法,但是如果没有这种强烈需求的话,尽量不要这么做。
五、高阶函数和嵌套函数
1.以下两个条件满足其一就可称之为高阶函数:
(1)把函数名当做实参传递给另一个函数
(2)返回值中包含函数名
2.函数可以嵌套,即函数体内可以定义函数
六、递归
定义:如果在函数内部可以调用自己,那么就称之为递归函数
递归需要满足的特性:
1.必须有一个明确的结束条件 (python内有预定的最大递归层数:999,程序的保护机制)
2.每深入一层时,问题的规模应该减小
3.递归的效率不高,递归层次过多容易导致栈溢出