本文主要记录python相关的原理知识,目的是为了记录python语言中的原理性规则性思考。
预编译
class A:
def __init__(self, a):
self.a = a
def p(self, b=self.a):
print b
gives an error NameError: name ‘self’ is not defined?
Default argument values are evaluated at function define-time, but self is an argument only available at function call time. Thus arguments in the argument list cannot refer each other. 函数定义阶段是不可以引用类相关变量的。
It’s a common pattern to default an argument to None and add a test for that in code:
def p(self, b=None):
if b is None:
b = self.a
print b
share
函数变量传输
1、介绍python函数传参
- Python不允许程序员选择采用传值还是传引用。Python参数传递采用的肯定是“传对象引用”的方式。实际上,这种方式相当于传值和传引用的一种综合。如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值——相当于通过“传引用”来传递对象。如果函数收到的是一个不可变对象(比如数字、字符或者元组)的引用,就不能直接修改原始对象——相当于通过“传值”来传递对象。
- 当人们复制列表或字典时,就复制了对象列表的引用同,如果改变引用的值,则修改了原始的参数。
- 为了简化内存管理,Python通过引用计数机制实现自动垃圾回收功能,Python中的每个对象都有一个引用计数,用来计数该对象在不同场所分别被引用了多少次。每当引用一次Python对象,相应的引用计数就增1,每当消毁一次Python对象,则相应的引用就减1,只有当引用计数为零时,才真正从内存中删除Python对象。
上面也就是说,当我们传的参数是int、字符串(string)、float、(数值型number)、元组(tuple) 时,无论函数中对其做什么操作,都不会改变函数外这个参数的值;
当传的是字典型(dictionary)、列表型(list)时,如果是重新对其进行赋值,则不会改变函数外参数的值,如果是对其进行诸如list append之类的操作,则会改变变量数值。即变量中存储的是引用 , 是指向真正内容的内存地址 , 对变量重新赋值 , 相当于修改了变量副本存储的内存地址 , 则此时的变量和函数体外的变量不是同一个了。 而在函数体之外的变量 , 依旧存储的是原本的内存地址 , 其值自然没有发生改变 。
def lists(l1, l2):
l1.append(1)
l2 = [1, 2, 3]
list1 = [9, 8, 7]
list2 = [9, 8, 7]
print (list1, list2)
lists(list1, list2)
print (list1, list2)
打印结果:
[9, 8, 7] [9, 8, 7]
[9, 8, 7, 1] [9, 8, 7]