当前位置:首页 » 编程语言 » python定义函数类型

python定义函数类型

发布时间: 2023-02-15 06:47:35

python 函数参数的类型

1. 不同类型的参数简述
#这里先说明python函数调用得语法为:

复制代码
代码如下:

func(positional_args,
keyword_args,
*tuple_grp_nonkw_args,
**dict_grp_kw_args)

#为了方便说明,之后用以下函数进行举例
def test(a,b,c,d,e):

print a,b,c,d,e

举个例子来说明这4种调用方式得区别:

复制代码
代码如下:

#
#positional_args方式
>>>
test(1,2,3,4,5)
1 2 3 4 5

#这种调用方式的函数处理等价于
a,b,c,d,e = 1,2,3,4,5
print a,b,c,d,e

#
#keyword_args方式
>>>
test(a=1,b=3,c=4,d=2,e=1)
1 3 4 2 1

#这种处理方式得函数处理等价于
a=1
b=3
c=4
d=2
e=1
print a,b,c,d,e

#
#*tuple_grp_nonkw_args方式
>>>
x = 1,2,3,4,5
>>> test(*x)
1 2 3 4
5

#这种方式函数处理等价于

复制代码
代码如下:

a,b,c,d,e = x
print
a,b,c,d,e
#特别说明:x也可以为dict类型,x为dick类型时将键传递给函数
>>> y
{'a': 1,
'c': 6, 'b': 2, 'e': 1, 'd': 1}
>>> test(*y)
a c b e d

#
#**dict_grp_kw_args方式
>>>
y
{'a': 1, 'c': 6, 'b': 2, 'e': 1, 'd': 1}
>>> test(**y)
1 2 6
1 1

#这种函数处理方式等价于
a = y['a']
b = y['b']
... #c,d,e不再赘述
print
a,b,c,d,e

2.
不同类型参数混用需要注意的一些细节
接下来说明不同参数类型混用的情况,要理解不同参数混用得语法需要理解以下几方面内容.

首先要明白,函数调用使用参数类型必须严格按照顺序,不能随意调换顺序,否则会报错. 如 (a=1,2,3,4,5)会引发错误,;
(*x,2,3)也会被当成非法.

其次,函数对不同方式处理的顺序也是按照上述的类型顺序.因为#keyword_args方式和**dict_grp_kw_args方式对参数一一指定,所以无所谓顺序.所以只需要考虑顺序赋值(positional_args)和列表赋值(*tuple_grp_nonkw_args)的顺序.因此,可以简单理解为只有#positional_args方式,#*tuple_grp_nonkw_args方式有逻辑先后顺序的.

最后,参数是不允许多次赋值的.

举个例子说明,顺序赋值(positional_args)和列表赋值(*tuple_grp_nonkw_args)的逻辑先后关系:

复制代码
代码如下:

#只有在顺序赋值,列表赋值在结果上存在罗辑先后关系
#正确的例子1
>>> x =
{3,4,5}
>>> test(1,2,*x)
1 2 3 4 5
#正确的例子2
>>>
test(1,e=2,*x)
1 3 4 5 2

#错误的例子
>>> test(1,b=2,*x)
Traceback (most recent call
last):
File "<stdin>", line 1, in <mole>
TypeError: test()
got multiple values for keyword argument 'b'

#正确的例子1,处理等价于
a,b = 1,2 #顺序参数
c,d,e = x #列表参数
print a,b,c,d,e

#正确的例子2,处理等价于
a = 1 #顺序参数
e = 2 #关键字参数
b,c,d = x #列表参数

#错误的例子,处理等价于
a = 1 #顺序参数
b = 2 #关键字参数
b,c,d = x
#列表参数
#这里由于b多次赋值导致异常,可见只有顺序参数和列表参数存在罗辑先后关系

函数声明区别

理解了函数调用中不同类型参数得区别之后,再来理解函数声明中不同参数得区别就简单很多了.

1. 函数声明中的参数类型说明

函数声明只有3种类型, arg, *arg , **arg 他们得作用和函数调用刚好相反.
调用时*tuple_grp_nonkw_args将列表转换为顺序参数,而声明中的*arg的作用是将顺序赋值(positional_args)转换为列表.
调用时**dict_grp_kw_args将字典转换为关键字参数,而声明中**arg则反过来将关键字参数(keyword_args)转换为字典.
特别提醒:*arg
和 **arg可以为空值.

以下举例说明上述规则:

复制代码
代码如下:

#arg, *arg和**arg作用举例
def
test2(a,*b,**c):
print a,b,c
#
#*arg 和
**arg可以不传递参数
>>> test2(1)
1 () {}
#arg必须传递参数
>>>
test2()
Traceback (most recent call last):
File "<stdin>", line 1,
in <mole>
TypeError: test2() takes at least 1 argument (0 given)

#
#*arg将顺positional_args转换为列表
>>>
test2(1,2,[1,2],{'a':1,'b':2})
1 (2, [1, 2], {'a': 1, 'b': 2})
{}
#该处理等价于
a = 1 #arg参数处理
b = 2,[1,2],{'a':1,'b':2} #*arg参数处理
c =
dict() #**arg参数处理
print a,b,c

