函数的定义
函数在数学上的定义:y=f(x) ,y是x的函数,x是自变量。y=f(x0, x1, ..., xn)
python的函数是由若干语句组成的语句块、函数名称、参数列表构成,它是组织代码的最小单元, 它能完成一定的功能 。
在Python中,用关键字def关键字来定义函数,如下:
def greet_hello(): print("hello!") greet_hello()
关键字def后面是函数名, 它其实就是一个标识符。后面的括号必须有,里面可以有参数。函数定义,只是声明了一个函数,它不会被执行,需要调用,函数的调用很简单,直接函数名加括号就调用了。定义需要在调用前,也就是说调用时,已经被定义过了,否则抛NameError异常。调用时写的参数是实际参数,是实实在在传入的值,简称实参。
函数名就是标识符,命名要求一样,同时语句块必须缩进,约定4个空格,python中的函数如果没有return语句,隐式会返回一个None值,定义中的参数列表称为形式参数,只是一种符号表达。而调用的时候写的参数是实际参数,是实实在在传入的值,简称实参。
def add(x, y): result = x+y return result out = add(4,5) print(out) 结果为: 9
callable(add)
结果为:True
上面只是一个函数的定义,有一个函数叫做add,接收2个参数,然后计算的结果,通过返回值来返回。
调用的时候通过函数名add加2个参数,同时返回值可使用变量接收。
定义需要在调用前,也就是说调用时,已经被定义过了,否则抛NameError异常
函数是可调用的对象,callable()
上面的代码可以复用,不仅仅是对数字可以相加,也可以对字符串相加。不需要事先声明。
函数的作用
- 功能封装,便于后面复用。
- 减少冗余代码。
- 代码更易读、简洁美观。
函数的返回值
首先先来看两个例子:
def showplus(x): print(x) return x+1 showplus(5) def showplus(x): print(x) return x+1 print(x+1) showplus(5) 结果都为: 5 6
上面的例子可以看到,return后,后面的语句就不会再执行了。
函数的返回值一般有四种情况:
- 没有返回值,默认会返回None。
- 直接一个return,默认返回None。
- 返回一个值。
- 返回多个值,返回多个值其实也相当于返回一个,返回的多个值用元组接收。可以返回任意数据类型的值。可以使用解构来提取值。
返回一个值
def mylen(s1): """计算s1的长度""" length = 0 for i in s1: length = length+1 return length #函数调用 str_len = mylen("abc") print('str_len : %s'%str_len)
结果为3
返回多个值
def ret_demo1(): '''返回多个值''' return 1,2,3,4 def ret_demo2(): '''返回多个任意类型的值''' return 1,['a','b'],3,4 ret1 = ret_demo1() print(ret1) ret2 = ret_demo2() print(ret2)
结果为:
(1, 2, 3, 4) (1, ['a', 'b'], 3, 4)
应当注意的是,函数执行时,一旦遇到return,就会结束整个函数。
def ret_demo(): print(111) return print(222) ret = ret_demo() print(ret) 结果为1111 None
可以看到,后面return后面的语句没有执行了。
多条return语句
def showplus(x): print(x) return x + 1 return x + 2 showplus(5) 结果为: 5 6 def guess(x): if x > 3: return "> 3" else: return "<= 3" print(guess(10)) 结果为: > 3
def fn(x): for i in range(x): if i > 3: return i else: print("{} is not greater than 3".format(x)) print(fn(5)) print(fn(3)) 结果为: 4 3 is not greater than 3 None
所以,Python函数使用return语句返回“返回值”,所有的函数都有返回值,如果没有return语句,会隐式的调用return none。同时应该注意return语句并不一定是函数语句块的最后一句。
一个函数可以有多个return语句,但是只有一条可以被执行。如果没有一条return语句被执行到,就会隐式调用return None。
如果有必要,可以显示调用return none,可以简写成为return。
如果函数执行了return语句,函数就会返回,当前被执行的return语句之后的其他语句就不会被执行了。
所以return的作用就是结束函数调用并返回值。
函数的参数
参数分为形参和实参,函数定义时为形参,函数调用时为实参。参数调用时传入的参数要和定义的个数相匹配(可变参数例外)
传递多个参数
def mymax(x,y): the_max = x if x > y else y return the_max ma = mymax(10,20) print(ma)
结果:20
位置实参
位置参数也就是按照参数定义顺序传入实参,传入的实参个数需要和定义的形参个数匹配。比如def f(x, y, z) 调用使用 f(1, 3, 5) ,此时是一一对应的。
关键字实参
def f(x, y, z) 调用使用 f(x=1, y=3, z=5) ,使用形参的名字来出入实参的方式,如果使用了形参名字,那么传参顺序就可和定义顺序不同。
位置实参和关键字实参混合
应当注意,位置参数必须在关键字参数的前面,位置参数是按位置对应的。同时,对于一个形参只能赋值一次。f(y=5, z=6, 2) 像这样的传参会报错。
def mymax(x,y): #此时x = 10,y = 20 print(x,y) the_max = x if x > y else y return the_max ma = mymax(10,y = 20) print(ma)
默认值参数
默认值参数也就是在定义的时候,给参数一个默认值,也叫缺省值。参数的默认值可以在未传入足够的实参的时候,对没有给定的参数赋值为默认值。默认值参数必须要在后面,不然会报错。
它在定义时,在形参后跟上一个值。 参数非常多的时候,并不需要用户每次都输入所有的参数,简化函数调用 。
def add(x=4, y=5): return x+y add(6, 10) 结果为: 16 add(6, y=7) 结果为: 13 add(x=5) 结果为: 10 add() 结果为: 9 add(y=7) 结果为: 11 add(x=5, 6) 结果为: File "<ipython-input-18-a7fe3f3b9351>", line 1 add(x=5, 6) ^ SyntaxError: positional argument follows keyword argument add(y=8,4) 结果为: File "<ipython-input-20-013636d44ec1>", line 1 add(y=8,4) ^ SyntaxError: positional argument follows keyword argument add(x=5, y=6) 结果为: 11 add(y=5, x=6) 结果为: 11
给他们传参的时候跟上面的关键字和默认值传参规则一样。
#定义一个函数login,参数名称为host、port、username、password def login(host='127.0.0.1',port='8080',username='xpc',password='xpc199151'): print('{}:{}@{}/{}'.format(host, port, username, password)) login() login('127.0.0.1', 80, 'tom', 'tom') login('127.0.0.1', username='root') login('localhost', port=80,password='com') login(port=80, password='magedu', host='www') 结果为: 127.0.0.1:8080@xpc/xpc199151 127.0.0.1:80@tom/tom 127.0.0.1:8080@root/xpc199151 localhost:80@xpc/com www:80@xpc/magedu
应该注意的是,如果默认值传参的是一个可变数据类型。后期的结果会保留前面的结果。
def defult_param(a,l = []): l.append(a) print(l) defult_param('alex') defult_param('egon')
结果为:
['alex'] ['alex', 'egon']
可变参数
可变参数分为位置参数的可变参数和关键字参数的可变参数。
位置参数的可变参数*
现在有一个需求,有多个数,但又不知道具体有几个数字,需要累加求和,这个时候需要传入一个可迭代对象,迭代元素求和。
def add(nums): sum = 0 for x in nums: sum += x return sum print(add([1,3,5])) add((2,4,6)) 结果为: 9 12
而可变位置参数,可以一个形参接受任意多个实参,按位置传值多余的参数都由args统一接收,保存成一个元组的形式。
def mysum(*args):
print(args) the_sum = 0 for i in args: the_sum+=i return the_sum the_sum = mysum(1,2,3,4) print(the_sum)
(1, 2, 3, 4) 10
关键字参数的可变参数**
形参前使用**符号,表示可以接收多个关键字参数,收集的实参名称和值组成一个字典 。
def stu_info(**kwargs): print(kwargs) print(kwargs['name'],kwargs['sex']) stu_info(name = 'alex',sex = 'male')
结果为:
{'name': 'alex', 'sex': 'male'} alex male
def showconfig(**kwargs):
for k,v in kwargs.items():
print('{} = {}'.format(k, v))
showconfig(host='127.0.0.1',port='8080',username='xpc',password='12345')
结果为:
host = 127.0.0.1 port = 8080 username = xpc password = 12345
位置可变参数和关键字可变参数的混合
混合使用参数的时候,可变参数要放到参数列表的最后,普通参数需要放到参数列表前面,位置可变参数需要在关键字可变参数之前。
def showconfig(username, *args, **kwargs): print(username) print(args) for k,v in kwargs.items(): print("{}={}".format(k,v)) showconfig("xpc") 结果为: xpc () showconfig("xpc",1,2,3,45) 结果为: xpc (1, 2, 3, 45) showconfig("xpc",a=1,b=2,c=3) 结果为: xpc () a=1 b=2 c=3
def showconfig(username,password, **kwargs,*args): print(username) print(args) for k,v in kwargs.items(): print("{}={}".format(k,v)) 结果为: File "<ipython-input-48-b281bd3c7e8c>", line 1 def showconfig(username,password, **kwargs,*args): ^ SyntaxError: invalid syntax
所以,可变参数有位置可变参数和关键字可变参数,位置可变参数在形参前使用一个星号* ,关键字可变参数在形参前使用两个星号** ,位置可变参数和关键字可变参数都可以收集若干个实参,位置可变参数收集形成一个tuple,关键字可变参数收集形成一个dict,
混合使用参数的时候,可变参数要放到参数列表的最后,普通参数需要放到参数列表前面,位置可变参数需要在关键字可变参数之前。
def fn(x, y, *args, **kwargs): print(x) print(y) print(args) print(kwargs) fn(3,5,7,9,10,a=1,b='python') 3 5 (7, 9, 10) {'a': 1, 'b': 'python'} fn(3,5) 3 5 () {} fn(3,5,7) 3 5 (7,) {} fn(3,5,a=1,b='python') 3 5 () {'a': 1, 'b': 'python'} fn(7,9,y=5,x=3,a=1,b='python') TypeError: fn() got multiple values for argument 'y'
keyword-only参数
当位置可变参数放在了最前面,后面的普通参数变成了keyword-only参数。
def fn(*args, x, y, **kwargs): print(x) print(y) print(args) print(kwargs) fn(3,5) TypeError: fn() missing 2 required keyword-only arguments: 'x' and 'y' fn(3,5,7) TypeError: fn() missing 2 required keyword-only arguments: 'x' and 'y' fn(3,5,a=1,b='python') TypeError: fn() missing 2 required keyword-only arguments: 'x' and 'y' fn(7,9,y=5,x=3,a=1,b='python') 3 5 (7, 9) {'a': 1, 'b': 'python'}
def fn(*args, x):#*号之后,普通形参都变成了必须给出的keyword-only 参数 print(x) print(args) fn(3,5) 结果为: --------------------------------------------------------------------------- TypeError Traceback (most recent call last) <ipython-input-53-c5c0648876f6> in <module> 2 print(x) 3 print(args) ----> 4 fn(3,5) TypeError: fn() missing 1 required keyword-only argument: 'x' fn(3,5,7) 结果为: --------------------------------------------------------------------------- TypeError Traceback (most recent call last) <ipython-input-55-da5434142bea> in <module> ----> 1 fn(3,5,7) TypeError: fn() missing 1 required keyword-only argument: 'x' fn(3,5,x=7) 结果为: 7 (3, 5)
上面的例子args可以看做已经截获了所有的位置参数,x不使用关键字参数就不可能拿到实参。
keyword-only参数
如果在一个星号参数后,或者一个位置可变参数后,出现的普通参数,实际上已经不是普通的参数了,而是keyword-only参数。就像上面的例子一样。而如果是关键字可变参数在前面,像def daf(**kwargs, x)则会直接报语法错误,可以理解为kwargs会截获所有的关键字参数,就算你写了x=5,x也永远得不到这个值,所以语法
错误。
def(**kwargs, x): print(x) print(kwargs) 结果为: File "<ipython-input-57-63e64f7a052a>", line 1 def(**kwargs, x): ^ SyntaxError: invalid syntax
def fn(*, x,y):#这个*没有意义,只是让后面的参数变成了keyword-only参数 print(x,y) fn(x=5,y=6) 结果为: 5 6
可变参数和参数默认值
def fn(*args,x = 5): print(x) print(args) fn()#等价于fn(x=5) 5 () fn(5) 5 (5,) fn(x=6) 6 () fn(1,2,3,x=6) 6 (1, 2, 3)
def fn(y,*args,x=5): print("x={},y={}".format(x,y)) print(args) fn() TypeError: fn() missing 1 required positional argument: 'y' fn(5) x=5,y=5 () fn(x=6) TypeError: fn() missing 1 required positional argument: 'y' fn(1,2,3,x=10)
x=10, y=1 (2, 3)
fn(y=17,2,3,x=10)
结果为:
File "<ipython-input-63-89c585e00820>", line 1
fn(y=17,2,3,x=10)
^
SyntaxError: positional argument follows keyword argument
fn(1,2,y=3,x=10)
结果为:
---------------------------------------------------------------------------
TypeError Traceback (most recent call last)
<ipython-input-65-ca712b8f6fdb> in <module>
----> 1 fn(1,2,y=3,x=10)
TypeError: fn() got multiple values for argument 'y'
可变参数和参数默认值
def fn(x=5,**kwargs): print("x={}".format(x)) print(kwargs) fn() x=5 {} fn(5) x=5 {} fn(x=6) x=6 {} fn(y=3,x=5) x=5 {'y': 3} fn(3,y=10) x=3 {'y': 10}
由此可见,参数列表参数一般顺序是,普通参数、缺省参数、可变位置参数、keyword-only参数(可带缺省值)、可变关键字参数 。
def fn(x, y, z=3, *arg, m=4, n, **kwargs): print(x,y,z,m,n) print(args) print(kwargs)
def connect(host='localhost', port='3306', user='admin', password='admin', **kwargs): print(host, port) print(user, password) print(kwargs) connect(db='cmdb') connect(host='192.168.1.123', db='cmdb') connect(host='192.168.1.123', db='cmdb', password='mysql') 结果为: localhost 3306 admin admin {'db': 'cmdb'} 192.168.1.123 3306 admin admin {'db': 'cmdb'} 192.168.1.123 3306 admin mysql {'db': 'cmdb'}
def add(x, y): return x+y add(*{'x': 5, 'y': 6}.keys()) 结果为: 'xy' def add(x, y): return x+y add(*{'x': 5, 'y': 6}.values()) 结果为: 11
def fn(*args,*,x,y): print(x,y) 结果为: File "<ipython-input-108-35d54ecda275>", line 1 def fn(*args,*,x,y): ^ SyntaxError: invalid syntax
def fn(x,y,z=3,*args,m=4,n,**kwargs): print(x,y,z,m,n) print(args) print(kwargs) fn(1,2,n=4) 结果为: 1 2 3 4 4 () {} fn(1,2,10,11,t=7,n=5) 结果为: 1 2 10 4 5 (11,) {'t': 7}
参数解构
给函数提供实参的时候,可以在集合类型前使用*或者**,把集合类型的结构解开,提取出所有元素作为函数的实参 。非字典类型使用*解构成位置参数 ,字典类型使用**解构成关键字参数 。提取出来的元素数目要和参数的要求匹配,也要和参数的类型匹配。
def add(x,y): return x+y add(4,5) 9 add((4,5)) TypeError: add() missing 1 required positional argument: 'y' add(*(4,5)) 9 t=(4,5) add(t[0],t[1]) 9 add(*t) 9 add(*[4,5]) 9 add(*{4,5})#集合 9 add(*range(1,3)) 3
d = {'x': 5, 'y': 6}
add(**d)
11
add(**{'a': 5, 'b': 6})
TypeError: add() got an unexpected keyword argument 'a'
add(*{'a': 5, 'b': 6})字典的结构可以用*d.keys()或者*d.values()
'ab'
def add(x, y):
return x+y
add(**{'x': 5, 'y': 6})
结果为:
11
参数解构和可变参数
给函数提供实参的时候,可以在集合类型前使用*或者**,把集合类型的结构解开,提取出所有元素作为函数的实参。
def add(*iterable): result = 0 for x in iterable: result += x return result add(1,2,3) 结果为: 6 add(*[1,2,3]) 结果为: 6 add(*range(10)) 结果为: 45
练习题1:
编写一个函数,能够接受至少2个参数,返回最小值和最大值。
import random def double_values(*nums): print(nums) return max(nums),min(nums) print(*double_values(*[random.randint(10,20) for _ in range(10)])) 结果为: (14, 11, 17, 13, 19, 15, 20, 16, 19, 10) 20 10 def nums(x,y,*args): print(min(x,y,*args)) print(max(x,y,*args)) nums(1,25,0,-3) 结果为: -3 25
练习题二:编写一个函数,接受一个参数n,n为正整数,左右两种打印方式,要求数字必须对齐。
def tan_print(n): for i in range(1,n+1): for j in range(n,0,-1): if i < j: print(" "*len(str(j)),end=" ") else: print(j,end=" ") print() tan_print(11) 结果为: 1 2 1 3 2 1 4 3 2 1 5 4 3 2 1 6 5 4 3 2 1 7 6 5 4 3 2 1 8 7 6 5 4 3 2 1 9 8 7 6 5 4 3 2 1 10 9 8 7 6 5 4 3 2 1 11 10 9 8 7 6 5 4 3 2 1
def show(n): tail = " ".join([str(i) for i in range(n,0,-1)]) width = len(tail) for i in range(1,n): print("{:>{}}".format(" ".join([str(j) for j in range(i,0,-1)]),width)) print(tail) show(11)
结果为:
def showtail(n): tail = " ".join([str(i) for i in range(n,0,-1)]) print(tail) #无需再次生成列表 for i in range(len(tail)): if tail[i]==" ": print(" "*i,tail[i+1:])#切片,这是个copy,空间复杂度比较大。 showtail(12)
结果为: