魔法属性

今天给大家介绍一些Python中的类属性中存在的一些具有特殊含义的属性,常常被我们称为魔法属性,详情如下:

1. __doc__

表示类的描述信息

class Person(object):
    '''这是类的描述信息'''
    country = 'China'

    def __init__(self,name,age):
        # print('__init__')
        self.name = name
        self.age = age
p = Person('laowang',20)
print(Person.__doc__) #这是类的描述信息

2. __module__和__class__

  • __module__: 表示当前操作的对象在那个模块
  • __class__:表示当前操作的对象的类是什么
class Person(object):
    '''这是类的描述信息'''
    country = 'China'

    def __init__(self,name,age):
        # print('__init__')
        self.name = name
        self.age = age
p = Person('laowang',20)
print(p.__module__) #__main__
print(p.__class__) #<class '__main__.Person'>

3. __init__

初始化方法,通过类创建对象时,自动调用该方法。

class Person(object):

    '''这是类的描述信息'''
    country = 'China'

    def __init__(self,name,age):
        print('__init__')
        self.name = name
        self.age = age
p = Person('laowang',20) # __init__

4. __del__

当对象在内存中被释放后,该方法自动被调用。

class Person(object):

    '''这是类的描述信息'''
    country = 'China'

    def __init__(self,name,age):
        print('__init__')
        self.name = name
        self.age = age
    def __del__(self):
        print('__del__')
        # pass
p = Person('laowang',20)
del p  # __del__

5. __call__

当对象可以被调用的时候,自动调用该方法,即 对象.( )

class Person(object):

    '''这是类的描述信息'''
    country = 'China'

    def __init__(self,name,age):
        print('__init__')
        self.name = name
        self.age = age
    def __call__(self, *args, **kwargs):
        print('__call__')

p = Person('laowang',20)
p() #__call__

6. __dict__

表示类或对象中的所有属性。

class Person(object):

    '''这是类的描述信息'''
    country = 'China'

    def __init__(self,name,age):
        print('__init__')
        self.name = name
        self.age = age

p = Person('laowang',20)
print(Person.__dict__) # {'__module__': '__main__', '__doc__': '这是类的描述信息', 'country': 'China', '__init__': <function Person.__init__ at 0x104ce7378>, '__dict__': <attribute '__dict__' of 'Person' objects>, '__weakref__': <attribute '__weakref__' of 'Person' objects>}
print(p.__dict__)  # {'name': 'laowang', 'age': 20}

7. __str__

打印对象的时候,如果需要打印对象的详细信息,而不只是打印对象的地址,可以重写该方法。

class Person(object):

    '''这是类的描述信息'''
    country = 'China'

    def __init__(self,name,age):
        print('__init__')
        self.name = name
        self.age = age
    def __str__(self):
        return 'haha'
p = Person('laowang',20)
print(p) #haha

8. __getitem__、__setitem__、__delitem__

用于索引操作,如字典。以上分别表示获取、设置、删除数据

class Person(object):

    '''这是类的描述信息'''
    country = 'China'

    def __init__(self,name,age):
        print('__init__')
        self.name = name
        self.age = age

    def __getitem__(self, key):
        print('__getitem__',key)

    def __setitem__(self, key, value):
        print('__setitem__',key,value)

    def __delitem__(self, key):
        print('__delitem__',key)

p2 = Person('xiaoli',18)
print(p2['a'])
p2['b'] = 'lala'
del p2['a']

9. __getitem__、__setitem__、__delitem__

这三个方法在Python2.7中还存在,但在Python3中就取消了。

class Foo(object):

    def __getslice__(self,i,j):
        print('__getslice',i,j)

    def __setslice__(self, i, j, sequence):
        print('__setslice__',i,j)

    def __delslice__(self, i, j):
        print('__delslice__',i,j)


obj = Foo()

obj[-1:1]
obj[0:1] = [11,22,33,44]
del obj[0:2]

10. __int__

__int__方法,在对象被int()包裹的时候会被执行,例如int(obj),如果obj对象没有__int__方法,那么就会报错。在这个方法中返回的值被传递到int类型中进行转换。

class Person(object):

    '''这是类的描述信息'''
    country = 'China'

    def __init__(self,name,age):
        print('__init__')
        self.name = name
        self.age = age

    def __int__(self):
        return 100

p1 = Person('laowang',30)
print(int(p)) #100

11. __add__

__add__方法在两个对象相加的时候,调用第一个对象的__add__方法,将第二个对象传递进来,至于怎么处理以及返回值,那是程序员自定义的.

class Person(object):

    '''这是类的描述信息'''
    country = 'China'

    def __init__(self,name,age):
        print('__init__')
        self.name = name
        self.age = age

    def __add__(self, other):
        return self.age + other.age

p1 = Person('laowang',30)
p2 = Person('xiaowang',20)
print(p1 + p2) # 50

12. __iter__

类的对象如果想要变成一个可迭代对象,那么对象中必须要有__iter__方法,并且这个方法返回的是一个迭代器。

for 循环的对象如果是一个可迭代的对象,那么会先执行对象中的iter方法,获取到迭代器,然后再执行迭代器中的__next__方法获取数据。如果for循环的是一个迭代器,那么直接执行迭代器中的__next__方法。

class Person(object):

    '''这是类的描述信息'''
    country = 'China'

    def __init__(self,name,age):
        print('__init__')
        self.name = name
        self.age = age

    def __iter__(self):
        return iter([1,2,3,4])

p1 = Person('laowang',30)
for num in p1:
    print(num)

13. __new__、__metaclass__

在python中,一切皆对象,我们定义的类其实也是一个对象,那么,类本身是谁的对象呢?在python2.2之前(或者叫经典类中),所有的类,都是class的对象,但是在新式类中,为了将类型(int,str,float等)和类统一,所以,所有的类都是type类型的对象。当然,这个规则可以被修改,在类中有一个属性 __metaclass__(元类) 可以指定当前类该由哪个类进行实例化。而创建对象过程中,其实构造器不是__init__方法,而是__new__方法,这个方法会返回一个对象,这才是对象的构造器。

class Mytype(type):

    def __init__(self, what, bases=None, dict=None):
        super(Mytype,self).__init__(what, bases, dict)

    def __call__(self, *args, **kwargs):
        obj=self.__new__(self)
        self.__init__(obj, *args, **kwargs)
        return obj

class Foo:
    __metaclass__=Mytype

    def __init__(self,name,age):
        self.name=name
        self.age=age

    def __new__(cls, *args, **kwargs):
        return object.__new__(cls)

obj=Foo("xiaoming",18)
print(obj.name,obj.age) #iaoming 18
  
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容

  • 〇、前言 本文共108张图,流量党请慎重! 历时1个半月,我把自己学习Python基础知识的框架详细梳理了一遍。 ...
    Raxxie阅读 18,942评论 17 410
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,094评论 1 32
  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,940评论 6 13
  • 这里是一个16岁的少女 笔名邓七七,大家叫我七七就好啦 平时爱写诗,爱自拍 写出来的东西是自己的人生感悟 虽然年龄...
    邓七七阅读 438评论 0 0
  • 互斥锁使用: @synchronized{锁对象}{//需要锁定的代码} 互斥锁: 能有效防止因多线程抢夺资源造成...
    frankisbaby阅读 268评论 0 0