#
#**arg将keyword_args转换为字典
>>>
test2(1,2,3,d={1:2,3:4}, c=12, b=1)
1 (2, 3) {'c': 12, 'b': 1, 'd': {1: 2, 3:
4}}
#该处理等价于
a = 1 #arg参数处理
b= 2,3 #*arg参数处理
#**arg参数处理
c =
dict()
c['d'] = {1:2, 3:4}
c['c'] = 12
c['b'] = 1
print
a,b,c

2. 处理顺序问题

函数总是先处理arg类型参数,再处理*arg和**arg类型的参数.
因为*arg和**arg针对的调用参数类型不同,所以不需要考虑他们得顺序.

复制代码
代码如下:

def test2(a,*b,**c):
print
a,b,c
>>> test2(1, b=[1,2,3], c={1:2, 3:4},a=1)
Traceback (most
recent call last):
File "<stdin>", line 1, in
<mole>
TypeError: test2() got multiple values for keyword argument
'a'
#这里会报错得原因是,总是先处理arg类型得参数
#该函数调用等价于
#处理arg类型参数:
a = 1
a = 1
#多次赋值,导致异常
#处理其他类型参数
...
print a,b,c

>>> def foo(x,y):
... def bar():
... print
x,y
... return bar
...
#查看func_closure的引用信息
>>> a =
[1,2]
>>> b = foo(a,0)
>>>
b.func_closure[0].cell_contents
[1, 2]
>>>
b.func_closure[1].cell_contents
0
>>> b()
[1, 2] 0

#可变对象仍然能被修改
>>> a.append(3)
>>>
b.func_closure[0].cell_contents
[1, 2, 3]
>>> b()
[1, 2, 3] 0

❷ Python函数和模块的定义与使用,包括变量的类型及匿名函数用法

函数是组织好,可重复使用的,用来实现相关功能的代码段

函数提高了代码的重复利用率和应用的模块性。

除Python自带的函数之外,也可以自己创建函数,叫做自定义函数

语法:

函数代码块以 def 开头

参数为 输入值 ,放于函数名后口号里面,

函数内容以冒号:开始,函数体缩进, return 返回 输出值

函数调用使用关键字参数来确定传入的参数值,此时,如果多个函数则不需要按照指定顺序。

在定义函数时,指定参数默认值。调用时如果不传入参数,则使用默认值

不定长部分如果没有指定参数,传入是一个空元组

加了 两个星号 ** 的参数会以字典的形式导入

/ 用来指明函数形参必须使用指定位置参数,不能使用关键字参数的形式。

3.8版本之后的才能使用

不使用 def 定义函数,没有函数名

lamdba主体时一个表达式,而不是代码块,函数体比def简单很多

定义在函数内部的为局部变量,仅能在函数内部使用

定义在函数外部的为全局变量,可在全局使用

模块是将包含所有定义的函数和变量的文件,一般将同类功能的函数组和在一起称为模块。

模块需要导入后,在调用相应函数进行使用

模块导入的方法:

从模块中导入一个指定的部分

把一个模块的所有内容全都导入

❸ python怎么定义函数

给你两个函数:

##插入排序
definsertion_sort(sort_list):
iter_len=len(sort_list)
ifiter_len<2:
returnsort_list
foriinrange(1,iter_len):
key=sort_list[i]
j=i-1
whilej>=0andsort_list[j]>key:
sort_list[j+1]=sort_list[j]
j-=1
sort_list[j+1]=key
returnsort_list
##计算两点之间的距离
defGetDistance(fPoint1,fPoint2):
x1=fPoint1.X
y1=fPoint1.Y
x2=fPoint2.X
y2=fPoint2.Y
returnpow((x1-x2),2)+pow((y1-y2),2)

❹ Python的函数都有哪些

Python 函数

函数是组织好的,可重复使用的,用来实现单一,或相关联功能的代码段。

函数能提高应用的模块性,和代码的重复利用率。你已经知道Python提供了许多内建函数,比如print()。但你也可以自己创建函数,这被叫做用户自定义函数。

定义一个函数

