07函数加函数进阶

一、函数

在实现某个功能对应的代码的时候,如果将实现功能对应的函数放到函数中,那么

1.什么是函数

函数就是实现某一特定功能的代码的封装。–函数就把实现某个功能的代码打成一个包,以后再需要这个功能的时候,直接调用函数就可以了。(不同的函数可以提供不同的功能)

分类:
1.系统函数:有系统创建好函数,只需要在需要他们提供的功能的时候去调用它,例如:print、input、max、type等
2.自定义函数:由程序员自己创建的函数

2.定义函数(创建函数)

1.语法:
def 函数名(形参列表):
函数说明文档
函数体
2.说明:
def — 关键字;固定写法
函数名 — 程序员自己命名:
两个要求:1.标识符2.不是关键字
三个规范:1.见名知义(看到函数名就大概知道这个函数的功能)
2.不使用系统函数名、类名、模块名
3.所有字母都小写,多个单词用下划线隔开
形参列表 — 以’变量名1,变量名2,变量名3…'的形式存在,每一个变量就是一个形参。
形参的个数可以是任意多个:0个、1个…。
形参可以将函数外部的数据传递到函数里面。
函数说明文档 — 本质就是一个多行注释
函数体 — 函数体就是和def保持一个缩进的一条或者多条语句(至少一条),逻辑上,函数体就是实现函数功能的代码
3.初学者定义函数的步骤:
第一步:确定函数功能
第二步:确定函数名
第三步:确定行参(看实现函数的功能需不需要额外的数据,需要几个)
第四步:通过函数体实现函数功能
第五步:确定函数返回值
第六步:写函数说明文档

# 案例:定义一个函数求任意两个数的和
def sum_2(num1,num2):
    print(num1+num2)
# 案例:统计一个字符串中中文的个数
def count_chinese(str1):
    """
    (功能说明区)统计一个字符串中中文的个数
    :param str1: (参数说明)需要统计的字符串
    :return:(返回值说明)None
    """
    count = 0
    for x in str1:
        if '\u4e00' <= x <= '\u9fa5':
            count += 1
    print(count)
# 案例:提取两个字符串中公共字符
def common(str1,str2):
    result = set(str1)&set(str2)
    print(''.join(result))
# 案例:求10!
def jc(num):
    sum = 1
    for x in range(num,0,-1):
        sum *= x
    print(sum)

3.调用函数(使用函数)

1.语法:函数名(实参列表)
2.说明:
函数名 — 必须是已经定义好的函数的函数名(需要哪个函数的功能就调用哪个函数,就写哪个函数的函数名)
() — 固定写法
实参列表 — 实参就是通过形参从函数外部传递到函数内部的数据,以‘数据1 ,数据2,数据3’的形式存在,这儿的每一个数据就是一个实参,实参的个数由形参的个数决定
重要结论:定义函数的时候不执行函数体,调用函数的时候才会执行。调用多少次,函数体就会执行多少次
调用过程:每次调用函数的时候:1.先回到函数定义的位置2.传参(用实参给形参赋值)3.执行函数体4.确定返回值5.回到函数调用的位置接着往后执行。

4.参数

1.位置参数和关键字操作

根据调用函数的时候
1.位置参数 — 将多个数据用逗号隔开,让实参和形参从位置上一一对应
2.关键字参数 — 调用函数的时候, 以’形参名1 = 实参1 ,形参名2 = 实参2,…'形式存在的参数

注意:
a.如果位置参数和关键字参数一起使用,那么位置参数必须在关键字参数的前面
b.不管以什么样的方式传参,必须保证每个参数都有值,并且每个参数只有一个值。

def func1(a, b, c):
   print(f'a:{a}, b:{b}, c:{c}')

# 位置参数
func1(10, 20, 30)
func1(20, 10, 30)

2.关键字参数

# 关键字参数
func1(a=10, b=20, c=30)
func1(b=20, a=10, c=30)

# 位置参数和关键字参数一起用
func1(10, b=20, c=30)
# func1(a=10, 20, c=30)     # 报错!
# func1(20, a=10, c=30)     # 报错!

