Python学习总结【连载】(九)
2018.06.23 Juttachen
概述
+ 28 函数变量
- 函数名
- 为其他变量赋值
- 作为参数
- 作为返回值
+ 29 匿名函数
- 格式
+ 30 递归函数
- 定义
- 作用
- 步骤
- 函数的调用过程
+ 31 ??橛牒?/h3>
- 模块的类型
二十八、函数变量
我们在声明函数的时候,其实就是在声明一个类型是函数类型(function)的变量。
普通变量能做的事情,函数变量都能做。
1.函数名就是函数类型变量(类型名:function),存的是函数入口的地址。
def func1():
print('aaa')
print(type(func1))
num = 10
print(type(num))
2.可以用一个变量给另外一个变量赋值
注意:a = func1() #这样赋值,是将func1的返回值赋给a
a = func1 #将函数变量func1赋给a,a就会变成一个函数
a() #现在就可以用函数变量a,去调用函数func1
他们指向同一个地址,所以运行方式一样。
3.函数变量作为函数的参数
def calculate(num1,num2,func1):
'''num1 = 10 num2 =20 fn=func_add'''
'''如果num1不是int 或者float数据'''
if not(isinstance(num1,int) or isinstance(num1,float)):
print('num1不是数字')
if not(isinstance(num2,int) or isinstance(num2,float)):
print('num2不是数字')
return None
return func1(num1,num2)
def func_add(m,n):
return m+n
def fumc_sub(m,n):
return m-n
func2(10)
result = calculate(10,20,func_add)
print(result)
4.函数作为返回值
例子:写一个函数,func2('+') -> 返回求和的功能;如果是('')——>返回求乘积的功能*
def func2(char):
if char == '+':
def func2_add(*nums):
sum1 = 0
for x in nums:
sum1 += x
return sum1
return func2_add
if char =='*':
def func2_product(*nums):
pro1 = 1
for x in nums:
pro1 += x
return pro1
return func2_product
result = func2('+') # result是一个求和功能的函数的函数变量
print('aaa',result(1,2,3,4))
result2 = func2('*')(10,20)
print(result2)
def func4():
list1 = [1,2,3]
return list1
# 列表作为返回(对象变量作为返回值)
result = func4()
print(result)
print(func4()[0])
二十九、匿名函数
python 中使用lambda关键字来声明一个匿名函数
1.格式
lambda 参数列表:返回值
参数列表:多个参数之间用逗号隔开
例子:使用普通函数实现求两个数的和
def func1(num1,num2):
return num1 + num2
func1(10,20)
# 使用匿名函数实现
func2 = lambda num1,num2 : num1 + num2
func2(10,20)
相当于匿名函数就是可以替代函数体里只有return或者只有一句命令的函数。
练习:使用匿名函数求指定列表中,指定的两个下标对应的元素的和
list1 = [1,2,3,4]
func3 = lambda list1,num1,num2 : list1[num1]+list1[num2]
print(func3(list1,0,3))
def calculate(num1,num2,fn):
return fn(num1,num2)
sum1 = calculate(10,20,lambda a,b:a+b)
print(sum1)
三十、递归函数
1.定义
在函数的声明中调用函数本身就是递归函数。
例子:这里的func1就是一个递归函数,但是是一个死循环
def func1()
print('aaa')
func1()
func1()
2.递归作用
理论上循环能够做到的事情递归都可以做(但实际除非是非用不可的情况一般不用递归)原因:要调动函数,调动函数需要内存空间,还需要销毁,消耗空间且运行麻烦,损耗cpu
3.步骤:
a.确定临界值(结束函数的时刻)
b.假设函数的功能已经实现,然后找到f(n)和f(n-1)关系
c.使用f(n-1)与f(n)的关系去实现f(n)的功能
练习:计算1+2+3+4+...+n
方法一:使用循环
def func1(n):
sum1 = 0
for x in range(1,n+1):
sum1 += x
return sum1
方法二:使用递归
def func2(n):
# 1.确定临界值
if n == 1:
return 1
# 2.假设func2功能实现了,找func2(n-1),func2(n)的关系
# func2(n-1)=1+2+3+...+n-1
# func2(n) = 1+2+3+...+n-1+n = func2(n-1)+n
# 3.通过func2(n-1)去实现func2(n)的功能
return func2(n-1)+n
print(func1(10))
print(func2(10))
练习:用递归实现:24...*n(要求n是偶数)
def func3(n):
if n % 2 == 1:
print('n必须是偶数')
return None
else:
if n == 2:
return 2
return func3(n-2)*n
print(func3(10))
注意:无用代码不要写,不必一定要else,在函数中可以用return结束的就多用return。
例子:
n = 5
*****
****
***
**
*
def func4(n):
if n == 1:
# 1.确定临界值(一定要让函数结束)
print('*')
return None
# 需要结束的话,直接结束就好,不要和返回值混淆在一起了。
print('*'*n)
func4(n-1)
# f(n)和f(n-1)关系:先打印一行n个*,再执行f(n-1)
func4(5)
例子:
*
**
***
****
*****
def func5(n):
if n == 1:
print('*')
return None
func5(n-1)
print('*'*n)
func5(5)
4.函数的调用过程
a.调用函数的时候,首先在栈中开辟一块内存空间,用来保存函数调用过程中产生的数据
(包括函数的参数和在函数中声明的变量)
b.当函数执行完成后,被调用函数对应的内存空间会被回收(释放)
栈:是内存中的一块特殊的区域。数字和字符串的值是存在栈里面
堆:对象是存在堆里面的(在Python中不用我们担心,但是在c语言中这些对象是需要程序员自己申请内存的)
总结:递归要慎用!能用循环做的就不要用递归,因为递归消耗内存也消耗CPU资源
三十一、??橛牒?/h3>
模块就是指一个.py文件(一个.py文件就是一个??椋?br> 可以在一个??槭褂闷渌?榈暮屠?/p>
怎么使用其他??橹械暮??
1.通过import关键字导入??椋缓笥霉?槊?函数的方法使用模块中的函数
注意??!当我们导入其他的??榈氖焙蚧峤渌?橹械乃心谌荻蓟岬既氲降鼻拔募?。
如何解决执行语句也被运行?
判断??榈哪谥檬粜?strong>name的值是否是'main'
将不需要被其他??榈既氲拇胄吹秸飧鰅f语句中
if __name__=='__main__':
print(1,2,3)
关于name属性:
name是每个模块都有的属性,作用是用来存储??榈拿?。
当??椴皇钦谥葱械哪?椋?strong>name属性的值是??槎杂Φ奈募奈募?br>
当??檎谥葱械氖焙?strong>name的值就是一个固定值'main'
2.通过from-import直接导入指定的函数或者类。导入后就可以直接调用函数
格式:
from 模块名 import 函数名一,函数二
from qf_math import sum
print(sum(10,2))
3.通过as关键字给导入的模块或者函数重命名
import qf_math as my_math
# 重命名后可以使用新的??槊サ饔煤? pirnt(my_math.sum(1,3,5))
from qf_math import sum as my_sum
# 重命名后使用新的函数名调用函数
print(my_sum(10,2))
4.??榈姆掷?/em>
a.内置函数 (python 自己的)
b.第三方函数(别人写的)
c.自定义函数(自己写的)