Python的函数和参数 parameter 是函数定义的参数形式
argument 是函数调用时传入的参数实体 。
对于函数调用的传参模式,一般有两种:
此外,
也是关键字传参
python的函数参数定义一般来说有五种:位置和关键字参数混合 , 仅位置参数,仅关键字参数,可变位置参数 , 可变关键字参数。其中仅位置参数的方式仅仅是一个概念 , python语法中暂时没有这样的设计 。
通常我们见到的函数是位置和关键字混合的方式 。
既可以用关键字又可以用位置调用
或
这种方式的定义只能使用关键字传参的模式
【python自省函数参数 python函数自我调用】 f(*some_list) 与 f(arg1, arg2, ...) (其中some_list = [arg1, arg2, ...])是等价的
网络模块request的request方法的设计
多数的可选参数被设计成可变关键字参数
有多种方法能够为函数定义输出:
非常晦涩
如果使用可变对象作为函数的默认参数,会导致默认参数在所有的函数调用中被共享 。
例子1:
addItem方法的data设计了一个默认参数,使用不当会造成默认参数被共享 。
python里面,函数的默认参数被存在__default__属性中,这是一个元组类型
例子2:
在例子1中,默认参数是一个列表 , 它是mutable的数据类型,当它写进 __defauts__属性中时,函数addItem的操作并不会改变它的id,相当于 __defauts__只是保存了data的引用,对于它的内存数据并不关心 , 每次调用addItem,都可以修改 addItem.__defauts__中的数据,它是一个共享数据 。
如果默认参数是一个imutable类型,情况将会不一样,你无法改变默认参数第一次存入的值 。
例子1中 , 连续调用addItem('world') 的结果会是
而不是期望的
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引入的特性 , 可参考: 。使用一个星号隔开:
在位置参数一节介绍过星号前面的参数可以是位置参数和关键字参数 。星号后面的参数都是强制关键字参数,必须以指定参数名的方式传参,如果强制关键字参数没有设置默认参数,调用函数时必须传参 。
执行结果:
也可以在可变参数后面命名关键字参数,这样就不需要星号分隔符了:
执行结果:
在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中的自省自省就是程序运行时能够知道对象的类型,比如type()/isinstance()/hasattr()这种函数
Python自省与反射这个也是python彪悍的特性.
自省就是面向对象的语言所写的程序在运行时,所能知道对象的类型.简单一句就是运行时能够获得对象的类型.比如type(),dir(),getattr(),hasattr(),isinstance().
反射机制就是在运行时,动态的确定对象的类型,并可以通过字符串调用对象属性、方法、导入模块,是一种基于字符串的事件驱动
通过源码注释我们知道,它返回对象是否具有指定名称的属性 。而且它是通过调用getattr并捕获AttributeError异常来判断的 。就像上面的属性调用,我们就可以使用hasattr(a, "test")来判断,通过源码注释我们也可以思考一下,eval这种是不是也可以实现这种方法呢?
但是这种方式是有缺陷的,因为test输出了两次,因为我们调用了两次test() , 这跟我们想要的效果不一样 。如果用hasattr呢,这个函数就不会在判断的时候调用一次了 。
有了判断属性是否存在的函数,那么就得有获取属性的函数了.
从源码注释我们就能知道获取object对象的名为name的属性,想到与object.name,如果提供了default参数,那么当属性不存在的时候 , 就会返回默认值 。同样是上面的例子:
从例子中我们可以看出,hasattr并没有调用test函数,而且getattr获取到的是函数对象,也没有调用它,通过我们主动执行func()才执行了a.test()函数,这样相比于exec和eval就灵活了许多 。
判断和获取属性有了,那么设置属性也是需要的.
python自省函数参数的介绍就聊到这里吧,感谢你花时间阅读本站内容,更多关于python函数自我调用、python自省函数参数的信息别忘了在本站进行查找喔 。
推荐阅读
- sqlserver查看库名的简单介绍
- 类似少年冒险王游戏推荐,跟少年冒险王一个系列的书
- vb.net获取磁盘信息 vb读取电脑硬件信息
- oracle锁表权限,oracle表锁定解锁
- 关于windows中系统还原的信息
- 原神2.1直播讲了什么,原神21直播内容
- 手指游戏教学动作,手指游戏教案20篇
- java写冒泡排序代码 java编写冒泡排序
- ktv怎么连接电脑麦克风,麦克风连接电脑怎么用