func1(10, c=30, b=20)
func1(10, 20, c=30)

2.参数默认值

1)定义函数的时候可以以’形参名=值’的方式给形参赋默认值,有默认值的参数在调用函数的时候可以不用传参。

2)定义函数的时候可以有的参数有默认值,有的参数没有默认值,但是没有默认值的参数必须在有默认值参数的前面。

def func2(a=1, b=2, c=3):
    # a = 10, b = 20
    print(f'a:{a}, b:{b}, c:{c}')


func2() #a=1, b=2, c=3
func2(10) #a:10, b:2, c:3
func2(10, 20) #a:10, b:20, c:3
func2(10, 20, 30) #a:10, b:20, c:30

# 如果要跳过前面有默认值的参数,直接给后面的参数传参,必须使用关键字参数
func2(b=200) #a:1, b:200, c:3
func2(c=300) #a:1, b:2, c:300

3.参数类型说明

定义函数的时候可以对参数类型进行说明:
1)没有默认值的参数:形参名:类型名
2)有默认值的参数,默认值的数据类型就是参数类型

def func7(str1: str, list1: list, str2=''):
    pass

4.返回值

1.什么是返回值

返回值就是从函数内部传递到函数外部的数据。
定义函数的时候,需不需要添加返回值的建议:看实现函数的功能有没有产生新的数据,如果有新的数据,就将新的数据作为返回值返回。
(函数内部产生的数据,如果不用返回值返回是无法在函数外部使用)

2.怎么确定函数返回值 —怎么将一个数据作为函数的返回值返回
返回值就是return关键字后面的表达式的值

return 数据
# 求两个数的和
def sum_2(num1, num2):
    # num1 = 10; num2 = 20
    result = num1 + num2    # result = 10 + 20
    return result     # return 30


# 求两个数的和以及平均值
def sum_average(num1, num2):
    s1 = num1 + num2
    avg = s1 / 2
    return s1, avg
3.怎么使用函数返回值

函数调用表达式的值就是函数的返回值(python中每个函数函数调用都是有结果,这个结果就是这个函数在调用的时候得到的返回值)

函数调用表达式 - 调用函数的语句

注意:每次调用函数的时候会执行函数体,并且获取函数返回值

result1 = print('hello world!')
print(f'result1:{result1}') #result1:None

result2 = sum([10, 20, 30])
print(f'result2:{result2}')
#result2:60

result3 = sum_2(10, 20)
print(f'result3:{result3}')
#result3:30

result4 = func1()
print(f'result4:{result4}')
#result4:None
def func2():
    return 100

func2()     # 100 相当于100能做的事情他都可以做

print(100)
print(func2())

a = 100
b = func2() #b=100
print(a, b)

print(100 + 200)
print(func2() + 200) #300

def func3():
    return 'abc'

print(func3()[-1])    # 'abc'[-1]
print(func3().split('b'))       # ['a', 'c']
print('---------------------------------------华丽的分割线------------------------------------')


def func4():
    print(200)
result = func4() #返回值:None
print(result) 


# 案例:定义函数,删除指定数字列表中所有的负数
# 方法1: 产生新的列表,需要返回值
def del_minus(nums: list):
    new_nums = []
    for x in nums:
        if x >= 0:
            new_nums.append(x)
    return new_nums


list1 = [10, -90, -89, 78, 2, 0]
result = del_minus(list1)
print(result)


# 方法2:直接删除原列表元素,不需要返回值
def del_minus2(nums: list):
    for x in nums[:]:
        if x < 0:
            nums.remove(x)

 
list1 = [10, -90, -89, 78, 2, 0]
del_minus2(list1)
print(list1)

4.return的作用

注意:return只能在函数体中使用

1.将数据作为函数的返回值返回

2.结束函数(执行函数体的时候如果遇到return,函数直接结束)

def func1():
    print('+++++++')
    return 100
    print('-------')
    print('=======')
print(func1())   # 100


def func2():
    print('+++++++')
    return
    print('-------')
    print('=======')
print(func2()) #None

案例

def func3(n):
    # n = 5
    for x in range(1, n):
        if x % 3 == 0:
            return x
        print(f'x:{x}')

