一、函数定义

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']

三、返回值

可以返回任何类型。基本数据类型(字符、整型等),列表(元组),字典,集合,甚至函数名也可以被返回

如果返回多个值,所有的返回值会被封装成一个元组返回

四、局部变量和全局变量

1.定义 

全局变量是定义在函数体外面的变量,局部变量是定义在函数体内的变量

全局变量的作用域和生命周期从被定义开始直到程序结束,而局部变量则是从被定义到所在的函数结束。

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.递归的效率不高,递归层次过多容易导致栈溢出