函数进阶学习总结
匿名函数
没有名字的函数
语法:
函数名 = lambda 形参列表: 返回值
相当于
def 函数名(形参列表):
return 返回值
注意: 匿名函数的本质还是函数;普通函数中的绝大部分内容匿名函数都支持
# 练习:求任意两个数据的和的匿名函数
# def sum1(num1: int, num2=2):
# return num1 + num2
x = lambda num1, num2=2: num1 + num2
print(x(10, 20))
print(x(num1=100, num2=200))
print(x(100))
# 练习1: 写一个匿名函数判断指定的年是否是闰年
is_leap_year = lambda year: (year % 4 == 0 and year % 100 != 0) or (year % 400 == 0)
print(is_leap_year(2001))
# 练习:求值最大的元素
nums = [12, 39, 18, 80, 34]
print(max(nums))
# # 练习:求个位数最大的元素
def func1(x):
return x % 10
print(max(nums, key=func1))
print(max(nums, key=lambda x: x % 10))
# 练习:求绝对值最大的元素
nums = [12, 39, -18, -80, 34]
print(max(nums, key=lambda x: x**2))
print(max(nums, key=lambda x: -x if x < 0 else x))
变量作用域
意义:描述变量的使用范围
根据变量作用域不同,可以将变量分为两种:全局变量、局部变量
-
全局变量
定义在函数或者类外面的变量就是全局变量(没有定义在函数里面和类里面的变量就是全局变量); 全局变量的作用域:从定义开始到程序结束
# a是全局变量 a = 10 print('外面a:', a) # b是全局变量 for b in range(3): # c是全局变量 c = 20 print('循环里面a:', a) print('循环里面b:', b) print('外面b:', b) print('外面c:', c) def func1(): print('函数里面a:', a) print('函数里面b:', b) print('函数里面c:', c) func1()
-
局部变量
定义在函数里面的变量就是局部变量。(形参也是局部变量) 局部变量的作用域:从定义开始到函数结束
# d、e和f都是局部变量 def func1(d, e): f = 100 print('函数里面:', d, e, f) func1(300, 400) # print('函数外面:', d) # print('函数外面:', f)
-
global关键字
变量能不能使用,看的是使用的时候内存中有没有。 定义全局变量的时候,全局变量保存在全局栈区间,程序结束后才会被自动释放; 局部变量是保存在函数对应临时栈区间中,函数调用结束就会被自动释放。
global是函数体中关键字,可以在函数体中修饰变量,让变量在使用和保存的时候都在全局栈区间中进行。 1)函数中修改全局变量的值 2)直接在函数中定义全局变量
aa = 100 bb = 100 cc = 100 def func2(): # 不会修改全局变量aa的值,而且创建一个局部变量aa aa = 200 print('里面aa:', aa) # 在函数里面使用的是局部变量aa的值 # print(bb) # 报错! global修改变量必须放在这个变量使用之前 global bb bb = 200 print('里面bb:', bb) global cc cc = 300 func2() print('外面aa:', aa) # 在函数外面使用的是全局变量aa的值 print('外面bb:', bb) print('外面cc:', cc)
高阶函数
函数就是变量
python中定义函数其实就是在定义一个类型是function的变量,函数名就是变量名。
变量能做的事情函数都可以做。
def sum2(num1, num2):
return num1 + num2
# sum2 = lambda num1, num2: num1 + num2
b = [10, 20, 30]
print(type(b), type(sum2)) # <class 'list'> <class 'function'>
print(b[0], sum2(1, 3))
c = b
d = sum2
print(c[0], d(10, 20))
list1 = [10, b, sum2]
print(list1[1][-1]) # b[-1] -> [10, 20, 30][-1] -> 30
print(list1[-1](10, 20)) # sum2(10, 20) -> 30
高阶函数
实参高阶函数、返回值高阶函数
-
实参高阶函数:
函数的参数是函数;应该怎么来确定函数的参数是什么? - 看函数体中这个参数怎么用的:
def func1(x): print(x(10, 20)[-1]) def temp(a, b): return 'abc' func1(temp)
-
返回值高阶函数:
函数的返回值是函数
def func2(): def temp(x): return 23 return temp print(func2()(7) + 20)
常用的实参高阶函数
-
max、min、sorted、sort
参数key要求是一个函数
max(序列, key=函数) - 按照函数制定的比较规则来获取序列中最大的元素 函数的要求:1)参数 - 有且只有一个参数;这个参数代表前面序列中的每个元素 2)返回值 - 有一个返回值;返回值就是比较对象 注意:如果一个函数的参数是函数,这个参数有两种传值方式:a.普通函数的函数名 b.匿名函数
nums = [10, 29, 87, 34, -231, 72] # 练习1:求元素最大值 print(max(nums)) # 87 # 练习2:求元素最大值 result = max(nums, key=lambda item: item) print(result) # 87 # 练习3:求个位数元素最大值 result = max(nums, key=lambda item: item % 10) print(result) # 29 # 练习4:求绝对值最小的元素 result = min(nums, key=lambda item: item ** 2) print(result) # 练习5:求nums中数值最大的元素: '1998' nums = ['235', '90', '71', '1998', '80'] result = max(nums, key=lambda item: int(item)) print(result) # '1998' # 练习6:将nums中的元素按照十位数的大小从小到大排序 nums = [913, 281, 1765, 92, 802] # [802, 913, 1765, 281, 92] new_nums = sorted(nums, key=lambda item: item // 10 % 10) print(new_nums) # [802, 913, 1765, 281, 92] # 练习7:获取nums中各个位数之和最小的元素 nums = [1002, 908, 99, 76, 502] # [3, 17, 18, 13, 7] -> 1002 # 方法一: def temp(item): s = 0 for x in str(item): s += int(x) return s result = min(nums, key=temp) print(result) # 方法二 # 1002 - > '1+0+0+2' -> '+'.join('1002') -> '1+0+0+2' result = min(nums, key=lambda item: eval('+'.join(str(item)))) print(result)
-
map
1)map(函数, 序列) - 按照函数制定的规则将原序列转换成新的序列列表, 返回值是map对象-本质是序列 函数要求:a.参数:有且只有1个参数;参数代表后面的这个序列中的元素 b.返回值:有一个返回值;返回值就是新序列中的元素 2)map(函数, 序列1, 序列2) 函数要求:a.参数:有且只有2个参数, 分别代码后面两个序列中的元素 b.返回值:有一个返回值;返回值就是新序列中的元素 3)map(函数, 序列1, 序列2, 序列3) 函数要求:a.参数:有且只有3个参数;分别代码后面3个序列中的元素 b.返回值:有一个返回值;返回值就是新序列中的元素 map(函数, 序列1, 序列2, 序列3,...)
# 获取nums中所有元素的个位数 nums = [128, 239, 87, 96, 102, 91] # [8, 9, 7, 6, 2, 1] result = list(map(lambda item: item % 10, nums)) print(result) # [8, 9, 7, 6, 2, 1] str1 = 'abcde' nums = [10, 20, 30, 40, 50] # -> ['a10', 'b20', 'c30', 'd40', 'e50'] result = list(map(lambda i1, i2: i1 + str(i2), str1, nums)) print(result) # ['a10', 'b20', 'c30', 'd40', 'e50']
-
reduce
注意:reduce在使用之前必须先导入
from functools import reduce
reduce(函数, 序列, 初始值) - 按照函数制定的规则将序列中所有的元素合并成一个数据 函数的要求:1)参数:有且只有两个参数,第一个参数指向初始值,第二参数代表序列中每个元素 2)返回值:有一个返回值;描述初始值和元素之间的合并方式
nums = [10, 30, 20] # 所有元素的和:60 # 0 + 10 + 30 + 20 -> i + item result = reduce(lambda i, item: i+item, nums, 0) print(result) # 60 nums = [10, 40, 20] # 所有元素的乘积:8000 # 1 * 10 * 40 * 20 -> i * item result = reduce(lambda i, item: i * item, nums, 1) print(result) # 8000 nums = [12, 34, 91] # 所有元素个位数的和:7 # 0 + 2 + 4 + 1 -> 0 + 12%10 + 34%10 + 91%10 -> i + item%10 result = reduce(lambda i, item: i + item % 10, nums, 0) print(result) # 7 nums = [10, 40, 20] # '104020' # '' + '10' + '40' + '20' -> '' + str(10) + str(40) + str(20) -> i + str(item) result = reduce(lambda i, item: i + str(item), nums, '') print(result) # '104020' # 求所有数字的和 list1 = [10, 'abc', '2.4', 'hans', 5.5, 3] # 18.5 # [10, 5.5, 3] # 方法1: result = reduce(lambda i, item: i + item, [x for x in list1 if type(x) in (int, float)], 0) print(result) # 方法2: # list1 = [10, 'abc', '2.4', 'hans', 5.5, 3] # 0 + 10 + 5.5 + 3 -> 0 + 10 + 0 + 0 + 0 + 5.5 + 3 -> i + item / i + 0 result = reduce(lambda i, item: i+item if type(item) in (int, float) else i+0, list1, 0) print(result) # 将所有偶数的十位数和奇数的个位数求和 nums = [23, 54, 801, 132, 92] # 3+5+1+3+9 # i + item % 10 、 i + item // 10 % 10 def temp(i, item): if item % 2: return i + item % 10 return i + item // 10 % 10 # result = reduce(lambda i, item: i + item % 10 if item % 2 else i + item // 10 % 10, nums, 0) result = reduce(temp, nums, 0) print(result)
总结:
1、函数参数传递的是内存地址
想重新创建一份数据再传递给参数,可以利用deepcopy手动拷贝一份。
特殊:参数是动态参数是,通过和**传参时,会将数据循环添加到参数中(类似于拷贝)
2、函数的返回值也是内存地址(函数执行完毕后,其内部的所有变量都会被销毁,引用计数器变为0时,数据也销毁)
3、当函数的参数有默认值 且 默认值是可变类型(列表、字典、集合) 且 函数内部会修改默认值(有坑警告)
4、定义函数写形式参数时可以用和**,执行函数时也可以使用。
5、函数名也是变量,可做列表、字典、集合等元素(可哈希)
6、函数名可以背重新赋值,也可以做另外一个函数的参数和返回值
7、print只是打印,return是将函数的执行结果返还给调用者
8、python以函数为作用域
9、在局部作用域中寻找某数据时,优先使用自己的,自己没有再去上级作用域中找
10、基于global关键字可以在局部作用域中实现对全局作用域中的变量(全局变量)重新赋值。