您的位置:首页 > 编程语言 > Python开发

python函数基础:嵌套函数、作用域、匿名函数、高阶函数、递归函数

2018-01-27 01:00 681 查看
嵌套函数:

1. 函数内部可以再定义函数

2. 函数只有被调用之后才会执行

看如下代码:

age = 18

def func1():
age = 22
print(age)

def func2():
age = 28    # 如果这个age没有被赋值,它会先向它的父级(func1)里面找,如果父级也没有就再向它的爷爷级(全局的age)找。  # 一层一层由内向外找
print(age)
func2()

func1()

# 输出结果:
#22
#28


注: 函数内部的变量都叫局部变量,只不过局部变量之间也有等级关系

#情景1:
age = 18

def func1():

def func2():
print(age)
age = 22  # age=22 依然是func1里面的变量
func2()   #程序从上到下运行到这一步的时候 age已经在函数内被赋值为22

func1()

#输出结果:
# 22

#情景2:
age = 18

def func1():

def func2():
print(age)
func2()   #程序由上到下运行到这一步时,由于func2中没有age,func2会向上一级找,找到了func1中的age=22,但由于变量需要先定义后使用(定于放在使用前面),所以程序会报错。例如情景3
age = 22  # age =22依然是函数func1里面的变量

func1()

# 输出结果会报错。

#情景3:
age = 18

def func1():

def func2():
print(age2)
func2()
age2 = 22

func1()

# 输出结果会报错。

#情景4:
age = 18

def func1():
global age  #程序执行到这一步的时候,age已经被声明成了全局变量,并被赋值为18
def func2():
print(age)  #执行到这一步的时候age直接调用age这个全局变量
func2()
age = 22   #  程序执行到这一步的时候age这个全局变量又被赋值成了22

func1()
print(age)

# 输出结果:
# 18
# 22

#情景5:

age = 18

def func1():
global age  #到这一步age被声明成全局变量,并且此时的值还是18
age = 22   # 到这一步时age又被重新赋值为22
def func2():
print(age)
func2()

func1()

#输出结果:
# 22

#情景6:
age = 18

def func1():
global age #执行到这一步是age还是18
def func2():
print(age)   #执行到这一步的时候age已经是22了
age = 22  # age这个全局变量 在func2调用之前又被改成了22
func2()

func1()

# 输出结果:
# 22


注:这几种情况用于分析,实际生产中很少用。

作用域:

在Python中一个函数的就是一个作用域,局部变量其实是放在它的作用域中

age = 18
def func1():
age = 22
def func2():
print(age)

return func2   #func2没加括号,返回的是这个函数名

val= func1()
print(val)  #打印的是函数名
val()     #val就是func2, 此时会执行

# 输出结果:
# <function func1.<locals>.func2 at 0x0000009F6AEAB9D8>
# 22   #虽然是在外部执行,但依然是通过其定义域的关系去调用,所以不会是18.


代码定义完成后,作用域就已经生成。以后调用时会通过其定义域关系向上查找(不管这个函数名是在哪里被调用,只要一执行,它还是会回来它定义的地方向上去查找。)。

匿名函数:

普通函数:

def calc( x,y):

return x*y

匿名函数: lambda x,y: x*y #声明一个匿名函数 (把多行语句变成一行)

调用:

func = lambda x,y: x*y

func(3,8)

匿名函数最复杂的也只能进行三元运算。

匿名函数通常跟其他的方法搭配使用,作用主要是节省代码量、如:

# 要求: 使range(10)里面的数各自跟自己相乘

data = list(range(10))
print(list(map(lambda x:x*x,data)))

# 输出结果:
# [0, 1, 4, 9, 16, 25, 36, 49, 64, 81]

# map的用法: map(func, *iterables) --> map object
# 就是把iterables中的每一个值都进行一下前面的函数。
# list(map())的作用是让map的结果变成列表的样子
# iterables有:list,str,tuple,dict,file,xrange等


高阶函数:

变量可以指向函数,函数的参数能够接收变量,一个函数可以接收另一个函数作为参数,这种函数就叫高阶函数。

另外,一个函数return了另外一个函数,这个也是高阶函数,如:

def func(x,y):
return abs,x,y   # return 里面有abs, 是一个函数, 所以 func是高阶函数; 假如 return abs(x+y) 那func就不是高阶函数, 因为返回的是 abs(x+y),这是一个函数的结果, 不是函数。
res = func(3,-10)
print(res)

# 输出结果:
# (<built-in function abs>, 3, -10)

# func也是高阶函数
# abs是求绝对值的函数,用法:abs( 数字)
# 函数返回多个值的时候,是以元祖形式返回的


总结:

只需满足以下一个条件即为高阶函数:

1. 接收一个或多个函数作为输入

2. return返回另一个函数

递归函数:

如果一个函数在其内部调用了它自己,这样的函数就是递归函数。 如:让10除以2,直到为0.

a= 10
def calc(n):
n = int(n/2)
print(n)
if n >0:
calc(n)   #在函数内部调用它自己,就会产生循环
calc(a)

# 输出结果:
# 5
# 2
# 1
# 0

## 而且,程序在结束退出的时候,是从内向外、一层一层逐渐结束的。
测试:
a= 10
def calc(n):
n = int(n/2)
print(n)
if n >0:   #这一步的代码运行分析:第1次print的n是5,由于5>0,n直接进入calc(n),由于calc(n)调用了它自己,就没有再走下面的print(‘程序退出测试:’,n)这一步,而是又返回上面去进行n=int(n/2)这几步。n是2和1的时候也是同样的道理。
calc(n)
print('程序退出测试:',n)  # 打印结果分析:最后一轮循环n的值是0,此时n==0不再进行calc(n),而是去运行下面的print(‘程序退出测试:’,n)这一步,此时最里面的这一层循环结束;然而上一步n==1时,程序是直接进入了if语句进行了calc(n),并没有进行下面的print语句,所以当最里面的n==0这层程序走完之后,n==1也要走完这个print语句。同理,n==2和5时也要依次走这个print语句。

calc(a)

# 输出结果:
# 5
# 2
# 1
# 0
# 程序退出测试: 0
# 程序退出测试: 1
# 程序退出测试: 2
# 程序退出测试: 5


递归函数返回值:

在递归函数的外面得到返回值是最外层的函数return回来的,想要得到递归函数的返回值,需要在递归函数内部每次调用时都要有return值,只有这样最内层的return值才能通过一层层的return返回到最外层。它的书写方式是:

if 条件成立:

return 调用自己

else:

return 最里面的值

例如下面两个例子:

例子1: 100除以2三次,函数外部得到返回值。

a = 100
def func(n,count):
if count < 3:
return func(n/2,count+1)
else:
return n
result = func(a,0)
print(result)

# 输出结果:
# 12.5

例子2: 求3的阶乘,函数外部得到返回值。

def f(n):
if n > 1:
return n*f(n-1)
else:
return n

print(f(3))

# 输出结果:
# 6


下面用流程图说明它们的返回值是怎么得到的:

1. 3的阶乘:



2. 100除以2三次:

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