你可以定义一个由自己想要功能的函数,以下是简单的规则:

  • 函数代码块以def关键词开头,后接函数标识符名称和圆括号()。

  • 任何传入参数和自变量必须放在圆括号中间。圆括号之间可以用于定义参数。

  • 函数的第一行语句可以选择性地使用文档字符串—用于存放函数说明。

  • 函数内容以冒号起始,并且缩进。

  • return [表达式]结束函数,选择性地返回一个值给调用方。不带表达式的return相当于返回 None。

  • 语法

    def functionname( parameters ): "函数_文档字符串"
    function_suite
    return [expression]

    默认情况下,参数值和参数名称是按函数声明中定义的顺序匹配起来的。

    实例

    以下为一个简单的Python函数,它将一个字符串作为传入参数,再打印到标准显示设备上。

    实例(Python 2.0+)

    def printme( str ): "打印传入的字符串到标准显示设备上"
    print str
    return

    函数调用

    定义一个函数只给了函数一个名称,指定了函数里包含的参数,和代码块结构。

    这个函数的基本结构完成以后,你可以通过另一个函数调用执行,也可以直接从Python提示符执行。

    如下实例调用了printme()函数:

    实例(Python 2.0+)

    #!/usr/bin/python# -*- coding: UTF-8 -*-
    # 定义函数def printme( str ): "打印任何传入的字符串"
    print str
    return
    # 调用函数printme("我要调用用户自定义函数!")printme("再次调用同一函数")

    以上实例输出结果:

  • 我要调用用户自定义函数!再次调用同一函数

  • 参数传递

    在 python 中,类型属于对象,变量是没有类型的:

    a=[1,2,3]
    a="Runoob"

    以上代码中,[1,2,3]是 List 类型,"Runoob"是 String 类型,而变量 a 是没有类型,她仅仅是一个对象的引用(一个指针),可以是 List 类型对象,也可以指向 String 类型对象。

    可更改(mutable)与不可更改(immutable)对象

    在 python 中,strings, tuples, 和 numbers 是不可更改的对象,而 list,dict 等则是可以修改的对象。

  • 不可变类型:变量赋值a=5后再赋值a=10,这里实际是新生成一个 int 值对象 10,再让 a 指向它,而 5 被丢弃,不是改变a的值,相当于新生成了a。

  • 可变类型:变量赋值la=[1,2,3,4]后再赋值la[2]=5则是将 list la 的第三个元素值更改,本身la没有动,只是其内部的一部分值被修改了。

  • python 函数的参数传递:

  • 不可变类型:类似 c++ 的值传递,如 整数、字符串、元组。如fun(a),传递的只是a的值,没有影响a对象本身。比如在 fun(a)内部修改 a 的值,只是修改另一个复制的对象,不会影响 a 本身。

  • 可变类型:类似 c++ 的引用传递,如 列表,字典。如 fun(la),则是将 la 真正的传过去,修改后fun外部的la也会受影响

  • python 中一切都是对象,严格意义我们不能说值传递还是引用传递,我们应该说传不可变对象和传可变对象。

    python 传不可变对象实例

    实例(Python 2.0+)

    #!/usr/bin/python# -*- coding: UTF-8 -*-
    def ChangeInt( a ): a = 10
    b = 2ChangeInt(b)print b # 结果是 2

    实例中有 int 对象 2,指向它的变量是 b,在传递给 ChangeInt 函数时,按传值的方式复制了变量 b,a 和 b 都指向了同一个 Int 对象,在 a=10 时,则新生成一个 int 值对象 10,并让 a 指向它。

    传可变对象实例

    实例(Python 2.0+)

    #!/usr/bin/python# -*- coding: UTF-8 -*-
    # 可写函数说明def changeme( mylist ): "修改传入的列表"
    mylist.append([1,2,3,4])
    print "函数内取值: ", mylist
    return
    # 调用changeme函数mylist = [10,20,30]changeme( mylist )print "函数外取值: ", mylist

    实例中传入函数的和在末尾添加新内容的对象用的是同一个引用,故输出结果如下:

  • 函数内取值: [10, 20, 30, [1, 2, 3, 4]]函数外取值: [10, 20, 30, [1, 2, 3, 4]]

  • 参数

    以下是调用函数时可使用的正式参数类型:

  • 必备参数

  • 关键字参数

  • 默认参数

  • 不定长参数

  • 必备参数

    必备参数须以正确的顺序传入函数。调用时的数量必须和声明时的一样。

    调用printme()函数,你必须传入一个参数,不然会出现语法错误:

    实例(Python 2.0+)

    #!/usr/bin/python# -*- coding: UTF-8 -*-
    #可写函数说明def printme( str ): "打印任何传入的字符串"
    print str
    return
    #调用printme函数printme()

    以上实例输出结果:

  • Traceback (most recent call last):

  • File "test.py", line 11, in <mole>

  • printme()TypeError: printme() takes exactly 1 argument (0 given)

  • 关键字参数

    关键字参数和函数调用关系紧密,函数调用使用关键字参数来确定传入的参数值。

    使用关键字参数允许函数调用时参数的顺序与声明时不一致,因为 Python 解释器能够用参数名匹配参数值。

    以下实例在函数 printme() 调用时使用参数名:

    实例(Python 2.0+)

    #!/usr/bin/python# -*- coding: UTF-8 -*-
    #可写函数说明def printme( str ): "打印任何传入的字符串"
    print str
    return
    #调用printme函数printme( str = "My string")

    以上实例输出结果:

  • My string

  • 下例能将关键字参数顺序不重要展示得更清楚:

    实例(Python 2.0+)

    #!/usr/bin/python# -*- coding: UTF-8 -*-
    #可写函数说明def printinfo( name, age ): "打印任何传入的字符串"
    print "Name: ", name
    print "Age ", age
    return
    #调用printinfo函数printinfo( age=50, name="miki" )

    以上实例输出结果:

  • Name: mikiAge 50

  • 默认参数

    调用函数时,默认参数的值如果没有传入,则被认为是默认值。下例会打印默认的age,如果age没有被传入:

    实例(Python 2.0+)

    #!/usr/bin/python# -*- coding: UTF-8 -*-
    #可写函数说明def printinfo( name, age = 35 ): "打印任何传入的字符串"
    print "Name: ", name
    print "Age ", age
    return
    #调用printinfo函数printinfo( age=50, name="miki" )printinfo( name="miki" )

    以上实例输出结果:

  • Name: mikiAge 50Name: mikiAge 35

  • 不定长参数

    你可能需要一个函数能处理比当初声明时更多的参数。这些参数叫做不定长参数,和上述2种参数不同,声明时不会命名。基本语法如下:

    def functionname([formal_args,] *var_args_tuple ): "函数_文档字符串"
    function_suite
    return [expression]

    加了星号(*)的变量名会存放所有未命名的变量参数。不定长参数实例如下:

    实例(Python 2.0+)

    #!/usr/bin/python# -*- coding: UTF-8 -*-
    # 可写函数说明def printinfo( arg1, *vartuple ): "打印任何传入的参数"
    print "输出: "
    print arg1
    for var in vartuple: print var
    return
    # 调用printinfo 函数printinfo( 10 )printinfo( 70, 60, 50 )

    以上实例输出结果:

  • 输出:10输出:706050

  • 匿名函数

    python 使用 lambda 来创建匿名函数。

  • lambda只是一个表达式,函数体比def简单很多。

  • lambda的主体是一个表达式,而不是一个代码块。仅仅能在lambda表达式中封装有限的逻辑进去。

  • lambda函数拥有自己的命名空间,且不能访问自有参数列表之外或全局命名空间里的参数。

  • 虽然lambda函数看起来只能写一行,却不等同于C或C++的内联函数,后者的目的是调用小函数时不占用栈内存从而增加运行效率。

  • 语法

    lambda函数的语法只包含一个语句,如下:

  • lambda [arg1 [,arg2,.....argn]]:expression

  • 如下实例:

    实例(Python 2.0+)

    #!/usr/bin/python# -*- coding: UTF-8 -*-
    # 可写函数说明sum = lambda arg1, arg2: arg1 + arg2
    # 调用sum函数print "相加后的值为 : ", sum( 10, 20 )print "相加后的值为 : ", sum( 20, 20 )

    以上实例输出结果:

  • 相加后的值为 : 30相加后的值为 : 40

  • return 语句

    return语句[表达式]退出函数,选择性地向调用方返回一个表达式。不带参数值的return语句返回None。之前的例子都没有示范如何返回数值,下例便告诉你怎么做:

    实例(Python 2.0+)

    #!/usr/bin/python# -*- coding: UTF-8 -*-
    # 可写函数说明def sum( arg1, arg2 ): # 返回2个参数的和."
    total = arg1 + arg2
    print "函数内 : ", total
    return total
    # 调用sum函数total = sum( 10, 20 )

    以上实例输出结果:

  • 函数内 : 30

  • 变量作用域

    一个程序的所有的变量并不是在哪个位置都可以访问的。访问权限决定于这个变量是在哪里赋值的。

  • 变量的作用域决定了在哪一部分程序你可以访问哪个特定的变量名称。两种最基本的变量作用域如下:
  • 全局变量

  • 局部变量

  • 全局变量和局部变量

    定义在函数内部的变量拥有一个局部作用域,定义在函数外的拥有全局作用域。

    局部变量只能在其被声明的函数内部访问,而全局变量可以在整个程序范围内访问。调用函数时,所有在函数内声明的变量名称都将被加入到作用域中。如下实例:

    实例(Python 2.0+)

    #!/usr/bin/python# -*- coding: UTF-8 -*-
    total = 0 # 这是一个全局变量# 可写函数说明def sum( arg1, arg2 ): #返回2个参数的和."
    total = arg1 + arg2 # total在这里是局部变量.
    print "函数内是局部变量 : ", total
    return total
    #调用sum函数sum( 10, 20 )print "函数外是全局变量 : ", total

    以上实例输出结果:

  • 函数内是局部变量 : 30函数外是全局变量 : 0