print(func3(5))
#x:1
#x:2
#3 当x=3的时候执行if条件,遇到return 返回值为3  结束进程  后面的x:3不会再打印

二、函数进阶

1.变量作用域

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

2.全局变量和局部变量

1)全局变量

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

# a是全局变量
a = 100

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

print(f'外面使用a:{a}')
print(f'外面使用x:{x}')
print(f'外面使用b:{b}')
#外面使用a:100
#外面使用x:4
#外面使用b:200

#全局变量可以在任何地方使用,不管是在外面单独使用还是在函数里或者在循环中都可以使用

2)局部变量

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

# m 和 n 是局部变量
def func2(m):
    n = 1000
    print(f'函数里面m:{m}')
    print(f'函数里面n:{n}')
func2(100)

# print(f'函数外面m:{m}') #报错
# print(f'函数外面n:{n}') #报错

3.全局变量和局部变量的存储原理

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

2)调用函数的时候系统会自动为这个函数创建一个临时栈区间,用来保存在函数中产生的数据(局部变量
就是保存在函数对应的临时栈区间中的),函数对应的临时栈区间会在函数调用结束的时候自动释放。

3)在函数中可以通过关键字global修改局部变量的保存方式,让局部变量保存在全局栈区间中。

name = '小明'
def func3():
    # 在函数内部定义全局变量用global
    global num
    num = 10

    # 在函数内部修改全局变量的值用global
    global name
    name = '小花'
func3()

print(num) #10
print(name) # 小花

4.匿名函数

1.匿名函数 – 没有名字的函数

函数名 = lambda 形参列表:返回值

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

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

注意:

lambda 形参列表:返回值
函数名 = lambda 形参列表:返回值

#案例 定义一个匿名函数,求任意两个数的和
sum_2  = lambda num1=1,num2=2:num1+num2

result = sum_2(100, 200)
print(result) #300

result = sum_2(num1=10, num2=20)
print(result)#30

result = sum_2()
print(result)#3


#案例:定义一个匿名函数,判断指定的年是否是闰年
is_leap_year = lambda year : (year % 4 == 0 and year % 100 !=0 )or (year % 400 == 0)
print(is_leap_year(2000))#True

2.实参高阶函数

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

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

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

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

1.max函数
①max(序列) — 直接比较序列中元素的大小求最大值
②max(序列,key=函数) — 按照函数制定的规则比较序列中元素的大小获取最大值

函数的要求:
a.有且只有一个参数(这个参数代表序列中的每个元素)
b.有一个返回值(返回值就是比较对象)

#案例:求个位数最大的元素
nums = [83,67,19,1,22,80,77,93]
result = max(nums,key=lambda item:item%10)
print(result) #19

# 案例:求nums中绝对值最大的元素
nums = [8, -283, 89, 100, 82, -34]
result = max(nums, key=lambda item: item ** 2)
print(result)       # -283

# 案例:求students中分数最高的学生
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}
]
result = max(students, key=lambda item: item['score'])
print(result)

# 案例:按照分数的高度对学生从大到小排序
result = sorted(students, key=lambda item: item['score'], reverse=True)
print(result)

# 练习1:求list1中长度最长的字符串
list1 = ['你好', 'hello', 'how are you', 'thank you! and you?', '好好学习,天天向上']
result = max(list1, key=lambda item: len(item))
print(result)

# 练习2:求nums中十位数最小的元素
nums = [92, 129, 37, 99, 150, 501]
result = min(nums, key=lambda item: item // 10 % 10)
print(result)

# 练习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}
]
students.sort(key=lambda item: item['age'])
print(students)


# 练习4:求nums中各个位之和最大的元素
nums = [123, 78, 90, 201, 192, 330]
# 123  -> 1 + 2 + 3
# '123' -> '1'、'2'、'3', [1, 2, 3]

# 方法1:
result = max(nums, key=lambda item: sum([int(x) for x in str(item)]))
print(result)

# 方法2:
# 123 -> eval('1+2+3')
# 123 -> '123'  -> '+'.join('123') -> '1+2+3'
result = max(nums, key=lambda item: eval('+'.join(str(item))))
print(result)


