一、判断对象是否属于类、判断类是否派生自某个类

class Zero:
pass class One(Zero):
pass class Two(One, Zero):
pass # 判断对象是否完全是某个类
print(type(Two()) == Two)#true
print(type(Two()) == Zero)#false
# 判断对象是否属于某个类
print(isinstance(Two(), Zero))#true
# 判断某个类是否是另一个类的子类
print(Zero in Two.__bases__)#true
print(issubclass(Two,Zero))

二、关于元类

在Python中,类型分为三个层次:元类、类、对象。

类是元类的实例化,对象是类的实例化。

在Python中,元类、类、对象三种东西相当于结点,它们之间的有向边表示继承关系。整个类型系统构成一个有向无环图。这个有向无环图分为三层,一个对象可以继承多个类。

metaclass的作用就是创建类

举一个小例子

class BaseMetaClass(type):
def __new__(cls, *args, **kwargs):
kind = type.__new__(cls, *args, **kwargs)
print(kind)
return kind class Base(metaclass=BaseMetaClass):
pass class One(Base):
pass class Two(Base):
pass

输出为

<class '__main__.Base'>
<class '__main__.One'>
<class '__main__.Two'>

在此例中,只进行了类的声明,没有进行类的实例化。在进行类的声明过程中,会调用metaclass的new方法,new方法的返回值就是一个具体的类型。类型是特殊的对象。

Python中,内置函数type起的作用非常关键,它具有两个作用

  • type(obj)查看obj的类型,type(cls)查看cls的元类
  • type(className,*arg,*karg)构造函数产生新的类

在上述代码的基础上加几句话

class Three:
pass
print(type(One))#<class '__main__.BaseMetaClass'>
print(type(Three))#<class 'type'>
print(type(type))#<class 'type'>

三、一切皆type

在Java中,一切皆是Object

在Python中,一切皆是type

def f():
pass ff = lambda: 0
x = []
for i in (list, dict, type, f, ff, x):
while type(i) != i:
print(i, end=' ')
i = type(i)
print(i)

输出为

<class 'list'> <class 'type'>
<class 'dict'> <class 'type'>
<class 'type'>
<function f at 0x000000CEE01B7F28> <class 'function'> <class 'type'>
<function <lambda> at 0x000000CEE07E5A60> <class 'function'> <class 'type'>
[] <class 'list'> <class 'type'>

四、函数中可以定义类,类中可以定义函数

函数是function类型,function是type类型

>>> def choose_class(name):
… if name == 'foo':
… class Foo(object):
… pass
… return Foo # 返回的是类,不是类的实例
… else:
… class Bar(object):
… pass
… return Bar

>>> MyClass = choose_class('foo')
>>> print MyClass # 函数返回的是类,不是类的实例
<class '__main__'.Foo>
>>> print MyClass() # 你可以通过这个类创建类实例,也就是对象
<__main__.Foo object at 0x89c6d4c>

五、使用type定义类型

type(类型名称,基类列表,属性字典)

A = type("A", (), {
'name': 'weidiao',
'age': 23,
'__str__': lambda self: '{},{}'.format(self.name, self.age)
})
x = A()
print(x)#输出为weidiao,23

六、__class__属性

__class__属性时是每一种类型必然包含的属性,该属性的取值相当于x.__class==type(x)

七、不要忘记python是一门脚本语言

下面定义了一个类A,它有name和age两个属性,但是我可以随意给这个对象添加属性。

A = type("A", (), {
'name': 'weidiao',
'age': 23,
'__str__': lambda self: '{},{},{}'.format(self.name, self.age,self.baga)#如果没有baga属性,此处会报错
})
x = A()
x.baga='3'
print(x)

八、调用父类构造函数的两种方法

在Java和C++中,子类的构造函数都会自动调用父类的构造函数。

在Python中,子类是不会自动调用父类的构造函数的,因为在Python中没有构造函数,而只有初始化函数!而初始化函数只相当于普普通通的函数,所以,这就需要我们手动来调用父类的构造函数。

首先来看,如果不调用父类构造函数会出现什么情况

