1格式:

         deffunctionName(参数列表):

                   方法体

例子1

>>>def greet_user():
        print(“hello”)
>>>greet_user()
hello

例子2

>>>def greet_user(username):  #username形参
        print(“hello,”+ username+“!”)
>>>greet_user(“zhangsan”)   #zhangsan 实参
hello, zhangsan!

2传递实参

鉴于函数定义中可能包含多个形参,因此在函数调用中也可能包含多个实参。向函数中传递实参的方式很多,可以使用位置实参,这要求实参的顺序和形参的顺序相同;也可以使用关键字形参,其中的每个实参都有变量名和值组成;还可以使用列表和字典。

                                                       

1)位置实参

         要求实参的顺序和形参的顺序相同

         例子:

     

    def describe_pet(animal_type,pet_name):
        print(“\n I have a”+animal_type+”.”)
        print(“My ”+animal_type+”’sname is ”+pet_name+”.”)
    describe_pet(‘hamster’,’harry’)

         当两个参数的位置交换后,表达的意思就会出现错误。

         这种传参方式应该是最好理解的,大部分语言都是这种传参方式。

2)默认参数

编写函数时,可给每个形参指定默认值。在调用函数中给形参提供了实参时,Python将使用指定实参,否则使用默认值。

def describe_pet(pet_name,animal_type=‘dog’):
      print(“\n I have a ”+animal_type+”.”)
      print(“My ”+animal_type+”’sname is ”+pet_name+”.”

 

注意:在使用默认参数时,在形参列表中必须先列出没有默认值的形参,再列出有默认值的形参。

当然,默认值不一定是字符串,数字等,同时也支持各种序列。

def f(a, L=[]):
    L.append(a)
    return L
 
print(f(1))
print(f(2))
print(f(3))


 

输出结果为

[1]
[1, 2]
[1, 2, 3]


 

如果你不想调用之间共享的默认设置,你可以这样写

def f(a, L=None):
    if L is None:
        L = []
    L.append(a)
    return L


感觉有点像java中单例模式中的懒汉式,虽然不能这么比。

 

3)关键字实参

         关键字实参是传递给函数的 名称-值对。你直接在实参中将名称和值关联起来,因此向函数传递实参时不会混淆

def parrot(voltage, state='a stiff', action='voom',type='Norwegian Blue'):
    print("--This parrot wouldn't", action, end=' ')
    print("ifyou put", voltage, "volts through it.")
    print("--Lovely plumage, the", type)
    print("--It's", state, "!")


 

parrot(1000)                                          # 1positional argument
parrot(voltage=1000)                                  # 1 keywordargument
parrot(voltage=1000000,action='VOOOOOM')             # 2 keywordarguments
parrot(action='VOOOOOM',voltage=1000000)             # 2 keywordarguments
parrot('amillion', 'bereft of life', 'jump')        # 3 positional arguments
parrot('athousand', state='pushing up the daisies') # 1 positional, 1 keyword

执行后的结果

>>>
===========RESTART: D:/python/pythonLearning/KeywordArguments.py ===========
-- This parrotwouldn't voom if you put 1000 volts through it.
-- Lovelyplumage, the Norwegian Blue
-- It's a stiff!
-- This parrotwouldn't voom if you put 1000 volts through it.
-- Lovelyplumage, the Norwegian Blue
-- It's a stiff!
-- This parrotwouldn't VOOOOOM if you put 1000000 volts through it.
-- Lovelyplumage, the Norwegian Blue
-- It's a stiff!
-- This parrotwouldn't VOOOOOM if you put 1000000 volts through it.
-- Lovelyplumage, the Norwegian Blue
-- It's a stiff!
-- This parrotwouldn't jump if you put a million volts through it.
-- Lovelyplumage, the Norwegian Blue
-- It's bereftof life !
-- This parrotwouldn't voom if you put a thousand volts through it.
-- Lovelyplumage, the Norwegian Blue
-- It's pushingup the daisies !

 

但是下面的调用方式会报错:

parrot()                     # 必须的参数没有填入

parrot(voltage=5.0, 'dead')  # 关键字传参后面的一定也是关键字传参

parrot(110, voltage=220)     # 同一个参数传了不同的实参

parrot(actor='John Cleese')  # 函数定义中没有这个形参。

关键字实参后面不能再有位置实参。

当必填参数没有时,会报错。

当必填参数以位置实参的方式传递时,默认参数可以按照关键字实参的方式传递,也可以以位置实参的方式传递,而且此时,必填参数一定在默认参数前面。

当必填参数以关键字实参方式传递的时候,默认参数传递的时候也一定是以关键字实参方式传递,而且此时与位置无关。

关键字实参的关键字必须是形参中有的,否则会报错。

4)任意多数目的参数“*

最不常使用的选项是指定可以使用任意数量的参数调用的函数。这些参数将被裹在一个元组中。在可变数目的参数之前, 零个或更多的正常参数可能会发生。

def write_multiple_items(file, separator, *args):
   file.write(separator.join(args))


 