# 方法3:
def t(item):
   sum1 = 0
   for x in str(item):
       sum1 += int(x)
   return sum1


result = max(nums, key=t)
print(result)

2.map函数 — 基于原序列中的元素创建一个新的序列
①map(函数,序列)

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

②map(函数,序列1,序列2)

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

③map(函数, 序列1, 序列2, 序列3,…)

# 案例1:将nums中所有的元素乘以10
nums = [19, 870, 34, 61, 78]
result = map(lambda item: item*10, nums)
print(list(result))     # [190, 8700, 340, 610, 780]

# 案例2:获取nums中所有元素的个位数
result = map(lambda item: item % 10, nums)
print(list(result))     # [9, 0, 4, 1, 8]

# 案例2:将nums1和nums2中相同位置上的元素相乘,得到一个新的序列
nums1 = [10, 20, 30, 10]
nums2 = [23, 30, 20, 55]
result = map(lambda i1, i2: i1 * i2, nums1, nums2)
print(list(result))     # # [230, 600, 600, 550]

# 练习1:获取names中每个人的姓
names = ['何晓东', '张三', '李四', '王五', '王二', '赵六']
result = map(lambda item: item[0], names)
print(list(result))

# 练习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',....]
result = map(lambda i1, i2: f'{i2}-{i1["name"]}', students, subjects)
print(list(result))

3.reduce函数 — 将序列中的元素合并成一个数据(基于原序列中所有的元素得到一个数据)
①reduce(函数,序列,初始值) — 按照函数制定的规则,将序列中的元素合并成一个数据

函数的要求:
a.有且只有两个参数(第一个参数指向初始值,第二个参数代表序列中的每个元素)
b.需要一个返回值(返回值就是合并规则)
初始值:累积求数值和,初始值是0
累积求数值乘积,初始值是1
字符串合并,初始值是空串

from functools import reduce

nums = [12, 301, 40, 55, 112]
# 案例1:12 + 301 + 40 + 55 + 112 的结果
# 0 + 12 + 301 + 40 + ... + 112
result = reduce(lambda x, item: x + item, nums, 0)
print(result)

# 案例:12 * 301 * 40 * 55 * 112 的结果
# 1 * 12 * 301 * ... * 112
result = reduce(lambda x, item: x * item, nums, 1)
print(result)

# 案例:123014055112  -> '123014055112'
# '' + '12' + '301' + '40' + ... + '112'
# '' + str(12) + str(301) +... + str(112)
result = reduce(lambda x, item: x + str(item), nums, '')
print(result)       # '123014055112'


# 练习:2+1+0+5+2
nums = [12, 301, 40, 55, 112]
# 0 + 2 + 1 + 0 + 5 + 2
# 0 + 12 % 10 + 301 % 10 + ... + 112 % 10
result = reduce(lambda x, item: x + item % 10, nums, 0)
print(result)

5.迭代器

1.什么是迭代器(iter)

1)迭代器是容器型数据类型(可以遍历,也转换成列表),无法直接提供一个迭代器,只能将其他序列转换成迭代器。
2)特点:
a.打印迭代器的时候无法查看到元素有哪些
b.无法通过len获取迭代器中元素的个数
c.如果要使用迭代器中的元素必须将元素从迭代器中取出来(取走),取走的元素会从迭代器中永远消失(用一个就少一个)

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

# 1)创建迭代器
i1 = iter('abc')
i2 = iter([10, 20, 30, 40])
i3 = iter((10, 1.23, 'abc', True, [10, 20]))

# 2)打印迭代器无法查看元素
print(i1)
print(i2)

# 3)迭代器无法统计个数
# print(len(i1))        # 报错!

2.获取迭代器中的元素

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

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

  1. 遍历迭代器
    for 变量 in 迭代器:
    循环体
print(next(i1))     # 'a'
print(next(i1))     # 'b'
print(next(i1))     # 'c'
# print(next(i1))     # 报错!

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

#遍历以后再转成列表,列表为空
for x in i3:
    print(x)
print(list(i3))     # []
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值