class A:
def __init__(self, name, age):
self.name = name
self.age = age def __str__(self):
return self.name + str(self.age) class B(A):
def __init__(self):
pass b = B()
print(b)

在此例中,B类没有调用A类的初始化函数,所以B类就缺少了name和age属性,B类的str相当于A类的str,print时就会因为缺少name和age属性报错

AttributeError: 'B' object has no attribute 'name'

有两种方法调用父类的构造函数

class A:
def __init__(self, name, age):
self.name = name
self.age = age def __str__(self):
return self.name + str(self.age) class B(A):
def __init__(self):
# A.__init__(self,'weidiao',23)
super(B,self).__init__('weidiao',23) b = B()
print(b)

这两种方法第一种方法比较灵活,完全可以取代第二种方法。

A.__init__(self,args)这种方式只初始化了父类A

super(B,self).__init__(self,args)这种方式初始化了第一个父类(根据继承的顺序),所以这种方式是完全可以用FirstFather.__init()进行取代的。

九、单例

Java中的对象的创建和对象的初始化这两个过程是紧密耦合在一起的,对象创建过程指的是为对象分配必要的内存空间,对象初始化指的是向内存中写入值。简言之,Java中一旦调用构造函数,对象空间开辟和对象初始化两种操作就同时完成了。在Java中,单例只创建一次且只初始化一次。

在Python中不是这样,Python中对象创建过程和对象初始化过程是分开的。class的__new__()函数负责对象的创建,class的__init__()负责对象的初始化。在Python中,单例可能创建一次,初始化多次。

下面看一种单例的写法:

class Dog:
dog = None def __new__(cls, *args, **kwargs):
if cls.dog is None:
cls.dog = object.__new__(cls)
print('create singleton over')
return cls.dog def __init__(self, name):
print('init is called')
self.name = name # 下面这句话会报错,因为Dog.dog目前为None
# print(Dog.dog.name)
x = Dog('x')
y = Dog('y')
print(x.name) # y
print(y.name) # y

此例中,会发现init is called输出两次,但是x和y实际上是同一个对象。也就是说,在Python中,只要调用对象的构造函数,必然会先调用__new__()函数,__new__()返回一个对象,然后Python会自动调用返回对象的__init__()函数。

上面例子的另一种实现是把重写new函数的工作放到一个父类里面,这样就能够实现:只需要继承父类就能够让自己变成单例。

class Singleton:
_instance = None def __new__(cls, *args, **kwargs):
if not cls._instance:
cls._instance = object.__new__(cls)
return cls._instance class Dog(Singleton):
def __init__(self, name):
self.name = name x = Dog('x')
y = Dog('y')
print(x.name, y.name)

在Python中,我们创建的任何一个对象,无需重写setattr方法即可随便往对象上“悬挂”成员变量。

class Ha:
pass x = Ha()
x.dog = 3
print(x.dog)

单例的实现

from functools import lru_cache

class Singleton:
@lru_cache()
def __new__(cls, *args, **kwargs):
obj = object.__new__(cls)
obj.inited = False
return obj class Haha(Singleton):
def __init__(self, name):
if self.inited:
return
self.inited = True
self.name = name
print("inited") x = Haha("x")
y = Haha("y")
print(x.name)
print(y.name)

总结

在语言的完美、统一程度上,Python无人可及,Java都比它差一点。