❺ Python函数的参数类型

Python函数的参数类型主要包括必选参数、可选参数、可变参数、位置参数和关键字参数,本文介绍一下他们的定义以及可变数据类型参数传递需要注意的地方。

必选参数(Required arguments)是必须输入的参数,比如下面的代码,必须输入2个参数,否则就会报错:

其实上面例子中的参数 num1和num2也属于关键字参数,比如可以通过如下方式调用:

执行结果:

可选参数(Optional arguments)可以不用传入函数,有一个默认值,如果没有传入会使用默认值,不会报错。

位置参数(positional arguments)根据其在函数定义中的位置调用,下面是pow()函数的帮助信息:

x,y,z三个参数的的顺序是固定的,并且不能使用关键字:

输出:

在上面的pow()函数帮助信息中可以看到位置参数后面加了一个反斜杠 / ,这是python内置函数的语法定义,Python开发人员不能在python3.8版本之前的代码中使用此语法。但python3.0到3.7版本可以使用如下方式定义位置参数:

星号前面的参数为位置参数或者关键字参数,星号后面是强制关键字参数,具体介绍见强制关键字参数。

python3.8版本引入了强制位置参数(Positional-Only Parameters),也就是我们可以使用反斜杠 / 语法来定义位置参数了,可以写成如下形式:

