与c/c++不同,Python/Java中的变量都是引用类型,没有值类型

Python赋值语句由三部分构成,例如:

  int   a  = 1

类型 标识 值

标识(identity):用于唯一标识一个对象(指向对象的内存地址);类型(type);值(value)

Python使用标识访问对象的值,也就是说Python中的对象其实相当于c的指针

引用(reference):对象的内存地址

可变类型:重新赋值时直接更改内存中原有对象,变量引用不变。float、list、dict

不可变类型:重新赋值时在内存中创建一个新对象,同时改变变量的引用。int、char、tuple

这样设置是为了提高效率,比如1、2这样的变量可能经常被用到,一次定义多次引用的话效率会比较高

知道了Python变量的内存机制,就可以避免一些坑。比如而a、b的类型都是List,令a=b后,对a、b任意一个List进行修改,都会影响另外一个List的值,因为他们实际指向同一个内存地址

 

要注意区分“对象(object)”和“变量(variable)”,对象是在程序中创建的,储存在内存中,变量是对象的引用(reference),变量赋值是让变量指向对象,一个对象可以被多个变量指向

python中所有的数据类型都是对象

可变对象(列表、字典、集合)的改变会影响所有指向该对象的变量

不可变对象(字符串、整型、元组),所有指向该对象的变量的值总是一样的,也不会改变,通过某些操作更新其值时会返回一个新的对象【一旦创建,不可改变】

变量可以被删除,但对象不可以,只能被垃圾回收机制回收



1 l1=[1,2,3]
2 print(id(l1))
3 l1.append(4)
4 l2=[1,2,3]
5 l3=l2
6 print(id(l1))
7 print(id(l2))
8 print(id(l3))
9
10 v1=1
11 print(id(v1))
12 v1=2
13 v2=1
14 v3=v2
15 print(id(v1))
16 print(id(v2))
17 print(id(v3))


输出:

154977800

154977800

138660424

138660424

8791557861120

8791557861152

8791557861120

8791557861120

可见,可变对象每次创建时都会创建一个新的对象,随后发生改变会影响所有指向它的变量



1 test=[1]
2 def change1(a):
3 a = [2]
4 print(id(a))
5 def change2(a):
6 a.append(3)
7 print(id(a))
8 print(id(test))
9 change1(test)
10 change2(test)
11 print(test)


输出:

154977800

128902536

154977800

[1, 3]

第一个函数是重新赋值,第二个函数是改变原对象的值自建对象属于可变对象

 



1 class Dog():
2 def __init__(self,name,age):
3 self.name = name
4 self.age = age
5
6 dog1 = Dog('Sam',2)
7 print(id(dog1))
8 dog2 = Dog('Tom',1)
9 print(id(dog2))
10 dog1.name = 'Petter'
11 print(id(dog1))


输出:

155199584

155198408

155199584

自建对象属于可变对象