Python3学习之路~7.2 类的特殊成员方法

1. __doc__ 表示类的描述信息

class Dog(object):
    """ 这个类是描述狗这个对象的 """
    def func(self):
        pass

print(Dog.__doc__)
# 输出: 这个类是描述狗这个对象的

2. __module__ 和  __class__ 

  __module__ 表示当前操作的对象在哪个模块

  __class__     表示当前操作的对象的类是什么

class C:

    def __init__(self):
        self.name = ‘aa‘

lib/aa.py

from lib.aa import C

obj = C()
print obj.__module__  # 输出 lib.aa,即:输出模块
print obj.__class__      # 输出 lib.aa.C,即:输出类

index

3. __init__ 构造方法,通过类创建对象时,自动触发执行。

4.__del__ 析构方法,当对象在内存中被释放时,自动触发执行。

5. __call__ 对象后面加括号,触发执行。

注:构造方法的执行是由创建对象触发的,即:对象 = 类名() ;而对于 __call__ 方法的执行是由对象后加括号触发的,即:对象() 或者 类()()
class Foo:
    def __init__(self):
        print("__init__")

    def __call__(self, *args, **kwargs):
        print(‘__call__‘)

obj = Foo()  # 执行 __init__
obj()  # 执行 __call__

6. __dict__ 查看类或对象中的所有成员   

class Province:
    country = ‘China‘

    def __init__(self, name,count):
        self.name = name
        self.count = count
    def func(self):
        print("func")

# 获取类的成员,不包括实例属性
print(Province.__dict__)
# 输出:
# {‘__module__‘: ‘__main__‘, ‘country‘: ‘China‘, ‘__init__‘: <function Province.__init__ at 0x00000000027EA9D8>,
# ‘func‘: <function Province.func at 0x00000000027EAA60>, ‘__dict__‘: <attribute ‘__dict__‘ of ‘Province‘ objects>,
# ‘__weakref__‘: <attribute ‘__weakref__‘ of ‘Province‘ objects>, ‘__doc__‘: None}

# 获取 对象obj1 的成员,不包括类属性
obj1 = Province("shandong",1000)
print(obj1.__dict__)
# 输出:{‘name‘: ‘shandong‘, ‘count‘: 1000}

7.__str__ 如果一个类中定义了__str__方法,那么在打印 对象 时,默认输出该方法的返回值。

class Foo:
     def __str__(self):
        return ‘alex li‘

obj = Foo()
print obj
# 输出:alex li

8.__getitem__、__setitem__、__delitem__

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

class Foo(object):
    def __getitem__(self, key):
        print(‘__getitem__‘, key)

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

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

obj = Foo()
result = obj[‘k1‘]  # 自动触发执行 __getitem__
obj[‘k2‘] = ‘alex‘  # 自动触发执行 __setitem__
del obj[‘k1‘] # 自动触发执行__delitem__

可用于写一个类 封装属于自己的字典,django里面会用到它,用来封装自己的一些底层的东西,用户以为自己是调用的字典,其实是调用的实例。可以对某些key加一些限制,达到使用户可以访问字典,却不能对其进行删除的目的。

9. __new__ \ __metaclass__

class Foo(object):
    def __init__(self,name):
        self.name = name

f = Foo("Alex")

上述代码中,f 是通过 Foo 类实例化的对象,其实,不仅 obj 是一个对象,Foo类本身也是一个对象,因为在Python中一切事物都是对象

如果按照一切事物都是对象的理论:f 对象是通过执行Foo类的构造方法创建,那么Foo类对象应该也是通过执行某个类的 构造方法 创建。

print(type(f)) # 输出:<class ‘__main__.Foo‘>     表示,f 对象由 Foo 类创建
print(type(Foo)) # 输出:<type ‘type‘>            表示,Foo类对象由 type 类创建

所以,f对象是Foo类的一个实例Foo类对象是 type 类的一个实例,即:Foo类对象 是通过type类的构造方法创建。

那么,创建类就可以有两种方式:

a). 普通方式

class Foo(object):

    def talk(self):
        print(‘hello Alice‘)

b). 特殊方式

def talk(self):
        print(‘hello Alice‘)

Foo = type(‘Foo‘, (object,), {‘talk‘: talk})
#注意:这里的(object,)是元组的形式,必须后面加一个逗号,如果不加,Python会认为(object)是一个值

def talk(self):
    print("hello %s" % self.name)

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

Foo = type("Foo",(object,),{"talk":talk,"__init__":__init__})

加上构造方法

So ,记住,类 是由 type 类实例化产生

那么问题来了,类默认是由 type 类实例化产生,type类中如何实现的创建类?类又是如何创建对象?

答:类中有一个属性 __metaclass__,其用来表示该类由 谁 来实例化创建,所以,我们可以为 __metaclass__ 设置一个type类的派生类,从而查看 类 创建的过程。

class MyType(type):
    def __init__(self,*args,**kwargs):
        print("Mytype __init__")

    def __call__(self, *args, **kwargs):
        print("Mytype __call__")
        obj = self.__new__(self)
        self.__init__(obj,*args, **kwargs)

    def __new__(cls, *args, **kwargs):
        print("Mytype __new__")
        return type.__new__(cls, *args, **kwargs)

