我首先把所有跟python参数有关的构成方法罗列出来作为printdif的参数,这种构成方法可以看成是混合参数构造,是参数构造中最为复杂的一种方式,作为对讲解参数*args与**kwargs的一个非常棒的方法。代码如下,仔细看:
#!/usr/bin/env python2
# -*- coding: utf-8 -*-
"""
Created on Sat May 20 18:43:07 2017
This demo shows the diffrent arguments between *args and **kwargs
@author: zhangll
"""
class argdemo:
def __init__(self,name):
self.name=name
def printdif(self,name,key=None,*args,**kwargs):
self.name=name
print self.name
print key
print "args=======%s"%str(args)
count=0
for value in args:
count+=1
print "args's No.%s value:%s"%(count,value)
print "kwargs=======%s"%str(kwargs)
for key in kwargs:
print "**kwargs's key=%s,value=%s"%(key,kwargs[key])
if __name__=='__main__':
arg=argdemo("name1")
print arg.name
arg.printdif("name2","B",["C","D"],{"key1":"E"},key1="hello",key2="world")
最后得到结果
name2
B
args=======(['C', 'D'], {'key1': 'E'})
args's No.1 value:['C', 'D']
args's No.2 value:{'key1': 'E'}
kwargs======={'key2': 'world', 'key1': 'hello'}
**kwargs's key=key2,value=world
**kwargs's key=key1,value=hello
这个代码通过调用arg对象的printdif方法传递了共六个参数
- "name2" 字符串
- "B" 字符串
- ["C","D"] 列表
- {"key1":"E"} 字典
- key1="hello"
- key2="world"
python参数相关知识
以上的定义printdif所使用的参数基本上涵盖了python的主流参数,根据主调函数**是否必要**传参的特点上大致把python参数分为:**必选参数**与**可选参数**。关于这两种类型的参数需要做如下语法规则的补充:
函数定义与主调函数调用的参数排序规则:在必选参数与可选参数同时存在的情况下,必须要先定义必选参数,然后再能定义可选参数------毕竟可选的太多,而必选的参数是反映函数功能的主体,当然是让能够反映函数主体的元素排在前头咯。
上面的[name]参数为**必选参数**,特点是在函数定义上,不需要指定内容,只需要关键字【name】就行了,但是**主调函数**必须要传入该关键字相关的内容(对象),可以不需要显式关键字声明参数,比如可以定义
非显式关键字:printdif(“abc”)
显式关键字:printdif(name=“abc”)调用函数结果
abc
None
args=======()
kwargs======={}
[key=None,\*args,\*\*kwargs]这三个参数为**可选参数**,也就是说在主调函数中可以不用传参,上面的必选参数例子就说明只需要必选参数的函数调用是可以执行的。如下,我们关注的是**可选参数**的内容
1.默认参数又称关键字参数(keyword argument):【key=None】在函数定义的时候已经事先定义好了内容,key关键字对应的是None(相当于什么都没有)内容:可以如下调用主调函数
非显式关键字:printdif(“abc”,“B”)
显式关键字:printdif(name=“abc”,key=“B”)调用函数结果:
abc
B
args=======()
kwargs======={}
问题1:printdif(key=“B”,“abc”)这样的写法正确吗?
答案当然是错误的,因为我们在上面的定义语法要求上说过了,key=“B”其实质是一种可选参数的传递,而“abc”为无关键字的必选参数的传递,必选参数需要放在可选参数之前,否则会出现
SyntaxError: non-keyword arg after keyword arg
问题2:printdif(“abc”,name=“B”)这样的写法正确吗?
答案当然还是错误的,这种写法在直观上看有重复调用参数的行为,因为我们不仅在定义参数的语法上有“必选参数优先放在可选参数之前”的说法,而且在函数执行过程中也有“必选参数优先选择权”,也就是说,在执行printdif(“abc”,name=“B”)语句的时候,python先执行了,“abc”并传值于name形参,之后又执行一遍name=“ B”,也就是又一次重新传值,就会出现
TypeError: printdif() got multiple values for keyword argument 'name'
因此python拒绝这种多此一举的行为
2.*args参数 **kwargs参数
那么为什么我要在最后讲解这两个参数的区别呢?这个两个难道优先级最低我才最后讲解吗?其实这两个参数还真是在函数定义优先级方面最低(*args参数优先于**kwargs参数),看printdif的参数顺序就行了,并且这种定义次序不可改变乱了顺序,否则会报“语法错误”。虽然级别较低,不过这两种参数的运用确实比较常见的,或者说这种参数的出现,重复发扬了python参数传递的灵活性,我们拿numpy中的函数来查看下:
>>>p.fromfunction?
Signature: np.fromfunction(function, shape, **kwargs)
...
大家有没有发现,在第一个printdif主调函数中序列1和2实参分别传入必选行参与关键字形参,而在分配之后还剩余的实参将会被默认分配给*args形参与**kwargs形参。序列3、4参数作为*args的参数依次传递,而第5、6个做为**kwargs的参数以字典的形式传入。从以上这个特点可以发现,*args传入的参数不需要赋值命名,并且构成元组类型的结构,我们称这种参数为元组参数【元组形式:args=(['C', 'D'], {'key1': 'E'})】;而**kwargs需要以命名关键字的实参作为传入,且构成字典类型的结构,并且可用key关键字值找到对应的参数内容,我们称这种参数为字典参数【字典形式kwargs={'key2': 'world', 'key1': 'hello'}】。
这里有个最后的问题:
如果我调用printdif(“abc”,“2”,“3”,“4”,key=“B”,key1=“C”),会得到什么结果?