函数的定义:函数是为了完成某一特定功能的,函数是逻辑结构化和过程化的一种编程方法
函数的定义格式,函数一般都是有返回值的
#语法
#函数名要能反映其意义
def 函数名(参数1,参数2,参数3,...): '''注释''' 函数体 return 返回的值def test(x): ''' 函数功能:计算 2*x+1 正规军写函数前都会写注释,说明这个函数要实现的功能 :param x:整形数字 :return: 返回计算结果 ''' y=2*x+1 return y#函数的调用,需要传入参数print(test(4)) print(test) # 代表番函数的内存地址被打印出来
#结果 9 # 再定义一个test函数,那么本段程序有两个test函数,怎么执行? # Python 解释器是从上到下执行的,当执行到test()时,是不带参数的,所以该段程序是不带参数的test()函数
def test(): ''' 2*x+1 :param x:整形数字 :return: 返回计算结果 ''' x=3 y=2*x+1 return y a=test() print(a)
为什么要使用函数 ?
那如果不使用函数会有如下问题:
1、代码的组织结构不清晰,可读性差2、遇到重复的功能只能重复编写实现代码,代码冗余3、功能需要扩展时,需要找出所有实现该功能的地方修改之,无法统一管理且维护难度极大 所以,如果使用函数,将得到以下好处: 1.代码重用
2.保持一致性,易维护
3.可扩展性
过程:过程本身也是一个函数,只不过是没有返回值
def test01(): # test01就是一个过程,因为无返回值 msg = 'test01' print(msg)def test02(): # test02 是一个函数,因为有返回值 msg = 'test02' print(msg) return msgt1 = test01()t2 = test02()print(t1) # 结果 Noneprint(t2) # 结果 test02
def test03(): msg = 'test03' print(msg) return 1,2,3,4,'a',['alex'],{'name':'alex'},None # 返回值可以是一个,也可是多个
# 结果
test01
test02test03test03Nonetest02(1, 2, 3, 4, 'a', ['alex'], {'name': 'alex'}, None)
def test01(): passdef test02(): return 0def test03(): return 0, 10, 'hello', ['alex', 'lb'], { 'WuDaLang': 'lb'}t1 = test01()t2 = test02()t3 = test03()print('from test01 return is [%s]: ' % type(t1), t1)print('from test02 return is [%s]: ' % type(t2), t2)print('from test03 return is [%s]: ' % type(t3), t3)#结果from test01 return is []: Nonefrom test02 return is [ ]: 0from test03 return is [ ]: (0, 10, 'hello', ['alex', 'lb'], { 'WuDaLang': 'lb'})
总结:当一个函数/过程没有使用return显示的定义返回值时,python解释器会隐式的返回None,
所以在python中即便是过程也可以算作函数。
总结:
返回值数 = 0,返回None
返回数值 = 1 ,返回Object
返回数值 > 1,返回tuple
函数参数
1.形参变量只有在被调用时才分配内存单元,在调用结束时,即刻释放所分配的内存单元。因此,形参只在函数内部有效。函数调用结束返回主调用函数后则不能再使用该形参变量
2.实参可以是常量、变量、表达式、函数等,无论实参是何种类型的量,在进行函数调用时,它们都必须有确定的值,以便把这些值传送给形参。因此应预先用赋值,输入等办法使参数获得确定值。
3.位置参数和关键字(标准调用:实参与形参位置一一对应;关键字调用:位置无需固定)
4.默认参数
5.参数组
def calc(x,y): #x=2,y=3 res=x**y return res # 函数如果碰到return,该函数就的结束了,即使你后面有多个return,也不会执行;如果需要有多个return,需要通过 条件判断的形式进行 return yres=calc(2,3)print(res)# 形参变量只有在被调用时才分配内存单元,在调用结束时,即刻释放所分配的内存单元。因此,形参只在函数内部有效。函数调用结束返回主调用函数后则不能再使用该形参变量,# 所以,此时想打印x,y,程序会报错print(x)print(y)# 结果NameError: name 'x' is not defined
def test(x,y,z):#x=1,y=2,z=3 print(x) print(y) print(z)#位置参数,必须一一对应,缺一不行多一也不行test(1,2) # TypeError: test() missing 1 required positional argument: 'z'test(1,2,3) # 1,2,3, 位置一一对应的关系,传值,成就就正常#关键字参数,无须一一对应,缺一不行多一也不行test(y=1,x=3,z=4)#位置参数一定要在关键字参数左边,不可混合使用test(1,y=2,3)#报错, SyntaxError: positional argument follows keyword argument, 位置参数跟在了关键字参数的后面。print(test(1,3,y=2))#报错, TypeError: test() got multiple values for argument 'y' ,就是参数y给传了多个参数test(1,3,z=2,y=4)#报错,test() got multiple values for argument 'y',就是参数y给传了多个参数test(z=2,1,3)#报错,位置参数必须在关键字参数的左边test(1,3,z=2)
# 默认参数def handle(x,type='mysql'): print(x) print(type)handle('hello')# 结果hellomysqlhandle('hello',type='sqlite') # 默认参数传参写关键字, # 结果hellosqlitehandle('hello','sqlite') # 默认参数传参不写关键字# 结果hellosqlite
# 还有, 比如安装软件, 有些功能默认是安装的,有些功能默认不是安装的 def install(func1=False,func2=True,func3=True): pass
# 参数组: 两个星** 表示字典字典, 一个星表示*列表# 一个*后面,最好按照规定就写args,实参的第一个参数就给形参的第一个参数,其他的实参都默认按照列表的方式处理,传给args了def test(x,*args): print(x) print(args)# test(1)# 结果1()test(1,2,3,4,5)# 结果1(2, 3, 4, 5)test(1,{ 'name':'alex'})#结果1({ 'name': 'alex'},)test(1,['x','y','z']) # 如果列表前没加*, 就表示把列表作为一个整体传给args参数的第一个元素# 结果# 1# (['x', 'y', 'z'],)test(1,*['x','y','z']) # 如果列表前加*, 就与*args对应起来了,表示把列表的元素遍历一遍,依次赋给args参数# 结果1('x', 'y', 'z')test(1,*('x','y','z'))# 结果1('x', 'y', 'z')上面示例的实参的传递方式都是按照位置的方式进行的。下面示例的实参的传递方式按照关键字的方式# 两个**后面最好也按照规定就写kwargs,def test(x,**kwargs): print(x) print(kwargs) test(1,y=2,z=3)#结果1{ 'y': 2, 'z': 3}test(1,1,2,2,2,2,2,y=2,z=3)# 结果TypeError: test() takes 1 positional argument but 7 were giveny前面的是按照位置参数的方式传的,必须一一对应,但是位置形参只有一个,实参却传递了7个test(1,y=2,z=3,z=3)#会报错 :一个参数不能传两个值# 结果SyntaxError: keyword argument repeated 关键字参数重复了下面的示例形参表示方法,意味着该函数无所不能,可以接受任意形式的参数 def test(x,*args,**kwargs): print(x) print(args) print(kwargs) test(1,1,2,1,1,11,1,x=1,y=2,z=3) #报错# 结果TypeError: test() got multiple values for argument 'x'test(1,1,2,1,1,11,1,y=2,z=3)1 -->给了x(1, 2, 1, 1, 11, 1) -->给了 args{ 'y': 2, 'z': 3} -->给了 kwargsdef test(x,*args,**kwargs): print(x) print(args) print(kwargs,kwargs['z'])test(1,1,2,1,1,11,1,y=2,z=3)# 结果1(1, 2, 1, 1, 11, 1){ 'y': 2, 'z': 3} 3def test(x,*args,**kwargs): print(x) print(args) print(kwargs,kwargs.get('y'))test(1,*[1,2,3],**{ 'y':1})#结果1(1, 2, 3){ 'y': 1} 1