通常情况下,可变参数将在参数列表中的最后(含有默认参数的时候,默认参数在可变参数后面。另外,传入字典参数也在可变参数之后,因为字典参数也是一种可变参数)。

在可变参数后面的参数,只能以关键字实参的方式传递,而不能使用位置实参传递方式。

def concat(*args, sep="/"):
    return sep.join(args)
 
print(concat("earth", "mars", "venus"))
print(concat("earth", "mars", "venus",sep="."))


结果为:

>>>
===========RESTART: D:/python/pythonLearning/KeywordArguments.py ===========
earth/mars/venus
earth.mars.venus

(5)需要字典类型的参数“**”

下面例子中**keywords代表传入字典类型的参数,而且*一定要在**之前。

def cheeseshop(kind, *arguments, **keywords):
    print("--Do you have any", kind, "?")
    print("--I'm sorry, we're all out of", kind)
    for arg inarguments:
        print(arg)
    print("-" * 40)
    keys =sorted(keywords.keys())
    for kw in keys:
        print(kw,":", keywords[kw])


可以这样调用

        cheeseshop("Limburger", "It's very runny, sir.",
         "It's really very, VERY runny, sir.",
          shopkeeper="Michael Palin",
          client="John Cleese",
          sketch="Cheese Shop Sketch")

         

当然结果如下:

 >>>
===========RESTART: D:/python/pythonLearning/KeywordArguments.py ===========
-- Do you haveany Limburger ?
-- I'm sorry,we're all out of Limburger
It's very runny,sir.
It's reallyvery, VERY runny, sir.
----------------------------------------
client : JohnCleese
shopkeeper :Michael Palin
sketch : CheeseShop Sketch

6Unpacking Argument Listspython3.6帮助文档上用的这个词,必应翻译给的翻译是开箱的参数列表,我感觉应该是参数列表的拆包,不够下面还是用了开箱这个词,大家理解下

         这是与之前把数据组合成list相反的操作,即把list中的数据进行开箱操作,把里面的数据全部取出来使用,有点不好理解,看个例子吧

 

>>> list(range(3,6))
[3, 4, 5]
>>> args=[3,6]
>>> list(range(*args))
[3, 4, 5]
>>> args=(3,6)
>>> list(range(*args))
[3, 4, 5]
    >>>

这里把args中的数据进行开箱操作,把里面的数据作为range的参数列表.很显然,这个开箱操作对于元组也是适用的.

 

另外,也可以对字典执行开箱操作,需要的是两个*号即可.

def parrot(voltage, state='a stiff',action='voom'):
   print("-- This parrot wouldn't", action, end=' ')
   print("if you put", voltage, "volts through it.", end='')
   print("E's", state, "!")
 
d = {"voltage": "fourmillion", "state": "bleedin' demised","action": "VOOM"}
parrot(**d)

 

结果为:

>>>
=========== RESTART:D:/python/pythonLearning/KeywordArguments.py ===========
-- This parrot wouldn't VOOM if you putfour million volts through it. E's bleedin' demised !

>>> 

(7)lambda表达式

  可以使用 lambda 关键字创建小的匿名函数。此函数返回其两个参数的总和         lambda a, b:a + b

Lambda 函数可以用于任何函数对象。他们在语法上限于单个表达式。语义上,他们都只是正常的函数定义的语法糖。嵌套的函数定义类似,lambda 函数可以从包含范围引用变量。(之后的文章会详细讲解lambda表达式的相关操作)

def make_incrementor1(n):
    return lambdax:x+n
 
f=make_incrementor1(42)
 
print(f(1))
 
>>>
=========== RESTART:D:/python/pythonLearning/KeywordArguments.py ===========
43
>>>

 

(8) Function Annotations函数注解

首先是一个简单的例子

def f(ham:str,eggs:str='eggs2')->str:
    print("Annotations:",f.__annotations__)
   print("Arguments:",ham,eggs)
   return str(ham)+' and ' +eggs
s=f(‘abc’)
print(s)

结果:

>>>
=========== RESTART:D:/python/pythonLearning/KeywordArguments.py ===========
Annotations: {'ham': <class 'str'>,'eggs': <class 'str'>, 'return': <class 'str'>}
Arguments: abc eggs2
abc and eggs2
>>>

 

函数注解是完全可选的元数据信息,它告诉人们这个函数返回的类型应该是什么类型,函数形参应该是什么类型。但是传入其他类型可能会报错,当然,这并非绝对,只不过是开发者给调用者的一个建议。上面的例子中,即使实参传入数字也是不会报错的。(这段在python3.6帮助文档上的说明看得我很头疼,比较乱,我领会精神写的)

 

注解存储在 __annotations__ 属性中,以字典的形式存在,对其他部分的功能没有任何影响。

参数注释:在参数名称的后面加一个冒号,其后是期待实参的类型。

返回值注解:函数的参数列表外面,冒号前面,加上一个 –>,后面是期待返回的类型。