函数基础
1.定义函数
1. 什么是函数
-
概念
函数就是实现某一特定功能的代码的封装 -> 一个函数对应一个功能(函数存储的功能)
-
分类(按照函数由谁创建的进行分类)
- 系统函数 - 有Python语言已经创建好的函数(Python自带的函数),例如:print、input、type、id、max、min、sorted、sum等
- 自定义函数 - 由程序员自己创建的函数
2.定义函数(造机器)
-
语法:
def 函数名(形参列表): 函数说明文档 函数体
-
说明:
-
def - 关键字;固定写法
-
函数名 - 由程序员自己命名;
要求:是标识符,不能是关键字
规范:见名知义(看到函数名就大概知道函数的功能是什么)、
不使用系统的函数名、类名、模块名、
字母都小写,多个单词用下划线隔开 -
(): - 固定写法
-
形参列表 - 以’变量名1, 变量名2, 变量名3,…'的形式存在,这儿每一个变量就是一个形参;形参可以一个都没有,也可以有多个。形参可以将函数外部的数据传递到函数内部;定义函数的时候需不需要形参需要几个看实现函数的功能需不需要额外的数据,需要几个
-
函数说明文档 - 本质就是和def保持一个缩进多行注释;用来对函数的功能、参数和返回值进行说明的 (说明书)
-
函数体 - 和def保持一个缩进的一条或者多条语句,本质就是实现函数功能的代码。 (电路结构和机械机构)
-
# 练习1: 写一个函数求两个数的和
def sum2(num1, num2):
"""
(功能说明区)求任意两个数的和
:param num1: (参数说明) 数字1
:param num2: 数字1
:return: (返回值说明) None
"""
print(num1 + num2)
sum2(10, 20)
# 练习2: 写一个函数统计指定字符串中数字字符的个数
def count_number(str1):
"""统计数字字符的个数"""
count = 0
for x in str1:
if x.isdigit():
count += 1
print(count)
count_number('ajhf1238山海经02')
count_number('数的2791')
# 练习3:定义一个函数获取任意一个整数(正负都可以)的十位数
def get_tens_digit(num):
"""获取十位数"""
if num < 0:
num *= -1
print(num // 10 % 10)
get_tens_digit(123)
get_tens_digit(-132)
# 练习4:定义一个函数,获取指定列表中所有数字元素
def get_numbers(list1):
"""获取数字元素"""
new_list = []
for x in list1:
if type(x) in (int, float):
new_list.append(x)
print(new_list)
get_numbers([19, '是d', 23.8, True, None])
# 练习5:定义一个函数,获取两个字符串的公共部分
def get_common_char(str1, str2):
"""获取两个字符串的公共部分"""
result = set(str1) & set(str2)
print(''.join(result))
get_common_char('abcn', '123ba92')
# 练习6:定义一个函数交换字典的键和值
def change_key_value(dict1):
new_dict = {}
for key in dict1:
new_dict[dict1[key]] = key
print(new_dict)
change_key_value({'a': 10, 'b': 20, 'c': 30})
2.调用函数
-
重要结论:定义函数的时候不会执行函数,调用的时候才会执行
-
语法:
函数名(实参列表)
-
说明:
-
函数名 - 需要哪个函数的功能就调用哪个函数,想要调用哪个函数就写哪个函数的函数名
注意:这儿的函数名必须是已经定义过的函数的函数名
-
() - 固定写法
-
实参列表 - 以 ‘数据1, 数据2, 数据3,…’ 的形式存在;实参就是真正通过形参传递到函数内部的数据;实参的个数由形参决定,默认情况被调用的函数有多少个形参调用的时候就需要多少个实参
-
-
函数调用过程:
当代码执行到函数调用语句的时候:
第一步:回到函数定义的位置
第二步:传参(用实参给形参赋值的过程),传参的时候必须保证每个形参都有值
第三步:执行函数体
第四步:确定返回值
第五步:回到函数调用的位置,接着往后执行
3.函数的参数
1.位置参数和关键字参数 - 根据实参的传递方式不同将函数的实参分为这两种
-
位置参数
调用函数的时候直接将多个数据用逗号隔开,实参和形参从位置上一一对应
-
关键字参数
调用函数的时候,在数据前面加上’形参名=’,实参和形参由形参名来对应
-
两种参数混合用
要求必须保证位置参数在关键字参数的前面
def func1(x, y, z):
print(f'x:{x}, y:{y}, z:{z}')
func1(10, 20, 30)
func1(20, 10, 30)
func1(x=100, y=200, z=300)
func1(z=3, x=1, y=2)
func1(10, y=20, z=30)
func1(10, z=30, y=20)
func1(10, 20, z=30)
# func1(10, b=20, 30) # 报错! SyntaxError: positional argument follows keyword argument
2.参数默认值
- 定义函数的时候可以给形参赋默认值,调用函数的时候已经有默认的参数可以不用传参,直接使用默认值。
- 如果是给部分参数赋默认值,必须保证没有默认值的参数在有默认值参数的前面
3.参数类型说明
参数类型说明:定义函数的时候指定参数类型
-
没有默认值的参数添加类型说明
形参名:数据类型
-
有默认值的参数,默认值的类型就是参数的类型
4.不定长参数
带*的不定长参数:
- 在形参前面加*,那么这个参数就变成了一个元组,用来接收对应的所有的实参(实参是元组中的元素)
- 记住:如果函数参数在带*的参数的后面,那么后面的这些参数在调用的时候必须使用关键字参数
4.函数的返回值
什么是返回值
-
意义:
- 返回值就是从函数内部传递到函数外部的数据
-
怎么确定返回值(怎么将函数内部的数据作为返回值传递到函数外部):
- 在函数体中,将需要返回的数据放到return的后面;
- return后面的值是多少,函数的返回值就是多少,如果没有return,返回值None
-
怎么获取返回值(在函数外部怎么获取从函数内部传递出来的数据):
- 在函数外部获取函数调用表达式的结果;
- 函数调用表达式的值就是函数的返回值
-
什么时候需要返回值:
- 如果实现函数的功能产生了新的数据,将新的数据作为返回值返回
def sum2(n1, n2):
# n1 = 10; n2 = 20
result = n1 + n2 # result = 30
print(f'内部:{result}')
return result # return 30
a = sum2(10, 20) # 函数调用表达式的值就是函数的返回值
print(f'a:{a}')
作业
-
编写一个函数,交换指定字典的key和value。
例如:dict1={'a':1, 'b':2, 'c':3} --> dict1={1:'a', 2:'b', 3:'c'}
def change_key_value(dict1: dict): dict0 = {} for key, value in dict1.items(): dict0.setdefault(value, key) return dict0 a = change_key_value({'a': 1, 'b': 2, 'c': 3}) print(a)
-
编写一个函数,提取指定字符串中所有的字母,然后拼接在一起产生一个新的字符串
例如: 传入'12a&bc12d-+' --> 'abcd'
def get_alphabet(str1: str): str0 = '' for x in str1: if 'a' <= x <= 'z' or 'A' <= x <= 'Z': str0 += x return str0 a = get_alphabet('1acXS3w4') print(a)
-
写一个自己的capitalize函数,能够将指定字符串的首字母变成大写字母
例如: 'abc' -> 'Abc' '12asd' --> '12asd'
def turn_majuscule(str1: str): for x, y in enumerate(str1): if 'a' <= y <= 'z': result = str1.replace(y, chr(ord(y) - 32), x+1) break return result a = turn_majuscule('12abc') print(a)
-
写一个自己的endswith函数,判断一个字符串是否已指定的字符串结束
例如: 字符串1:'abc231ab' 字符串2:'ab' 函数结果为: True 字符串1:'abc231ab' 字符串2:'ab1' 函数结果为: False
def judge_ending(str1: str, str0: str): return str1[-len(str0):] == str0 judge_ending('abc231ab', 'ab1')
-
写一个自己的isdigit函数,判断一个字符串是否是纯数字字符串
例如: '1234921' 结果: True '23函数' 结果: False 'a2390' 结果: False
def judge_pure_number(str1: str): for x in str1: if x < '0' or x > '9': return(False) return True judge_pure_number('123xd')
-
写一个自己的upper函数,将一个字符串中所有的小写字母变成大写字母
例如: 'abH23好rp1' 结果: 'ABH23好RP1'
def minuscule_turn_majuscule(str1: str): str0 = '' for x in str1: if 'a' <= x <= 'z': x = chr(ord(x) - 32) str0 += x return str0 a = minuscule_turn_majuscule('abH23好rp1') print(a)
-
写一个自己的rjust函数,创建一个字符串的长度是指定长度,原字符串在新字符串中右对齐,剩下的部分用指定的字符填充
例如: 原字符:'abc' 宽度: 7 字符:'^' 结果: '^^^^abc' 原字符:'你好吗' 宽度: 5 字符:'0' 结果: '00你好吗'
# 方法一: def rjust_function(str1: str, width: int, element): result = '' for x in range(width - len(str1)): result += element result += str1 return result a = rjust_function('你好吗', 5, '0') print(a) # 方法二: def rjust(str1: str, width: int, char: str): """ :param str1: 原字符串 :param width: 新的长度 :param char: 填充字符,需要一个长度是1的字符串 :return: 填充后的新字符串 """ if len(char) != 1: raise TypeError len1 = len(str1) if width <= len1: return str1 return char * (width - len1) + str1 print(rjust('sa', 5, 'x'))
-
写一个自己的index函数,统计指定列表中指定元素的所有下标,如果列表中没有指定元素返回-1
例如: 列表: [1, 2, 45, 'abc', 1, '你好', 1, 0] 元素: 1 结果: 0,4,6 列表: ['赵云', '郭嘉', '诸葛亮', '曹操', '赵云', '孙权'] 元素: '赵云' 结果: 0,4 列表: ['赵云', '郭嘉', '诸葛亮', '曹操', '赵云', '孙权'] 元素: '关羽' 结果: -1
def index(list1: list, item): indexs = [] for i, x in enumerate(list1): if x == item: indexs.append(i) if indexs: return ','.join(str(x) for x in indexs) return -1 print(index([10, 20, 30, 10, 40], 100))
-
写一个自己的len函数,统计指定序列中元素的个数
例如: 序列:[1, 3, 5, 6] 结果: 4 序列:(1, 34, 'a', 45, 'bbb') 结果: 5 序列:'hello w' 结果: 7
def count_number(sequence): count = 0 for x in sequence: count += 1 return count a = count_number('hell0 wo') print(a)
-
写一个自己的max函数,获取指定序列中元素的最大值。如果序列是字典,取字典值的最大值
例如: 序列:[-7, -12, -1, -9] 结果: -1 序列:'abcdpzasdz' 结果: 'z' 序列:{'小明':90, '张三': 76, '路飞':30, '小花': 98} 结果: 98
def max2(seq): if type(seq) == dict: seq = list(seq.values()) else: seq = list(seq) m = seq[0] for x in seq[1:]: if x > m: m = x return m print(max2([10, 23, 78, 2])) print(max2({230, 9, 7281, 90})) print(max2({'xiaoming': 90, '小花': 89, '小红': 76, '小黑': 98}))
-
写一个函数实现自己in操作,判断指定序列中,指定的元素是否存在
例如: 序列: (12, 90, 'abc') 元素: '90' 结果: False 序列: [12, 90, 'abc'] 元素: 90 结果: True
def judge_in(sequence, element): for x in sequence: if x == element: return True return False judge_in([12, 90, 'abc'], 90)
-
写一个自己的replace函数,将指定字符串中指定的旧字符串转换成指定的新字符串
例如: 原字符串: 'how are you? and you?' 旧字符串: 'you' 新字符串:'me' 结果: 'how are me? and me?'
def replace(str1: str, old: str, new: str): # 方法一: # new_str = new.join(str1.split(old)) # return new_str # 方法二: index = 0 old_len = len(old) new_str = '' while index < len(str1): if str1[index: index+old_len] == old: new_str += new index += old_len else: new_str += str1[index] index += 1 return new_str print(replace('how are you? and you?', 'you', 'me'))