day12- 函数进阶
1、匿名函数
-
匿名函数的本质还是函数,他们出来定义的语法不同,其他的东西都一样
-
注意:匿名函数只能实现功能很简单的函数
语法:
lambda 形参列表:返回值
转换成普通:
def (形参列表):
return 返回值
2、变量的作用域
-
变量的作用域:变量定义以后可以使用的范围
-
根据变量作用域的不同我们可以将变量分为全局变量和局部变量
-
全局变量和局部变量
-
全局变量
没有定义在函数里面的变量都是全局变量 全局变量的作用域是从定义开始到程序结束 全局变量保存在全局的栈区间,全局栈区间在程序结束后才会自动释放
-
局部变量
定义在函数中的变量是局部变量,其中包括形参; 局部变量的作用域是从定义开始到函数结束 调用函数的时候系统会自动给这个函数创建一个临时栈区间用来保存在函数中产生的数据(包括局部变量) 当函数调用结束这个临时栈区间会自动释放
-
global
-
作用1:在函数里面去修改一个全局变量的值,修改之前先用global对变量进行说明
-
作用2:在函数中定义一个全局变量
3、函数就是变量
- 在python中定义函数其实就是定义类型是function的变量,函数名就是变量名(变量能干的事情函数都能做)
- 高阶函数:实参高阶函数、返回值高阶函数
- 实参高阶函数:他有一个参数是高阶函数
4、实参高阶函数
-
常用的实参高阶函数:max、min、sorted(sort);map、reduce
-
如果函数的参数是函数提供实参的方式:a.用普通的函数的函数名、b.使用匿名函数
-
1)max、min、sorted(sort)
-
max(序列) - 求序列中值最大的元素 max(序列,key=函数) - 根据函数指定的规则获取序列中最大的元素 函数的要求:1)有且只有一个参数(代表序列中的元素) min(序列,key=函数) - 根据函数指定的规则获取序列中最小的元素 sorted(序列,key=函数) - 根据函数指定的规则将序列中的元素从小到大排序 列表.sort(序列,key=函数) - 根据函数指定的规则将序列中的元素从小到大排序
# 练习3:获取nums中绝对值最大的元素
nums = [29, 73, 64, -31, 68, 55, -76]
print(max(nums, key=lambda item: item if item > 0 else -item))
# 练习4:获取points中y坐标最大的元素
points = [(19, 2), (0, 18), (3, 4)]
print(max(points, key=lambda item: item[-1]))
# 练习5:获取年龄最大的学生;获取成绩最高的学生
students = [
{'name': 'stu1', 'age': 19, 'score': 72},
{'name': 'stu2', 'age': 22, 'score': 89},
{'name': 'stu3', 'age': 30, 'score': 60},
{'name': 'stu4', 'age': 17, 'score': 77}
]
print(max(students, key=lambda item: item['age']))
print(max(students, key=lambda item: item['score']))
-
2)map - 将序列中的元素按照指定的规则进行变换得到一个新的序列
-
map(函数,序列) 函数的要求:1、有且只有一个参数(代表原序列中元素) 2、有一个返回值(新序列中的元素 - 在这儿只需要描述清楚新序列中元素与原序列元素的关系)
# 练习2:将nums中所有的元素乘以2
nums = [29, 73, 64, 301, 68, 55, 76]
# [58, 146, 128, 602, 136, 110, 152]
num1 = list(map(lambda item: item * 2, nums))
print(num1)
# 练习4:
nums1 = [238, 89, 78, 23]
nums2 = [92, 78, 61, 44]
# [82, 98, 81, 34]
num2 = list(map(lambda item1, item2: item1 % 10 * 10 + item2 % 10, nums1, nums2))
print(num2)
# 练习5
list1 = ['python', 'java', 'h5']
list2 = [22, 21, 19]
list3 = [2, 9, 18]
# ['python2202', 'java2109', 'h51918']
num2 = list(map(lambda item1, item2, item3: f'{item1}{item2}{item3:0>2}', list1, list2, list3))
print(num2)
-
3)reduce - 将序列中的元素按照指定的规则合并成一个数据
```python
reduce(函数,序列,初始值)
初始值一般是0、1、空串
函数的要求:1、有且只有两个参数(第一个参数指向初始值;第二个参数指向序列中的每个参数)
2、有一个返回值(决定合并方式)
注意:使用reduce之前必须先从functools模块中导入
```
from functools import reduce
students = [
{'name': 'stu1', 'age': 19, 'score': 72},
{'name': 'stu2', 'age': 22, 'score': 89},
{'name': 'stu3', 'age': 30, 'score': 60},
{'name': 'stu4', 'age': 17, 'score': 77}
]
# 求总分数
nums = reduce(lambda x, item: x + item['score'], students, 0)
print(nums)
num3 = [10, 23, 4, 8]
# 求所有元素的乘积: 10x23x4x8
nums = reduce(lambda x, item: x * item, num3, 1)
print(nums)
作业练习
-
写一个匿名函数,判断指定的年是否是闰年
num1 = lambda item: (item % 4 == 0 and item % 100 != 0) or item % 400 == 0 print(num1(2008))
-
写一个函数将一个指定的列表中的元素逆序( 如[1, 2, 3] -> [3, 2, 1])(注意:不要使用列表自带的逆序函数)
num1 = lambda x: list(x[-i - 1] for i in range(len(x))) print(num1([1, 2, 3]))
-
编写一个函数,计算一个整数的各位数的平方和
例如: sum1(12) -> 5(1的平方加上2的平方) sum1(123) -> 14
def sum1(int1):
num1 = 0
for x, y in enumerate(str(int1)):
num1 += int(y) ** 2
return num1
print(sum1(1234))
num1 = lambda x: sum(int(i) ** 2 for i in str(x))
print(num1(123))
- 求列表 nums 中绝对值最小的元素
例如:nums = [-23, 100, 89, -56, -234, 123], 最大值是:-23
nums = [-23, 100, 89, -56, -234, 123]
num1 = min(nums, key=lambda item: -item if item < 0 else item)
print(num1)
- 已经两个列表A和B,用map函数创建一个字典,A中的元素是key,B中的元素是value
A = ['name', 'age', 'sex']
B = ['张三', 18, '女']
新字典: {'name': '张三', 'age': 18, 'sex': '女'}
num1 = dict(map(lambda item1, item2: (item1, item2), A, B))
print(num1)
- 已经三个列表分别表示5个学生的姓名、学科和班号,使用map将这个三个列表拼成一个表示每个学生班级信息的的字典
names = ['小明', '小花', '小红', '老王']
nums = ['1906', '1807', '2001', '2004']
subjects = ['python', 'h5', 'java', 'python']
结果:{'小明': 'python1906', '小花': 'h51807', '小红': 'java2001', '老王': 'python2004'}
names = ['小明', '小花', '小红', '老王']
nums = ['1906', '1807', '2001', '2004']
subjects = ['python', 'h5', 'java', 'python']
dict1 = dict(map(lambda names1, subjects1, nums1: (names1, f'{subjects1}{nums1}'), names, subjects, nums))
print(dict1)
- 已经一个列表message, 使用reduce计算列表中所有数字的和
message = ['你好', 20, '30', 5, 6.89, 'hello']
结果:31.89
from functools import reduce
num1 = reduce(lambda x, item: x + item if type(item) in (int, float) else x + 0, message, 0)
print(num1)
- 已经列表points中保存的是每个点的坐标(坐标是用元组表示的,第一个值是x坐标,第二个值是y坐标)
points = [
(10, 20), (0, 100), (20, 30), (-10, 20), (30, -100)
]
1)获取列表中y坐标最大的点
num1 = max(points, key=lambda item: item[-1])
print(num1)
2)获取列表中x坐标最小的点
num1 = min(points, key=lambda item: item[0])
print(num1)
3)获取列表中距离原点最远的点
num2 = max(points, key=lambda item: (item[-1] ** 2 + item[0] ** 2) ** 0.5)
print(num2)
4)将点按照点到x轴的距离大小从大到小排序
num3 = sorted(points, key=lambda item: item[-1])
print(num3)