DAY11 函数进阶
一. 匿名函数
没有名字的函数
-
语法:(只能写在一行)
函数名 = lambda 形参列表:返回值
相当于:
def 函数名():
return 返回值
- 注意:匿名函数的本质就是函数;普通函数中的绝大部分内容都支持
# 练习:求任意两个数据的和的匿名函数
# def sum1(num1, num2):
# return num1 + num2
x = lambda num1, num2=2: num1 + num2
print(x(10, 20))
print(x(num1=100, num2=200))
print(x(100))
# x = lambda num1:int, num2=2: num1 + num2 报错,不能写参数类型
# x = lambda *num1, num2=2: num1 + num2 支持不定长参数
#
is_leap_year = lambda year: (year % 4 == 0 and year % 100 != 0) or year % 400 == 0
print(is_leap_year(2000))
print(is_leap_year(2001))
# 练习:求个位数最大是元素
print(max([19, 25, 31], key=lambda x: x % 10))
# 练习:求绝对值最大的元素
nums = [12, 39, -18, -80, 34]
print(max(nums, key=lambda x:x if x >= 0 else -x))
print(max(nums, key=lambda x: x**2))
print(max(nums, key=lambda x: abs(x)))
二. 变量的作用域
2.1 变量的作用域 - 变量使用范围
根据变量作用域不同,可以将变量分为两种:全局变量、局部变量
2.2 全局变量
- 定义在函数或者类外面的变量就是全局变量(没有定义在函数里面或者类里面的变量就是全局变量)
- 全局变量的作用域:从定义开始到程序结束
# a, b, c都是全局变量
a = 10
for b in range(3):
c = 20
2.3 局部变量
- 定义在函数里面的变量就是局部变量(形参也是局部变量)
- 局部变量作用域:从定义开始到函数结束
# d,e,f都是局部变量
def func1(d, e):
f = 100
2.4 global关键字
- 变量能不能使用,看的是使用的时候内存中有没有
- 定义全局变量的时候,全局变量保存在全局栈区间,程序结束后才会被自动释放;局部变量是保存在函数对应的临时栈区间中,函数调用结束就会被自动释放
- global是函数体中关键字,可以在函数体中修饰变量,让变量在使用和保存的时候都在全局栈区间中进行
aa = 100
bb = 100
def func2():
aa = 200 # 又创建了局部变量aa,在临时栈区间,不会修改全局变量
print('函数内aa=', aa) # 200,局部变量aa的值
# print(bb) 报错!global修改变量必须放在这个变量使用之前
global bb # global后,使用全局变量,只能写global XX,操作写下面
bb = 200
print('函数内bb=', bb) # 200
global cc # 全局变量中没有cc,函数中也可以创建全局变量
cc = 300
func2()
print('函数外aa=', aa) # 100,全局栈区间的
print('函数外bb=', bb) # 200
print('函数外cc=', cc) # 300
三. 高阶函数
3.1 函数就是变量
python中定义函数就是在定义一个function的变量,函数名就是变量名,变量能做的事函数都可以做
def sum2(num1, num2):
return num1 + num2
print(type(sum2)) # <class 'function'>
d = sum2
print(d(10, 20)) # 30
list1 = [10, sum2]
print(list1[-1](10, 20)) # 30
3.2 高阶函数
-
实参高阶函数 - 函数的参数是函数
Q:应该怎么确定函数的参数是什么?
A:看函数体中这个参数怎么用
-
返回值高阶函数 - 函数的返回值是函数
def func2():
def temp(x):
return 23
return temp
print(func2()(7) + 20)
3.3 装饰器
既是实参高阶函数,又是返回值高阶函数,需要自己写
四. 常用的实参高阶函数
4.1 max, min, sorted, sort - 参数key要求是一个函数
-
max(序列, key=函数) - 按照函数制定的比较规则来获取序列中最大的元素
-
函数的要求:1) 参数 - 有且只有一个参数;这个参数代表前面序列中
的每个元素
2)返回值:有一个返回值;返回值就是比较对象
- 注意:如果一个函数的参数的函数,这个参数有两种传值方式:普通函数的函数名、匿名函数
nums = [10, 29, 87, 34, -231, 72]
# 练习1:求元素最大值
print(max(nums)) # 87
result = max(nums, key=lambda item: item)
print(result) # 87
# 练习2:求个位数元素最大值
result = max(nums, key=lambda item: item % 10)
print(result) # 29
# 练习3:求绝对值最小的元素
result = min(nums, key=lambda item: item ** 2)
print(result)
# 练习4:求nums中数值最大的元素'1998'
nums = ['235', '90', '71', '1998', '80']
result = max(nums, key=lambda item: int(item))
print(result)
# 练习6:将nums中的元素按照十位数的大小从小到大排序
nums = [913, 281, 1765, 92, 802] # [802, 913, 1765, 281, 92]
result = sorted(nums, key=lambda item: item % 100 // 10)
print(result)
# 练习7:获取nums中各个位数之和最小的元素
nums = [1002, 908, 99, 76, 502] # [3, 17, 18, 13, 7] -> 1002
# 方法1:普通函数
def sum_ep(num):
str1 = str(num)
s = 0
for x in str1:
s += int(x)
return s
# print(sum_ep(123))
result = min(nums, key=sum_ep) # 直接写函数名,不写括号
print(result)
# 方法2
result = min(nums, key=lambda item: eval('+'.join(str(item))))
print(result)
4.2 map
-
map(函数,序列) - 按照函数指定的规则将原序列转换成新的序列列表,返回值是一个map对象,本质是序列
函数要求:a.参数:有且只有一个参数;参数代表后面序列中的元素
b.返回值:有一个返回值;返回值就是新序列中的元素 -
map(函数,序列1,序列2)
函数要求:a.参数:有且只有两个参数;分别代表后面两个序列中的元素
b.返回值:有一个返回值;返回值就是新序列中的元素 -
map(函数,序列1,序列2,序列3)
函数要求:a.参数:有且只有三个参数;分别代表后面三个序列中的元素
b.返回值:有一个返回值;返回值就是新序列中的元素
# 获取nums中所有元素的个位数
nums = [128, 23, 343, 545, 123, 98]
result = list(map(lambda item: item % 10, nums))
print(result)
str1 = 'abcde'
nums = [10, 20, 30, 40, 50]
result = list(map(lambda i1, i2: i1 + str(i2), str1, nums))
print(result)
4.3 reduce
# 注意:reduce在使用之前必须先导入
from functools import reduce
- reduce(函数,序列,初始值) - 按照函数指定的规则将序列中所有的元素合并成一个数据
- 函数的要求:1)参数:有且只有两个参数,第一个参数指向初始值,第二个参数代表序列中每个元素
2)返回值:有一个返回值;描述初始值和元素之间的合并方式
nums = [10, 30, 20]
result = reduce(lambda i, item: i + item, nums, 0) # i:初始值,item:序列中每个元素
print(result) # 60,所有元素的和
# 练习:求所有数字的和
list1 = [10, 'abc', '2.4', 'hans', 5.5, 3] # 18.5
list2 = [x for x in list1 if type(x) in (int, float)]
result = reduce(lambda i, item: i + item, list2, 0)
print(result)
result = reduce(lambda i, item: i + item if type(item) in (int, float) else i + 0,list1, 0)
nums = [23, 54, 801, 132, 92] # 3+5+1+3 偶数加十位,奇数加个位数
result = reduce(lambda i, item: i + item % 10 if item % 2 else i + item % 100 // 10, nums, 0)
print(result)