来看下面的例子:

python3.8运行:

不能使用关键字参数形式赋值了。

可变参数 (varargs argument) 就是传入的参数个数是可变的,可以是0-n个,使用星号( * )将输入参数自动组装为一个元组(tuple):

执行结果:

关键字参数(keyword argument)允许将任意个含参数名的参数导入到python函数中,使用双星号( ** ),在函数内部自动组装为一个字典。

执行结果:

上面介绍的参数可以混合使用:

结果:

注意:由于传入的参数个数不定,所以当与普通参数一同使用时,必须把带星号的参数放在最后。

强制关键字参数(Keyword-Only Arguments)是python3引入的特性,可参考:https://www.python.org/dev/peps/pep-3102/。 使用一个星号隔开:

在位置参数一节介绍过星号前面的参数可以是位置参数和关键字参数。星号后面的参数都是强制关键字参数,必须以指定参数名的方式传参,如果强制关键字参数没有设置默认参数,调用函数时必须传参。

执行结果:

也可以在可变参数后面命名关键字参数,这样就不需要星号分隔符了:

执行结果:

在Python对象及内存管理机制中介绍了python中的参数传递属于对象的 引用传递 (pass by object reference),在编写函数的时候需要特别注意。

先来看个例子:

执行结果:

l1 和 l2指向相同的地址,由于列表可变,l1改变时,l2也跟着变了。

接着看下面的例子:

结果:

l1没有变化!为什么不是[1, 2, 3, 4]呢?

l = l + [4]表示创建一个“末尾加入元素 4“的新列表,并让 l 指向这个新的对象,l1没有进行任何操作,因此 l1 的值不变。如果要改变l1的值,需要加一个返回值:

结果:

下面的代码执行结果又是什么呢?

执行结果:

和第一个例子一样,l1 和 l2指向相同的地址,所以会一起改变。这个问题怎么解决呢?

可以使用下面的方式:

也可以使用浅拷贝或者深度拷贝,具体使用方法可参考Python对象及内存管理机制。这个问题在Python编程时需要特别注意。

本文主要介绍了python函数的几种参数类型:必选参数、可选参数、可变参数、位置参数、强制位置参数、关键字参数、强制关键字参数,注意他们不是完全独立的,比如必选参数、可选参数也可以是关键字参数,位置参数可以是必选参数或者可选参数。

另外,python中的参数传递属于对象的 引用传递 ,在对可变数据类型进行参数传递时需要特别注意,如有必要,使用python的拷贝方法。

参考文档:

--THE END--

❻ Python函数的定义(构造)和调用

这里来给大家演示一下,函数的定义或构造,并调用函数来实现封装后的效果。

首先我们来看看想实现下面的这个效果,如果不使用函数应该怎么实现。

以上两种返回结果都是1-9这几个数字。

以上两种方法,第一种代码重复率太高,代码美观效果太差,虽然能实现效果,但是因为数量比较少,还能手工打出来这几行代码,如果是打印1-100000就很难实现了。这时候for循环还是可以实现的,但是for循环只能实现类似的数字和变量循环,无法进行复杂的功能开发。鉴于此,函数这个概念就被python引入了,下面先来看看函数是怎么实现上面的效果的,还是两种方法。

这时候如果想实现上面的打印结果就直接使用函数名+小括号调用函数就可以了,这种类型的语法,不仅可以反复使用,而且封装后的代码更美观。

❼ python怎么定义函数

Python中定义函数格式为,def+函数名:代码块
如:
def print_hello():
print("hello")

❽ python-函数可变参数类型

python中的函数,大多需要配置参数,以下是几种函数的参数类型:

1.必备参数:以正确的顺序、个数传入函数。调用时的参数情况要和声明时一样。最常用的情况。

def tplink(a,b):

    c=a+b+b

    return c 

tplink(4,2)

2.关键字参数:使用关键字参数允许函数调用时参数的顺序和声明时不一致,因为python解析器会在调用函数时,用参数名匹配参数值。

def tplink(age1,age2):

    ageall=age1+age2+age2

    return ageall

tplink(age2=4,age1=2)

3.默认参数:默认某个参数的取值

def tplink(age1,age2=5):

ageall=age1+age2+age2

return ageall

tplink(age1=4)

4.不定长参数:在声明时并不确定 调用时的参数数量。这种情况,可以用不定长参数进行解决,具体操作是在参数名前用*。

但不能和 关键字参数并用。一般在正常参数arg之后。

*args、**kwargs的定义:

这两个都是python中的不定长参数,又称为可变参数。

*args 表示任何多个无名参数,它是一个 tuple ;

**kwargs 表示关键字参数,它是一个dict。

同时使用 * args和 ** kwargs 时,必须 * args参数列要在 ** kwargs前。且都在arg之后。

函数在调用时,会根据顺序,看是否放进 *args 或者 **kwargs中。

具体可根据实际情况使用,可以 更方便灵活的接收信息。

❾ python中的函数是什么

Python3 函数

函数是组织好的,可重复使用的,用来实现单一,或相关联功能的代码段。

函数能提高应用的模块性,和代码的重复利用率。你已经知道Python提供了许多内建函数,比如print()。但你也可以自己创建函数,这被叫做用户自定义函数。

