2012年5月18日星期五

python mutable 和 immutable

 
 

satan 通过 Google 阅读器发送给您的内容:

 
 

于 12-5-4 通过 白杨 baiyang 作者:baiyang

对于一个大三的人来说,学习一门新的语言,其基本语法几分钟就搞定了,可是要深入其机制,可不是一二天的事。
如果你认为你了解python,那么你得首先回答如下问题?

  • 1. 是否了解动态语言的鸭子模型?
  • 2. 是否了解可变参数与关键字参数?
  • 3. 对函数式编程有初步了解。
  • 4. 是否知道列表生成式?
  • 5. 是否知道lambda/decorator/slots?
  • 6. 为什么要把缺省参数设为immutable?
  • 7. 是否知道Mixin?
  • 8. 是否知道WSGI接口?
  • 9. 是否知道异步框架如gevent/tornado?
  • 10. 是否深入了解过Python的GC和GIL?
  • 11.是否了解python对象的查找机制?

啊哈,看到这个的时候,自己傻眼了,貌似对于学了半年的python的我来说,只了解2,3,4,9和11啊,其中5,10只是略懂,可是1,7基本没有听过,那好吧,这几天就根据这些提示来逐步解释python的面纱。
今天研究了一下6,回答了什么是python的immutable,从字面上说,很好理解,理解好了它,对于python的变量申请也是一大帮助啊….

 >>> a = 1 >>> id(a) 22862960 >>> a = 2 >>> id(a) 22862948 

我一直以为a的id不会变,可是从这里看,a的id的确发生了变化,这以为着python为其重新分配了内存空间,而不是像c那样修改内存空间的值。为何会这样呢,那我们在来看一些另外一个例子

 >>> a = 1 >>> b = a >>> b += 1 >>> a 1 >>> b 2 >>> 

哎,看到这里又让我大失所望,之前一直说python的赋值就想C里面的引用一样,如果是那样的话,按理说这里的a,b的值应该一致的,可是还是让我大吃一惊。好了,看了这么多的问题,直入正题吧,就是今天python里面的类型其实也分为immutable和mutable二种,之所以会导致上面的现象,就是因为常数是immutable类型,回想之前说python任何数据都是对象,既然1,2也是对象,而且还是immutable,当然不能被b修改,所以会为b重新开辟空间存放这个immutable的对象2。
那好,如果a是一个mutable的引用呢?

 >>> a = [1, 2] >>> b = a >>> b += [3] >>> a [1, 2, 3] >>> b [1, 2, 3] >>> id(a) 27814608 >>> id(b) 27814608 >>> 

对,和你预想的是一样的,这里并没有开辟新的内存,不需要作何解释了。
那么在python那些是immutable呢?
numbers, strings, tuples, frozensets
其实,还有一种特殊情况,就是自定义的类型呢?
一般情况下,程序员自定义的python类型都是mutable的,但是如果你想定制immutable的数据类型,那么你必须重写object的__setattr__和__delattr__方法,如下:

 class Immutable(object):     def __setattr__(self, *args):         raise TypeError("can't modify the value of immutable instance")      __delattr__ = __setattr__      def __init__(self, value):          super(Immutable, self).__setattr__("value", value) 

我们可以做如下测试

 >>> x = Immutable("baiyang") >>> x.value 'baiyang' # 重新赋值 >>> x.value = "ibaiyang" Traceback (most recent call last):   File "", line 1, in    File "", line 3, in __setattr__ TypeError: can't modify immutable instance # 删除 >>> del x.value Traceback (most recent call last):   File "", line 1, in    File "", line 3, in __setattr__ TypeError: can't modify immutable instance 

好吧,今天就总结到此,之后分别把这些问题再深入理解一些。。。。python都在说很容易学,突然发觉他们都错了。。。。


 
 

可从此处完成的操作:

 
 

没有评论:

发表评论