day 12-函数的进阶

day 12-函数的进阶

一、变量作用域

1.变量作用域

变量作用域指的是变量定义完成后可以使用的有效范围。
根据变量变量作用域的不同,可以将变量分为全局变量和局部变量两种。

2. 全局变量和局部变量

1)全局变量
python中没有定义在函数中或者类中的变量默认都是全局变量。
作用域:从定义开始到程序结束

2)局部变量
定义在函数中的变量就是局部变量(形参也是局部变量)
作用域:从定义开始到函数结束

a.全局变量
a = 100
# x和b都是全局变量
for x in range(5):
    b = 200

print(f'外面使用:{a}')
print(f'外面使用:{b}')
print(f'外面使用:{x}')

# 循环中也可以使用
for y in range(3):
    print(f'外面使用:{a}')
    print(f'外面使用:{b}')
    print(f'外面使用:{x}')
b.局部变量
# m 和 n 是局部变量
def fun2(m):
    n = 1000
    print(f'函数里面m:{m}')
    print(f'函数里面m:{n}')

fun2(200)
# print(f'函数外面m:{m}')  ---- 报错
# print(f'函数外面m:{n}')
3.全局变量和局部变量的存储原理

1)全局变量默认保存在全局栈区间,全局栈区间会在程序结束后自动释放。

2)调用函数的时候系统会自动为这个函数创建一个临时栈区间,用来保存在函数张中产生的数据(局部变量就是保存在函数对应的临时区间中的),函数对应的临时栈区间会在函数调用结束的时候自动释放。
3)在函数中可以通过关键字global修改局部变量的保存方式,让局部变量保存到全局栈区间——变为全局变量

def fun3():
    global num
    num = 10
    nume = '小花'

fun3()
print(f'num:{num}')  # num:10
# print(f'name:{name}') ---报错,name是局部变量

def fun4():
    global name
    name = '小花'

fun4()
print(f'name:{name}')   # name:小花

二、匿名函数

1.匿名函数 – 没有函数名的函数
lambda 形参列表:返回值

相当于:
def 函数名(形参列表):
return 返回值

注意:

1)匿名函数形参至少一个
2)匿名函数的调用和普通函数一样
3)无默认值的类型说明在匿名函数中不能用

案例:定义一个匿名函数,求任意两个数的和

sum_2 = lambda num1, num2: num1 + num2
num1 = 1
num2 = 2
print(sum_2(num1, num2))  # 3
print(sum_2(100, 200))  # 300

练习:定义一个匿名函数,判断指定的年是否是闰年

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))

三、实参高阶函数

函数的参数是函数的函数就是实参高阶函数

给参数是函数的参数传参:a.使用普通函数函数名;b.使用匿名函数

重点:掌握系统或者第三方库提供的实参高阶函数的用法

常见的实参高阶:max、min、sorted、列表.sort、map、reduce

# func6就是一个实参高阶函数
def func6(x):
    # x = t
    x()     # t()

def t():
    print('t函数')
    
func6(t)    # t作为一个参数也是函数
# func7是实参高阶函数
def func7(x):
    # x = t2
    # x = lambda n1, n2: n1 + n2
    print(x(10, 20) + 30)
    # print(t2(10, 20) + 30)  print(200 + 30)
    # print(30 + 30)

def t2(n1, n2):
    # n1 = 10, n2 = 20
    return n1 * n2      # return 200

func7(t2)   # t2函数有两个参数,调用func7时得给两个参数的值
func7(lambda n1, n2: n1 + n2)

四、常见的实参高阶函数

1.max、min、sorted、列表.sort
1)max(序列)  - 直接比较序列元素的大小求最大值
2)max(序列,key = 函数)   -  按照函数制定的规则比较序列中元素的大小获取最大值
函数的要求:
     a.有且只有一个参数(这个参数代表序列中的每个元素)
     b.有一个返回值(返回值就是比较对象)

案例:求个位数最大的元素

num = [21, 92, 86, 48, 69, 55]
result = max(num, key=lambda item: item % 10)
print(result)

案例:求num中绝对值最大的元素

num = [-23, 45, 67, -99]
result = max(num, key=lambda item:item ** 2)

练习3:将所有的学生按照年龄值从小到大排序

students = [
    {'name': 'stu1', 'age': 18, 'score': 90},
    {'name': 'stu2', 'age': 22, 'score': 98},
    {'name': 'stu3', 'age': 25, 'score': 78},
    {'name': 'stu4', 'age': 19, 'score': 81},
    {'name': 'stu5', 'age': 20, 'score': 92}
]
# 1
students.sort(key=lambda item: item['age'])
print(students)
# 2
print(sorted(students, key=lambda item: item['age'], reverse=True))
2.map - 基于原序列中的元素创建一个新序列(原序列可以是任意多个)
1map(函数,序列)   - 通过函数描述的规则基于序列中的元素创建一个新的序列
函数要求:
      a.有且只有一个参数(代表后面的这个序列中的每个元素)
      b.有一个返回值(返回值就是新序列中的元素)

2map(函数, 序列1, 序列2)
函数要求:
      a.有且只有2个参数(代表后面的两个序列中的每个元素)
      b.有一个返回值(返回值就是新序列中的元素)

3map(函数, 序列1, 序列2, 序列3...)

案例2:将相同位置上的元素想乘得到一个新序列

nums1 = [1, 2, 3, 4, 5, 6]
nums2 = [10, 20, 30, 40, 50, 60]
reuslt = map(lambda i1, i2: i1*i2, nums1, nums2)
print(list(reuslt))

练习1:获取names中每个人的姓

names = ['何晓东', '张三', '李四', '王五', '王二', '赵六']

reuslt = map(lambda item: item[0], names)
print(list(reuslt))    # ['何', '张', '李', '王', '王', '赵']

练习2:

students = [
    {'name': 'stu1', 'age': 18, 'score': 90},
    {'name': 'stu2', 'age': 22, 'score': 98},
    {'name': 'stu3', 'age': 25, 'score': 78},
    {'name': 'stu4', 'age': 19, 'score': 81},
    {'name': 'stu5', 'age': 20, 'score': 92}
]
subjects = ['电子信息', '金融数学', '园林设计', '经济', '计算机软件']

# ['电子信息-stu1', '金融数学-stu2',....]
reuslt = map(lambda i1, i2: f'{i2}-{i1["name"]}', students, subjects)
print(list(reuslt))
reuslt = map(lambda i1, i2: '-'.join([i2, i1['name']]), students, subjects)
print(list(reuslt))    # .join是在序列中以'-'加
3.reduce - 将序列中的元素合并成一个数据(基于序列中所有的元素得到一个数据)
reduce(函数,序列, 初始值)   -   按照函数制定的规则将序列中的元素合并成一个数据
函数要求:
    a.有且只有两个参数(第一个参数指向初始值,第二个参数代表序列中的每个元素)
    b.需要一个返回值(返回值就是合并规则)
    
初始值:累计求数值和,初始值是0
       累计求数值乘积,初始值是1
       字符串合并,初始值是空串
from functools import reduce
# 求元素和
nums1 = [1, 2, 3, 4, 5, 6]
reuslt = reduce(lambda x, item: x + item, nums1, 0)
print(reuslt)  # 21

# 求个位数数字之和
nums = [12, 301, 40, 55, 112]
reuslt = reduce(lambda x, item: x + item % 10, nums, 0)
print(reuslt)

五、迭代器

1.什么是迭代器

1)迭代器是容器型数据类型(可以遍历,转换成列表),无法直接提供一个迭代器,只能将其他序列转换成迭代器
2)特点:

a.打印迭代器的时候无法查看到元素有哪些
b.无法通过len获取迭代器中元素的个数
c.如果要使用迭代器中的元素必须将元素从迭代器中取出来,取走的元素会从迭代器中永远消失(用一个少一个)

3)任何数据都可以作为迭代器的元素

1)创建迭代器
i1 = iter('abc')
i2 = iter([10, 20, 30, 40])
i3 = iter()
2)打印迭代器无法查看元素
print(i1)
print(i2)
3)无法统计元素个数
print(len(i1))   ---- 报错
2.获取迭代器中的元素

无论以任何方式获取到迭代器中的某个元素,那么这个元素一定会从迭代器中消失。

1)获取单个元素:next(迭代器) - 获取迭代器最前面的元素

2)遍历迭代器

for 变量 in 迭代器:
    循环体
print(next(i1))   # a
print(next(i1))   # b
print(next(i1))   # c

# print(next(i1))   # 报错!(此时i1迭代器已经空了)

print(next(i2))   # 10
print(list(i2))   # [20, 30, 40]

for x in i3:
    print(x)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值