#####此篇文章来源于我的老师Wayne,仅作记录以备复习
数学定义
Python函数
函数的作用
函数的分类
def 函数名 (参数列表):
函数体 (代码块)
[return 返回值]
def add(x,y): 函数定义
result = x + y 函数体
return result 返回值
out = add(4,5) 函数调用,可能有返回值,使用变量接收这个返回值
print(out) print函数加上括号也是调用
上面代码解释:
函数在定义时要约定好形式参数,调用时也提供足够的实际参数,一般来说,形参和实参个数要一致(可变参数除外)
位置传参
定义时 def f(x,y,z) ,调用使用 f(1,3,5) ,按照参数定义顺序传入实参
关键字传参
定义时 def f(x,y,z) ,调用使用 f(x=1,y=3,z=5) ,使用形参的名字来传入实参的方式,如果使用了形参名字,那么传参顺序就可和定义顺序不同
要求位置参数必须在关键字参数之前传入,位置参数是按位置对应的
def f(x,y,z):
pass
f(z=None,y=10,x=[1])
f((1,),z=6,y=4.1)
f(y=5,z=6,2) 错误传参,关键字传参放到了位置传参之前
缺省值也成为默认值,可以在函数定义时,为形参添加一个缺省值。其作用:
def add(x=4,y=5)
return x+y
测试调用 add()、add(x=5)、add(y=7)、add(6,10)、
add(6,y=7)、add(x=5,y=6)、add(y=6,x=5)
注意==add(x=5,6)、add(y=8,4)、add(11,x=20)==这
几种是错误传参,关键字传参放到了位置传参之前
注意
:def add(x=4,y) 这种设定缺省值的形式是错误的,而
def add(x,y=4) 这样是可以的,有点类似于位置传参与关键字传参的关系。
def login(host='127.0.0.1',port='8080',username='kiffen',password='python'):
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='python',host='www')
需求:写一个函数,可以对多个数累加求和
def sum(iterable):
sum = 0
for x in iterable:
sum += x
return sum
print(sum([1,2,3,4,5]))
print(sum(range(4)))
上例,传入可迭代对象,并累加每一个元素
也可以使用可变参数完成上面的函数
def sum(*nums):
sum = 0
for x in nums:
sum += x
return sum
print(sum(1,3,5))
print(sum(1,2,3))
def showconfig(**kwargs):
for k,v in kwargs.items():
print('{}={}'.format(k,v),end=',')
showconfig(host='127.0.0.1',port=8080,username='kiffen',password='python')
混合使用
可以定义下列方式么?
def showconfig(username,password,**kwargs):
def showconfig(username,*args,**kwargs):
下面这种定义方法是错误的,可变关键字参数应放置可变位置参数之后
def showconfig(username,**kwargs,*args) :?
* 有可变位置参数和可变关键字参数
* 可变位置参数在形参前使用一个星号 *
* 可变关键字参数在形参前使用两个星号 **
* 可变位置参数和可变关键字参数都可以收集若干实参,可变位置参数收集形成一个元组 tuple ,可变关键字参数收集形成一个字典 dict
* 混合使用参数时,普通参数需要放到参数列表前面,可变参数要放到参数列表后面,可变位置参数要放到可变关键字参数之前
使用举例
def fn(x,y,*args,**kwargs):
print(x,y,args,kwargs,sep='\n',end='\n\n')
fn(3,5,7,9,10,a=1,b='abc')
fn(3,5)
fn(3,5,7)
fn(3,5,a=1,b='abc')
注意下面两个函数调用
fn(x=3,y=8,7,9,a=1,b='abc')
fn(7,9,y=5,x=3,a=1,b='abc')
fn(x=3,y=8,7,9,a=1,b=‘abc’),错在位置传参放在了关键字传参之前
fn(7,9,y=5,x=3,a=1,b=‘abc’),错在形参x,y,已经按照位置传参了,y=5,x=3有重复的传参
让我们先看一段代码
def fn(*args,x,y,**kwargs):
print(x,y,args,kwargs,sep='\n',end='\n\n')
fn(3,5)
fn(3,5,7)
fn(3,5,a=1,b='abc')
注意上面三个函数调用
fn(3,5,x=6,y=7,a=1,b='abc')
在Python 3 之后,新增了keyword-only参数。
keyword-only参数:在形参定义时,在一个 * 星号之后,或一个可变位置参数之后,出现的普通参数,就已经不是普通参数了,称为keyword-only参数。
def fn(*args,x):
print(x,args,sep='\n',end='\n\n')
fn(3,5)
fn(3,5,7)
注意上面两个函数调用
fn(3,5,x=7)
keyword-only参数,言下之意就是这个参数必须要采用关键字传参。
可以认为,上例中,args可变位置参数已经截获了所有位置参数,其后的变量 x 不可能通过位置传参传入了。
思考:def fn(**kwargs,x) 可以吗?
def fn(**kwargs,x):
print(x,kwargs,sep'\n',end='\n\n')
File "" , line 1
def fn(**kwargs,x):
^
SyntaxError: invalid syntax
直接就语法错误了。
可以认为,kwargs会截获所有关键字传参,就算写了 x=5 ,x也没有机会得到这个值,所以这种语法不存在。
keyword-only参数另一种形式
def fn(*,x,y):
print(x,y)
fn(x=6,y=9)
fn(y=6,x=9)
可变位置参数,keyword-only参数、缺省值
def fn(*args,x=5):
print(x)
print(args)
fn() 等价于fn(x=5)
fn(5)
fn(x=6)
fn(1,2,3,x=10)
普通参数,可变位置参数,keyword-only参数
def fn(y,*args,x=5):
print('x={},y={}'.format(x,y))
print(args)
fn() 错误,缺少传入普通参数 y
fn(5)
fn(5,6)
fn(x=6) 错误,缺少传入普通参数 y
fn(1,2,3,x=10)
fn(y=17,1,2,3,x=10) 错误,关键字参数放到了位置参数之前
fn(1,2,3,y=3,x=10) 错误,y 已经按照位置传参了,y=3 有重复的传参出现
普通参数、缺省值、可变关键字参数
def fn(x=5,**kwargs):
print(x)
print(kwargs)
fn()
fn(5)
fn(x=6)
fn(y=3,x=10)
fn(3,y=10)
fn(y=3,z=20)
参数列表参数一般顺序是:普通参数,缺省值,可变位置参数,keyword-only参数(可带缺省值),可变关键字参数
注意:
def fn(x,y,z=3,*args,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')
def add(x,y):
print(x,y)
return x+y
add(4,5)
add((4,5)) 错误,因为元组作为一个实参整体传入x,而y缺少实参的传入
t=4,5
add(t[0],t[1])
add(*t)
add(*(4,5))
add(*[4,5])
add(*{4,5})
add(*range(4,6))
以下三条可以么?
add(*{'a':10,'b':11})
add(**{'a':10,'b':11})
add(**{'x':10,'y':11})
第一条输出结果:
b a
'ba'
第二条输出结果:
---------------------------------------------------------------------------
TypeError Traceback (most recent call last)
<ipython-input-205-c73998831612> in <module>
----> 1 add(**{'a':10,'b':11})
TypeError: add() got an unexpected keyword argument 'b'
第三条输出结果:
10 11
21
def add(*iterable):
result = 0
for x in iterable:
result += x
return result
add(1,2,3)
add(*[1,3,5])
add(*range(5))
def maxmin(x,y,*args):
print('max={}'.format(max(x,y,*args)))
print('max={}'.format(min(x,y,*args)))
import random
print(select(*[random.randint(10,20)for i in range(random.randint(2,10))]))
or
def select(x,y,*args):
print(x,y,args)
return max(x,y,*args),min(x,y,*args)
import random
print(*select(*[random.randint(10,20)for i in range(random.randint(2,10))]))
2.完成一个函数,可以接收输入的多个数,每一次都能返回到目前为止的最大值、最小值
def double_values():
max_=min_=None
while True:
x = input('>>>')
if isinstance(x,str):
print('Input Error')
break
nums = [int(i) for i in x.replace(',',' ').split()]
if not nums:
continue
if max_ is None:
max_ = min_ = nums[0]
max_ = max(max_,*nums)
min_ = min(min_,*nums)
print(max_,min_)