一、说明
1、函数的作用:
a.代码组织结构清晰,可读性强;
b.遇到重复的功能只用写一个函数即可,其它调用即可;
c.功能需要扩展时,修改功能函数即可。
二、函数的种类
1、内置函数
为了方便开发,针对一些简单的功能,Python解释器已经为我们定义好的函数即内置函数。如:max(),min(),len()....
2、自定义函数
由于内置函数提供的功能有限,我们需要自定义函数来实现某种功能,遇到相同场景调用函数即可。
三、定义函数
1、语法
1 def 函数名(参数1,参数2,参数3,...): 2 '''注释''' 3 函数体 4 return 返回的值 5 6 #函数名要能反映其意义
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
1 def auth(user:str,password:str)->int: 2 ''' 3 auth function 4 :param user: 用户名 5 :param password: 密码 6 :return: 认证结果 7 ''' 8 if user == 'egon' and password == '123': 9 return 1 10 # print(auth.__annotations__) #{'user': <class 'str'>, 'password': <class 'str'>, 'return': <class 'int'>} 11 12 user=input('用户名>>: ').strip() 13 pwd=input('密码>>: ').strip() 14 res=auth(user,pwd) 15 print(res)
四、函数采用先定义,再调用的原则
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
1 函数即“变量”,“变量”必须先定义后引用。未定义而直接引用函数,就相当于在引用一个不存在的变量名 2 #测试一 3 def foo(): 4 print('from foo') 5 bar() 6 foo() #报错 7 8 #测试二 9 def bar(): 10 print('from bar') 11 def foo(): 12 print('from foo') 13 bar() 14 foo() #正常 15 16 #测试三 17 def foo(): 18 print('from foo') 19 bar() 20 21 def bar(): 22 print('from bar') 23 foo() #会报错吗? 24 25 26 #结论:函数的使用,必须遵循原则:先定义,后调用 27 #我们在使用函数时,一定要明确地区分定义阶段和调用阶段 28 29 #定义阶段 30 def foo(): 31 print('from foo') 32 bar() 33 def bar(): 34 print('from bar') 35 #调用阶段 36 foo()
PS:函数在定义时只检测语法,不执行代码。
五、定义函数的三种形式
无参:应用场景仅仅只是一些操作,比如用户交互、打印
有参:需要根据外部传进来参数才能执行相应的逻辑,比如统计长度,求最大值最小值
空函数:设计代码结构
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
1 #定义阶段 2 def tell_tag(tag,n): #有参数 3 print(tag*n) 4 5 def tell_msg(): #无参数 6 print('hello world') 7 8 #调用阶段 9 tell_tag('*',12) 10 tell_msg() 11 tell_tag('*',12) 12 13 ''' 14 ************ 15 hello world 16 ************ 17 ''' 18 19 #结论: 20 #1、定义时无参,意味着调用时也无需传入参数 21 #2、定义时有参,意味着调用时则必须传入参数 22 23 24 # 空函数 25 def auth(user,password): 26 ''' 27 auth function 28 :param user: 用户名 29 :param password: 密码 30 :return: 认证结果 31 ''' 32 pass 33 34 def get(filename): 35 ''' 36 :param filename: 37 :return: 38 ''' 39 pass 40 41 def put(filename): 42 ''' 43 :param filename: 44 :return: 45 ''' 46 def ls(dirname): 47 ''' 48 :param dirname: 49 :return: 50 ''' 51 pass 52 53 #程序的体系结构立见
六、函数的调用
1、函数的调用:函数名加括号
a.语句形式:foo()
b.表达式形式:3*len('hello')
c.当中另外一个函数的参数:range(len('hello')
2、函数的返回值:return 一个值/多个值
3、函数的参数:实参与形参(形参即变量名,实参即变量值。)
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
1 #1、位置参数:按照从左到右的顺序定义的参数 2 位置形参:必选参数 3 位置实参:按照位置给形参传值 4 5 #2、关键字参数:按照key=value的形式定义的实参 6 无需按照位置为形参传值 7 注意的问题: 8 1. 关键字实参必须在位置实参右面 9 2. 对同一个形参不能重复传值 10 11 #3、默认参数:形参在定义时就已经为其赋值 12 可以传值也可以不传值,经常需要变得参数定义成位置形参,变化较小的参数定义成默认参数(形参) 13 注意的问题: 14 1. 只在定义时赋值一次 15 2. 默认参数的定义应该在位置形参右面 16 3. 默认参数通常应该定义成不可变类型 17 18 19 #4、可变长参数: 20 可变长指的是实参值的个数不固定 21 而实参有按位置和按关键字两种形式定义,针对这两种形式的可变长,形参对应有两种解决方案来完整地存放它们,分别是*args,**kwargs 22 23 ===========*args=========== 24 def foo(x,y,*args): 25 print(x,y) 26 print(args) 27 foo(1,2,3,4,5) 28 29 def foo(x,y,*args): 30 print(x,y) 31 print(args) 32 foo(1,2,*[3,4,5]) 33 34 35 def foo(x,y,z): 36 print(x,y,z) 37 foo(*[1,2,3]) 38 39 ===========**kwargs=========== 40 def foo(x,y,**kwargs): 41 print(x,y) 42 print(kwargs) 43 foo(1,y=2,a=1,b=2,c=3) 44 45 def foo(x,y,**kwargs): 46 print(x,y) 47 print(kwargs) 48 foo(1,y=2,**{'a':1,'b':2,'c':3}) 49 50 51 def foo(x,y,z): 52 print(x,y,z) 53 foo(**{'z':1,'x':2,'y':3}) 54 55 ===========*args+**kwargs=========== 56 57 def foo(x,y): 58 print(x,y) 59 60 def wrapper(*args,**kwargs): 61 print('====>') 62 foo(*args,**kwargs) 63 64 #5、命名关键字参数:*后定义的参数,必须被传值(有默认值的除外),且必须按照关键字实参的形式传递 65 可以保证,传入的参数中一定包含某些关键字 66 def foo(x,y,*args,a=1,b,**kwargs): 67 print(x,y) 68 print(args) 69 print(a) 70 print(b) 71 print(kwargs) 72 73 foo(1,2,3,4,5,b=3,c=4,d=5) 74 结果: 75 2 76 (3, 4, 5) 77 3 78 {'c': 4, 'd': 5}
七、练习题
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
1 # 1、写函数,,用户传入修改的文件名,与要修改的内容,执行函数,完成批了修改操作 2 def modify_file(filename,old,new): 3 import os 4 with open(filename,'r',encoding='utf-8') as read_f, 5 open('.bak.swap','w',encoding='utf-8') as write_f: 6 for line in read_f: 7 if old in line: 8 line=line.replace(old,new) 9 write_f.write(line) 10 os.remove(filename) 11 os.rename('.bak.swap',filename) 12 modify_file('/Users/jieli/PycharmProjects/爬虫/a.txt','alex','SB') 13 # 2、写函数,计算传入字符串中【数字】、【字母】、【空格] 以及 【其他】的个数 14 def check_str(msg): 15 res={ 16 'num':0, 17 'string':0, 18 'space':0, 19 'other':0, 20 } 21 for s in msg: 22 if s.isdigit(): 23 res['num']+=1 24 elif s.isalpha(): 25 res['string']+=1 26 elif s.isspace(): 27 res['space']+=1 28 else: 29 res['other']+=1 30 return res 31 32 res=check_str('hello name:aSB passowrd:alex3714') 33 print(res) 34 35 # 3、写函数,判断用户传入的对象(字符串、列表、元组)长度是否大于5。 36 # 略 37 # 4、写函数,检查传入列表的长度,如果大于2,那么仅保留前两个长度的内容,并将新内容返回给调用者。 38 def func1(seq): 39 if len(seq) > 2: 40 seq=seq[0:2] 41 return seq 42 print(func1([1,2,3,4])) 43 # 5、写函数,检查获取传入列表或元组对象的所有奇数位索引对应的元素,并将其作为新列表返回给调用者。 44 def func2(seq): 45 return seq[::2] 46 print(func2([1,2,3,4,5,6,7])) 47 # 6、写函数,检查字典的每一个value的长度,如果大于2,那么仅保留前两个长度的内容,并将新内容返回给调用者。 48 # dic = {"k1": "v1v1", "k2": [11,22,33,44]} 49 # PS:字典中的value只能是字符串或列表 50 def func3(dic): 51 d={} 52 for k,v in dic.items(): 53 if len(v) > 2: 54 d[k]=v[0:2] 55 return d 56 print(func3({'k1':'abcdef','k2':[1,2,3,4],'k3':('a','b','c')}))