定义一个函数

你可以定义一个由自己想要功能的函数,以下是简单的规则:

  • 函数代码块以def关键词开头,后接函数标识符名称和圆括号()。

  • 任何传入参数和自变量必须放在圆括号中间,圆括号之间可以用于定义参数。

  • 函数的第一行语句可以选择性地使用文档字符串—用于存放函数说明。

  • 函数内容以冒号:起始,并且缩进。

  • return [表达式]结束函数,选择性地返回一个值给调用方,不带表达式的 return 相当于返回 None。

  • 语法

    Python 定义函数使用 def 关键字,一般格式如下:

  • def 函数名(参数列表):

  • 函数体

  • 默认情况下,参数值和参数名称是按函数声明中定义的顺序匹配起来的。

    实例

    让我们使用函数来输出"Hello World!":

    #!/usr/bin/python3

    defhello():
    print("Hello World!")

    hello()

    更复杂点的应用,函数中带上参数变量:

    实例(Python 3.0+)

    比较两个数,并返回较大的数:

    #!/usr/bin/python3
    def max(a, b): if a > b: return a
    else: return b
    a = 4b = 5print(max(a, b))

    以上实例输出结果:

  • 5

  • 实例(Python 3.0+)

    计算面积函数:

    #!/usr/bin/python3
    # 计算面积函数def area(width, height): return width * height
    def print_welcome(name): print("Welcome", name)
    print_welcome("Runoob")w = 4h = 5print("width =", w, " height =", h, " area =", area(w, h))

    以上实例输出结果:

  • Welcome Runoobwidth = 4 height = 5 area = 20

  • 函数调用

    定义一个函数:给了函数一个名称,指定了函数里包含的参数,和代码块结构。

    这个函数的基本结构完成以后,你可以通过另一个函数调用执行,也可以直接从 Python 命令提示符执行。

    如下实例调用了printme()函数:

    实例(Python 3.0+)

    #!/usr/bin/python3
    # 定义函数def printme( str ): # 打印任何传入的字符串
    print (str)
    return
    # 调用函数printme("我要调用用户自定义函数!")printme("再次调用同一函数")

    以上实例输出结果:

  • 我要调用用户自定义函数!再次调用同一函数

  • 参数传递

    在 python 中,类型属于对象,变量是没有类型的:

  • a=[1,2,3]a="Runoob"

  • 以上代码中,[1,2,3]是 List 类型,"Runoob"是 String 类型,而变量 a 是没有类型,她仅仅是一个对象的引用(一个指针),可以是指向 List 类型对象,也可以是指向 String 类型对象。

    可更改(mutable)与不可更改(immutable)对象

    在 python 中,strings, tuples, 和 numbers 是不可更改的对象,而 list,dict 等则是可以修改的对象。

  • 不可变类型:变量赋值a=5后再赋值a=10,这里实际是新生成一个 int 值对象 10,再让 a 指向它,而 5 被丢弃,不是改变 a 的值,相当于新生成了 a。

  • 可变类型:变量赋值la=[1,2,3,4]后再赋值la[2]=5则是将 list la 的第三个元素值更改,本身la没有动,只是其内部的一部分值被修改了。

  • python 函数的参数传递:

  • 不可变类型:类似 C++ 的值传递,如 整数、字符串、元组。如 fun(a),传递的只是 a 的值,没有影响 a 对象本身。如果在 fun(a))内部修改 a 的值,则是新生成来一个 a。

  • 可变类型:类似 C++ 的引用传递,如 列表,字典。如 fun(la),则是将 la 真正的传过去,修改后 fun 外部的 la 也会受影响

  • python 中一切都是对象,严格意义我们不能说值传递还是引用传递,我们应该说传不可变对象和传可变对象。

    python 传不可变对象实例

    通过id()函数来查看内存地址变化:

    实例(Python 3.0+)

    def change(a): print(id(a)) # 指向的是同一个对象
    a=10
    print(id(a)) # 一个新对象
    a=1print(id(a))change(a)

    以上实例输出结果为:


  • 可以看见在调用函数前后,形参和实参指向的是同一个对象(对象 id 相同),在函数内部修改形参后,形参指向的是不同的 id。

    传可变对象实例

    可变对象在函数里修改了参数,那么在调用这个函数的函数里,原始的参数也被改变了。例如:

    实例(Python 3.0+)

    #!/usr/bin/python3
    # 可写函数说明def changeme( mylist ): "修改传入的列表"
    mylist.append([1,2,3,4])
    print ("函数内取值: ", mylist)
    return
    # 调用changeme函数mylist = [10,20,30]changeme( mylist )print ("函数外取值: ", mylist)

    传入函数的和在末尾添加新内容的对象用的是同一个引用。故输出结果如下:

  • 函数内取值: [10, 20, 30, [1, 2, 3, 4]]函数外取值: [10, 20, 30, [1, 2, 3, 4]]

  • 参数

    以下是调用函数时可使用的正式参数类型:

  • 必需参数

  • 关键字参数

  • 默认参数

  • 不定长参数

  • 必需参数

    必需参数须以正确的顺序传入函数。调用时的数量必须和声明时的一样。

    调用 printme() 函数,你必须传入一个参数,不然会出现语法错误:

    实例(Python 3.0+)

    #!/usr/bin/python3
    #可写函数说明def printme( str ): "打印任何传入的字符串"
    print (str)
    return
    # 调用 printme 函数,不加参数会报错printme()

    以上实例输出结果:

  • Traceback (most recent call last):

  • File "test.py", line 10, in <mole>

  • printme()TypeError: printme() missing 1 required positional argument: 'str'

  • 关键字参数

    关键字参数和函数调用关系紧密,函数调用使用关键字参数来确定传入的参数值。

    使用关键字参数允许函数调用时参数的顺序与声明时不一致,因为 Python 解释器能够用参数名匹配参数值。

    以下实例在函数 printme() 调用时使用参数名:

    实例(Python 3.0+)

    #!/usr/bin/python3
    #可写函数说明def printme( str ): "打印任何传入的字符串"
    print (str)
    return
    #调用printme函数printme( str = "菜鸟教程")

    以上实例输出结果:

  • 菜鸟教程

  • 以下实例中演示了函数参数的使用不需要使用指定顺序:

    实例(Python 3.0+)

    #!/usr/bin/python3
    #可写函数说明def printinfo( name, age ): "打印任何传入的字符串"
    print ("名字: ", name)
    print ("年龄: ", age)
    return
    #调用printinfo函数printinfo( age=50, name="runoob" )

    以上实例输出结果:

  • 名字: runoob年龄: 50

  • 默认参数

    调用函数时,如果没有传递参数,则会使用默认参数。以下实例中如果没有传入 age 参数,则使用默认值:

    实例(Python 3.0+)

    #!/usr/bin/python3
    #可写函数说明def printinfo( name, age = 35 ): "打印任何传入的字符串"
    print ("名字: ", name)
    print ("年龄: ", age)
    return
    #调用printinfo函数printinfo( age=50, name="runoob" )print ("------------------------")printinfo( name="runoob" )

    以上实例输出结果:

  • 名字: runoob年龄: 50------------------------名字: runoob年龄: 35

  • 不定长参数

    你可能需要一个函数能处理比当初声明时更多的参数。这些参数叫做不定长参数,和上述 2 种参数不同,声明时不会命名。基本语法如下:

  • def functionname([formal_args,] *var_args_tuple ):

  • "函数_文档字符串"

  • function_suite return [expression]

  • 加了星号*的参数会以元组(tuple)的形式导入,存放所有未命名的变量参数。

    实例(Python 3.0+)

    #!/usr/bin/python3
    # 可写函数说明def printinfo( arg1, *vartuple ): "打印任何传入的参数"
    print ("输出: ")
    print (arg1)
    print (vartuple)
    # 调用printinfo 函数printinfo( 70, 60, 50 )

    以上实例输出结果:

  • 输出: 70(60, 50)

  • 如果在函数调用时没有指定参数,它就是一个空元组。我们也可以不向函数传递未命名的变量。如下实例:
  • 实例(Python 3.0+)

    #!/usr/bin/python3
    # 可写函数说明def printinfo( arg1, *vartuple ): "打印任何传入的参数"
    print ("输出: ")
    print (arg1)
    for var in vartuple: print (var)
    return
    # 调用printinfo 函数printinfo( 10 )printinfo( 70, 60, 50 )

    以上实例输出结果:

  • 输出:10输出:706050

  • 还有一种就是参数带两个星号**基本语法如下:

  • def functionname([formal_args,] **var_args_dict ):

  • "函数_文档字符串"

  • function_suite return [expression]

  • 加了两个星号**的参数会以字典的形式导入。

    实例(Python 3.0+)

    #!/usr/bin/python3
    # 可写函数说明def printinfo( arg1, **vardict ): "打印任何传入的参数"
    print ("输出: ")
    print (arg1)
    print (vardict)
    # 调用printinfo 函数printinfo(1, a=2,b=3)

    以上实例输出结果:

  • 输出: 1{'a': 2, 'b': 3}

  • 声明函数时,参数中星号*可以单独出现,例如:

  • def f(a,b,*,c):

  • return a+b+c

  • 如果单独出现星号*后的参数必须用关键字传入。

  • >>> def f(a,b,*,c):... return a+b+c... >>> f(1,2,3) # 报错Traceback (most recent call last):

  • File "<stdin>", line 1, in <mole>TypeError: f() takes 2 positional arguments but 3 were given>>> f(1,2,c=3) # 正常6>>>

  • 匿名函数

    python 使用 lambda 来创建匿名函数。

    所谓匿名,意即不再使用 def 语句这样标准的形式定义一个函数。

  • lambda 只是一个表达式,函数体比 def 简单很多。

  • lambda的主体是一个表达式,而不是一个代码块。仅仅能在lambda表达式中封装有限的逻辑进去。

  • lambda 函数拥有自己的命名空间,且不能访问自己参数列表之外或全局命名空间里的参数。

  • 虽然lambda函数看起来只能写一行,却不等同于C或C++的内联函数,后者的目的是调用小函数时不占用栈内存从而增加运行效率。

  • 语法

    lambda 函数的语法只包含一个语句,如下:

  • lambda [arg1 [,arg2,.....argn]]:expression

  • 如下实例:

    实例(Python 3.0+)

    #!/usr/bin/python3
    # 可写函数说明sum = lambda arg1, arg2: arg1 + arg2
    # 调用sum函数print ("相加后的值为 : ", sum( 10, 20 ))print ("相加后的值为 : ", sum( 20, 20 ))

    以上实例输出结果:

  • 相加后的值为 : 30相加后的值为 : 40

  • return语句

    return [表达式]语句用于退出函数,选择性地向调用方返回一个表达式。不带参数值的return语句返回None。之前的例子都没有示范如何返回数值,以下实例演示了 return 语句的用法:

    实例(Python 3.0+)

    #!/usr/bin/python3
    # 可写函数说明def sum( arg1, arg2 ): # 返回2个参数的和."
    total = arg1 + arg2
    print ("函数内 : ", total)
    return total
    # 调用sum函数total = sum( 10, 20 )print ("函数外 : ", total)

    以上实例输出结果:

  • 函数内 : 30函数外 : 30

  • 强制位置参数

    Python3.8 新增了一个函数形参语法 / 用来指明函数形参必须使用指定位置参数,不能使用关键字参数的形式。

  • 在以下的例子中,形参 a 和 b 必须使用指定位置参数,c 或 d 可以是位置形参或关键字形参,而 e 或 f 要求为关键字形参:def f(a, b, /, c, d, *, e, f):

  • print(a, b, c, d, e, f)

  • 以下使用方法是正确的:

  • f(10, 20, 30, d=40, e=50, f=60)

  • 以下使用方法会发生错误:

  • f(10, b=20, c=30, d=40, e=50, f=60) # b 不能使用关键字参数的形式f(10, 20, 30, 40, 50, f=60) # e 必须使用关键字参数的形式