Python面向对象几个知识点的更多相关文章

  1. python面向对象进阶(八)

    上一篇<Python 面向对象初级(七)>文章介绍了面向对象基本知识: 面向对象是一种编程方式,此编程方式的实现是基于对 类 和 对象 的使用 类 是一个模板,模板中包装了多个“函数”供使 ...

  2. python 面向对象(进阶篇)

    上一篇<Python 面向对象(初级篇)>文章介绍了面向对象基本知识: 面向对象是一种编程方式,此编程方式的实现是基于对 类 和 对象 的使用 类 是一个模板,模板中包装了多个“函数”供使 ...

  3. python 面向对象学习

    ------Python面向对象初 下面写一个类的简单实用,以便方便理解类 #python 3.5环境,解释器在linux需要改变 #阅读手册查询readme文件 #作者:S12-陈金彭 class ...

  4. Python开发【第七篇】:面向对象 和 python面向对象进阶篇(下)

    Python开发[第七篇]:面向对象   详见:<Python之路[第五篇]:面向对象及相关> python 面向对象(进阶篇)   上一篇<Python 面向对象(初级篇)> ...

  5. python 面向对象进阶之内置方法

    一 isinstance(obj,cls)和issubclass(sub,super) 1.1,isinstance(obj,cls)检查是否obj是否是类 cls 的对象 class Foo(obj ...

  6. 【转】python 面向对象(进阶篇)

    [转]python 面向对象(进阶篇) 上一篇<Python 面向对象(初级篇)>文章介绍了面向对象基本知识: 面向对象是一种编程方式,此编程方式的实现是基于对 类 和 对象 的使用 类 ...

  7. python 面向对象(进阶篇)转载武沛齐

    上一篇<Python 面向对象(初级篇)>文章介绍了面向对象基本知识: 面向对象是一种编程方式,此编程方式的实现是基于对 类 和 对象 的使用 类 是一个模板,模板中包装了多个“函数”供使 ...

  8. Python 面向对象 (进阶篇)

    <Python 面向对象(初级篇)>文章介绍了面向对象基本知识: 面向对象是一种编程方式,此编程方式的实现是基于对 类 和 对象 的使用 类 是一个模板,模板中包装了多个“函数”供使用(可 ...

  9. python 面向对象初级篇

    Python 面向对象(初级篇) 概述 面向过程:根据业务逻辑从上到下写垒代码 函数式:将某功能代码封装到函数中,日后便无需重复编写,仅调用函数即可 面向对象:对函数进行分类和封装,让开发" ...

随机推荐

  1. xgboost入门与实战

    xgboost入门与实战(实战调参篇) https://blog.csdn.net/sb19931201/article/details/52577592 前言 前面几篇博文都在学习原理知识,是时候上 ...

  2. Linux中盘符的两种挂载方法

    相信接触过Linux系统的人对于mount命令都不陌生,今天是农历2017年的最后一个工作日,趁着时光还在,就说一下两种不同的挂载方法吧. 课前小知识: 命令格式:mount [-t vfstype] ...

  3. 与IE的战斗

    对第2版的改进,工作量几乎都在UI上,不断的写css,写js,还别说,总体挺愉快的.特别是把360浏览器用顺了之后,烦人的无法刷新问题也不能困扰我了,改了js或者css文件的话,只要清除一下缓存,就可 ...

  4. You must have a copy of the scp binary locally to use the scp feature

    在运行docker-machine scp 命令的时候,报错: "You must have a copy of the scp binary locally to use the scp ...

  5. .NET Framework 工具

    您可以使用 .NET Framework 工具轻松创建.部署和管理面向 .NET Framework 的应用程序和组件. 此节中介绍的大部分 .NET Framework 工具将自动随 Visual ...

  6. powerdesigner 生成数据库脚本

    PowerDesigner导出所有SQL脚本 操作:Database=>Generate Database PowerDesigner怎么导出建表sql脚本 1 按照数据库类型,切换数据库. D ...

  7. 关于Selenium Chrome Driver相关的一些资源

    这里摘录一些处理所需要的jar包,以及对照关系等. 参考: selenium-chrome-driver-2.22.0.jar:http://www.java2s.com/Code/Jar/s/Dow ...

  8. Python3中使用Mysql的用法。

    一.Python2中一般使用MySqldb来调用Mysql,但是在Python3中不支持该包,使用pymysql来代替了,用法一模一样. 二.安装: pip install pymysql 三.例子: ...

  9. 【Python】文件读写操作

    Python的文件读写有点类似php的文件读写.php的文件读写已经在<[php]让记事本成为你调控变量的控制台>(点击打开链接)说过了,以下用一个小样例说明Python的文件读写. 在F ...

  10. 斯坦福《机器学习》Lesson1-3感想-------3、线性回归二

    从上一篇可知.在监督学习里最重要的就是确定假想函数h(θ),即通过使得代价函数J(θ)最小,从而确定h(θ). 上一篇通过梯度下降法求得J(θ)最小,这篇我们将使用矩阵的方法来解释. 1.普通最小二乘 ...