概要:
-
参数的补充
-
函数名,函数名到底是什么?
-
返回值和print
-
函数的作用域
1.参数的补充
补充的内容包含:内存地址相关、面试题相关等,在特定情况下也可以让代码更加简洁,提升开发效率。
1.1 参数内存地址相关
在开始开始讲参数内存地址相关之前,我们先来学习一个技能:
如果想要查看下某个值的在内存中的地址?
注意:函数执行传参时,传递的是内存地址。
面试题:请问Python的参数默认传递的是什么?
Python参数的这一特性有两个好处:
-
节省内存
-
对于可变类型且函数中修改元素的内容,所有的地方都会修改。可变类型:列表、字典、集合。
其他很多编程语言执行函数时,默认传参时会将数据重新拷贝一份,会浪费内存。
提示注意:其他语言也可以通过 ref 等关键字来实现传递内存地址。
当然,如果你不想让外部的变量和函数内部参数的变量一致,也可以选择将外部值拷贝一份,再传给函数。
1.2 函数的返回值是内存地址
上述代码的执行过程:
执行func函数
data = [11, 22, 33] 创建一块内存区域,内部存储[11,22,33],data变量指向这块内存地址。
return data 返回data指向的内存地址
v1接收返回值,所以 v1 和 data 都指向 [11,22,33] 的内存地址(两个变量指向此内存,引用计数器为2)
由函数执行完毕之后,函数内部的变量都会被释放。(即:删除data变量,内存地址的引用计数器-1)
所以,最终v1指向的函数内部创建的那块内存地址。
上述代码的执行过程:
-
执行func函数
-
data = [11, 22, 33]
创建一块内存区域,内部存储[11,22,33]
,data变量指向这块内存地址 1000001110。 -
return data
返回data指向的内存地址 -
v1接收返回值,所以 v1 和 data 都指向
[11,22,33]
的内存地址(两个变量指向此内存,引用计数器为2) -
由函数执行完毕之后,函数内部的变量都会被释放。(即:删除data变量,内存地址的引用计数器-1)
所以,最终v1指向的函数内部创建的那块内存地址。(v1指向的1000001110内存地址)
-
执行func函数
-
data = [11, 22, 33]
创建一块内存区域,内部存储[11,22,33]
,data变量指向这块内存地址 11111001110。 -
return data
返回data指向的内存地址 -
v2接收返回值,所以 v1 和 data 都指向
[11,22,33]
的内存地址(两个变量指向此内存,引用计数器为2) -
由函数执行完毕之后,函数内部的变量都会被释放。(即:删除data变量,内存地址的引用计数器-1)
1.3 参数的默认值【面试题】
这个知识点在面试题中出现的概率比较高,但真正实际开发中用的比较少。
def func(a1,a2=18):
print(a1,a2)
原理:Python在创建函数(未执行)时,如果发现函数的参数中有默认值,则在函数内部会创建一块区域并维护这个默认值。
执行函数未传值时,则让a2指向 函数维护的那个值的地址。
func("root")
执行函数传值时,则让a2指向新传入的值的地址。
func("admin",20)
在特定情况【默认参数的值是可变类型 list/dict/set】 & 【函数内部会修改这个值】下,参数的默认值 有坑 。
-
坑
-
大坑
-
深坑
1.4 动态参数
动态参数,定义函数时在形参位置用 *或**
可以接任意个参数。
在定义函数时可以用 *和**
,其实在执行函数时,也可以用。
-
形参固定,实参用
*和**
-
形参用
*和**
,实参也用*和**
所以,在使用format字符串格式化时,可以可以这样:
2. 函数和函数名
函数名其实就是一个变量,这个变量只不过代指的函数而已。
注意:函数必须先定义才能被调用执行(解释型语言)。
2.1 函数做元素
既然函数就相当于是一个变量,那么在列表等元素中是否可以把行数当做元素呢?
注意:函数同时也可被哈希,所以函数名通知也可以当做 集合的元素、字典的键。
掌握这个知识之后,对后续的项目开发有很大的帮助,例如,在项目中遇到根据选择做不同操作时:
-
情景,例如:某个特定情况,要实现发送短信、微信、邮件。
上述情景,在参数相同时才可用,如果参数不一致,会出错。所以,在项目设计时就要让程序满足这一点,如果无法满足,也可以通过其他手段时间,例如:
情景:
2.2 函数名赋值
-
将函数名赋值给其他变量,函数名其实就个变量,代指某函数;如果将函数名赋值给另外一个变量,则此变量也会代指该函数,例如:
-
对函数名重新赋值,如果将函数名修改为其他值,函数名便不再代指函数,例如:
注意:由于函数名被重新定义之后,就会变量新被定义的值,所以大家在自定义函数时,不要与python内置的函数同名,否则会覆盖内置函数的功能
2.3 函数名做参数和返回值
函数名其实就一个变量,代指某个函数,所以,他和其他的数据类型一样,也可以当做函数的参数和返回值。
-
参数
-
返回值
3.返回值和print
对于初学者的同学,很多人都对print和返回值分不清楚,例如:
这两个函数是完全不同的
-
在函数中使用print,只是用于在某个位置输出内容而已。
-
在函数中使用return,是为了将函数的执行结果返回给调用者,以便于后续其他操作。
在调用并执行函数时,要学会分析函数的执行步骤。
4. 作用域
作用域,可以理解为一块空间,这块空间的数据是可以共享的。通俗点来说,作用域就类似于一个房子,房子中的东西归里面的所有人共享,其他房子的人无法获取。
4.1 函数为作用域
Python以函数为作用域,所以在函数内创建的所有数据,可以此函数中被使用,无法在其他函数中被使用。
学会分析代码,了解变量到底属于哪个作用域且是否可以被调用:
4.2 全局和局部
Python中以函数为作用域,函数的作用域其实是一个局部作用域。
COUNTRY
和CITY_LIST
是在全局作用域中,全局作用域中创建的变量称之为【全局变量】,可以在全局作用域中被使用,也可以在其局部作用域中被使用。
download
和upload
函数内部维护的就是一个局部作用域,在各自函数内部创建变量称之为【局部变量】,且局部变量只能在此作用域中被使用。局部作用域中想使用某个变量时,寻找的顺序为:优先在局部作用域中寻找,如果没有则去上级作用域中寻找
。
注意:全局变量一般都是大写。
示例1:在局部作用域中读取全局作用域的变量。
4.3 global关键字
默认情况下,在局部作用域对全局变量只能进行:读取和修改内部元素(可变类型),无法对全局变量进行重新赋值。
-
读取
-
修改内部元素(可变类型)
-
无法对全局变量重新赋值
如果想要在局部作用域中对全局变量
重新赋值,则可以基于 global
关键字实现,例如:
总结
-
函数参数传递的是内存地址。
-
想重新创建一份数据再传递给参数,可以手动拷贝一份。
-
特殊:参数是动态参数时,通过*或**传参时,会将数据循环添加到参数中(类似于拷贝一份)
def fun(*args, **kwargs): print(args, kwargs) fun(*[11, 22, 33], **{"k1": 1, "k2": 2})
-
-
函数的返回值也是内存地址。(函数执行完毕后,其内部的所有变量都会被销毁,引用计数器为0时,数据也销毁)
def func(): name = [11,22,33] data = name func()
def func(): name = [11,22,33] return name data = func() while True: print(data)
-
当函数的参数有默认值 & 默认值是可变类型 & 函数内部会修改内部元素(有坑)
# 内部会维护一个列表 [],只要b不传值则始终使用都是这个列表。 def func(a,b=[]): b.append(a)
-
定义函数写形式参数时可以使用
*
和**
,执行函数时也可以使用。 -
函数名其实也是个变量,他也可以做列表、字典、集合等元素(可哈希)
-
函数名可以被重新赋值,也可以做另外一个函数的参数和返回值。
-
掌握 print 和 return的区别,学会分析代码的执行流程。
-
python是以函数为作用域。
-
在局部作用域中寻找某数据时,优先用自己的,自己没有就在上级作用域中寻找。
-
基于 global关键字可以在局部作用域中实现对全局作用域中的变量(全局变量)重新赋值。