❿ Python中定义函数的使用方法

4.6. 定义函数
我们可以创建一个用来生成指定边界的斐波那契数列的函数:
>>> def fib(n): # write Fibonacci series up to n
... """Print a Fibonacci series up to n."""
... a, b = 0, 1
... while a < n:
... print(a, end=' ')
... a, b = b, a+b
... print()
...
>>> # Now call the function we just defined:
... fib(2000)
0 1 1 2 3 5 8 13 21 34 55 89 144 233 377 610 987 1597
关键字 def 引入了一个函数 定义。在其后必须跟有函数名和包括形式参数的圆括号。函数体语句从下一行开始,必须是缩进的。
函数体的第一行语句可以是可选的字符串文本,这个字符串是函数的文档字符串,或者称为 docstring。(更多关于 docstrings 的信息请参考 文档字符串) 有些工具通过 docstrings 自动生成在线的或可打印的文档,或者让用户通过代码交互浏览;在你的代码中包含 docstrings 是一个好的实践,让它成为习惯吧。
函数 调用 会为函数局部变量生成一个新的符号表。确切的说,所有函数中的变量赋值都是将值存储在局部符号表。变量引用首先在局部符号表中查找,然后是包含函数的局部符号表,然后是全局符号表,最后是内置名字表。因此,全局变量不能在函数中直接赋值(除非用 global 语句命名),尽管他们可以被引用。
函数引用的实际参数在函数调用时引入局部符号表,因此,实参总是 传值调用 (这里的 值 总是一个对象 引用 ,而不是该对象的值)。[1] 一个函数被另一个函数调用时,一个新的局部符号表在调用过程中被创建。
一个函数定义会在当前符号表内引入函数名。函数名指代的值(即函数体)有一个被 Python 解释器认定为 用户自定义函数 的类型。 这个值可以赋予其他的名字(即变量名),然后它也可以被当作函数使用。这可以作为通用的重命名机制:
>>> fib
>>> f = fib
>>> f(100)
0 1 1 2 3 5 8 13 21 34 55 89
如果你使用过其他语言,你可能会反对说:fib 不是一个函数,而是一个方法,因为它并不返回任何值。事实上,没有 return 语句的函数确实会返回一个值,虽然是一个相当令人厌烦的值(指 None )。这个值被称为 None (这是一个内建名称)。如果 None 值是唯一被书写的值,那么在写的时候通常会被解释器忽略(即不输出任何内容)。如果你确实想看到这个值的输出内容,请使用 print() 函数:

热点内容
存储计算逻辑 发布:2024-11-08 13:49:35 浏览:543
java算法排序算法 发布:2024-11-08 13:42:20 浏览:883
u盘随身系统linux 发布:2024-11-08 13:34:34 浏览:411
b1422压缩机锁定 发布:2024-11-08 13:32:43 浏览:635
上传按钮图片 发布:2024-11-08 13:30:57 浏览:920
安卓手机相机如何拍摄短视频 发布:2024-11-08 13:28:42 浏览:411
网站的并发访问 发布:2024-11-08 13:27:56 浏览:514
脉冲压缩调制 发布:2024-11-08 12:49:56 浏览:126
松茸菌存储 发布:2024-11-08 12:49:05 浏览:333
超市wifi密码大概都是什么 发布:2024-11-08 12:48:19 浏览:590