class Foo(object,metaclass = MyType):
    # __metaclass__ = MyType
    def __init__(self,name):
        self.name = name
        print("Foo __init__")

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

f = Foo("Alex")

# 输出:
# Mytype __new__
# Mytype __init__
# Mytype __call__
# Foo __new__
# Foo __init__

类创建的过程

类的生成 调用 顺序依次是 __new__ --> __init__ --> __call__

metaclass 详解文章:http://stackoverflow.com/questions/100003/what-is-a-metaclass-in-python 得票最高那个答案写的非常好

原文地址:https://www.cnblogs.com/zhengna/p/9618190.html

时间: 2024-10-02 10:45:11

Python3学习之路~7.2 类的特殊成员方法的相关文章

Python3 学习第五弹:类与面向对象

对于面向对象总是要提到,万物皆对象.好似博大精深的感觉. 接下来一起看看python的面向对象的例子 创建一个对象 class Person: type = 'person' def __init__(self, name = 'Noname'): self.name = name def hello(self): print("hello, I'm " + self.name) >>> Mike = Person('Mike') >>> Mike.

【C++学习之路】派生类的构造函数(二)

二.有内嵌对象的派生类 1.一般来说,我们会这样定义构造函数 student( int i, string nam, int pid, string pnam, int sid) : person( i, nam),parent(pid,pnam){ stuid = sid; } person是基类的构造函数,parent是该派生类内嵌的person子对象   2.具体例子 1 #include <iostream> 2 using namespace std; 3 class A 4 { 5

【C++学习之路】派生类的构造函数(一)

一.简单派生类的构造函数: 1.所谓简单派生类,就是指派生类中不包含基类的内嵌对象的派生类. 2.一般来说,这样的派生类的构造函数的形式是: student( int i, string nam, int sid) : person( i, nam) { stuid = sid; } person(是基类的初始化列表) 3.构造函数的初始化列表的使用 3.1所有的构造函数都可以拿参数的初始化表来构造完全属于自己的数据成员,比如: 1 #include <iostream> 2 using na

jquery学习之路之元素类的切换toggle

对元素类的控制(增删改切换)达到不同的页面效果<head><script src="/jquery/jquery-1.11.1.min.js"></script><script>$(document).ready(function(){ $("#blue").click(function(){ $("h1,h2,p").toggleClass("blue"); }); $(&qu

python3学习之路(目录)

一.Python学习(第一章) 1.python入门 -->第一个python程序 -->注释 -->变量 -->输入输出 -->流程控制 -->while循环 -->练习题 2.运算符 -->算数运算 -->比较运算 -->赋值运算 -->逻辑运算 -->成员运算 3.基本数据类型 -->整型 -->布尔型 -->字符串 -->列表 -->元组 -->字典 -->练习题 目录 持续更新中..

【C++学习之路】派生类的构造函数(三)

三.多层继承的派生类 1.多层继承的派生类只需在构造函数的初始化列表中写出直接基类的构造函数即可 1 class student 2 { 3 public: 4 student(int n, string nam) 5 { 6 num = n; name = nam; 7 } 8 }; 9 class student1 : public student 10 { 11 public: 12 student1(int n, string nam, int a) :student(n, nam) {

C++学习之路: class类外的成员函数定义 和 友元 的讨论

引言:成员函数定义在类内和类外的区别在于是否内联展开. 定义在类内一般都是内联展开的, 节省了调用函数的开销.如果函数体过于庞大,编译器会忽视内联建议 如果定义在类外,需要在类内声明,则程序运行时 是采用调用的方式 访问该函数,是非内联的方式. 1 #include <iostream> 2 #include <string> 3 #include <vector> 4 using namespace std; 5 6 class Student 7 { 8 publi

Python3学习之路~3.1 函数基本语法及特性、返回值、参数、局部与全局变量

1 函数基本语法及特性 定义: 函数是指将一组语句的集合通过一个名字(函数名)封装起来,要想执行这个函数,只需调用其函数名即可 特性: 减少重复代码 使程序变的可扩展 使程序变得易维护 语法定义: 1 def sayhi():#函数名 2 print("Hello, I'm nobody!") 3 4 sayhi() #调用函数 可以带参数 1 #下面这段代码 2 a,b = 5,8 3 c = a**b 4 print(c) 5 6 7 #改成用函数写 8 def calc(x,y)

Python3学习之路~6.1 编程范式:面向过程 VS 面向对象

编程范式 编程是程序员用特定的语法+数据结构+算法组成的代码来告诉计算机如何执行任务的过程,一个程序是程序员为了得到一个任务结果而编写的一组指令的集合,正所谓条条大路通罗马,实现一个任务的方式有很多种不同的方式,对这些不同的编程方式的特点进行归纳总结得出来的编程方式类别,即为编程范式.不同的编程范式本质上代表对各种类型的任务采取的不同的解决问题的思路, 大多数语言只支持一种编程范式,当然也有些语言可以同时支持多种编程范式.两种最重要的编程范式分别是面向过程编程和面向对象编程. 面向过程编程(Pr