Python -- 面向对象编程类和实例访问限制

Posted limalove

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Python -- 面向对象编程类和实例访问限制相关的知识,希望对你有一定的参考价值。

面向对象编程

      Object Oriented Programming,简称OOP,是一种程序设计思想。OOP把对象作为程序的基本单元,一个对象包含了数据和操作数据的函数。

      面向过程的程序设计把计算机程序视为一系列的命令集合,即一组函数的顺序执行。为了简化程序设计,面向过程把函数继续切分为子函数,即把大块函数通过切割成小块函数来降低系统的复杂度。

      而面向对象的程序设计把计算机程序视为一组对象的集合,而每个对象都可以接收其他对象发过来的消息,并处理这些消息,计算机程序的执行就是一系列消息在各个对象之间传递。

     在Python中,所有数据类型都可以视为对象,当然也可以自定义对象。自定义的对象数据类型就是面向对象中的类(Class)的概念。

     面向对象的三大特点:数据封装、继承、多态

     面向对象的设计思想是抽象出类Class,然后根据类Class创建实例Instance。

     面向对象的抽象程度又比函数要高,因为一个Class既包含数据,又包含操作数据的方法。

 

 

类Class

任何类最终都可以追溯到根类object.

定义:类是抽象的模板,而实例是根据类创建出来的一个个具体的“对象”,每个对象拥有的数据都互相独立,互不影响。

(定义的另一种理解:类是对一组有相同数据和相同操作的对象的定义,是一组对象共同属性和行为的抽象模板。)

创建类

  (1) 关键字class

  (2) 类名:通常是大写开头的单词,如Student

  (3)(继承类):表示该类是从哪个类继承下来的,通常,如果没有合适的继承类,就是用object类,这是所有类最终都会继承的类。

   注:当我们定义一个class的时候,实际上就定义了一种数据类型。定义的class数据类型和Python自带的数据类型没什么区别,比如str, dict, list。

创建实例:类名+()

方法(Method):是指在类的内部定义的用来访问实例数据的函数。

特殊方法:__init__方法

  • 作用:绑定属性
  • 注意:特殊方法“__init__”前后分别有两个下划线!!!

小结

和普通函数相比,在类中定义的函数,第一个参数永远是实例变量self,并且调用时,不用传递self参数,python解释器会把实例变量传递给self。

 

 

私有变量(Private)的2种形式

私有变量的作用:确保了外部代码不能随意修改对象内部的状态,这样通过访问限制的保护,代码更加健壮。

   (1) 一个下划线开头,如_name

      这种形式的私有变量外部是可以访问的,但是,按照约定俗成的规定,当你看到这样的变量时,意思是:“虽然我可以被访问,但是,请把我视为私有变量,不要随意访问。”

   (2)两个下划线开头,如__name

问题:双下划线开头的实例变量是不是一定不能从外部访问呢?

答:也不是,只是不能直接访问__name,是因为python解释器对外把__name变量改成了_Student__name,所以,仍然可以通过_Student__name来访问__name变量。

但是强烈建议不要这么做,因为不同版本的Python解释器可能会把__name改成不同的变量名。

>>>总的来说,Python本身没有任何机制阻止你干坏事,一切全靠自觉。

 

 

继承和多态

 继承的好处:

1,继承可以把父类的所有功能都直接拿过来,这样就不必重零做起,子类只需要新增自己特有的方法。

2,继承也可以把父类不适合的方法覆盖重写。

子类、父类:

在OOP程序设计中,当我们定义一个class的时候,可以从某个现有的class继承,新的class称为子类(Subclass),而被继承的class称为基类、父类或超类(Base class、Super class)。

 

 

静态语言VS动态语言

     对于静态语言(如Java)来说,如果需要传入Animal类型,则传入的对象必须是Animal类型或者它的子类,否则,将无法调用run( )方法。

    对于Python这样的动态语言来说,如果需要传入Animal类型,则不一定要传入Animal类型,我们只需要保证传入的对象有一个run( )方法就可以了。

    动态语言的“鸭子类型”,它并不要求严格的继承体系,一个对象只要“看起来像鸭子,走起路来像鸭子”,那它就可以被看做是鸭子。

       Python的 " file-like object "就是一种鸭子类型。对于真正的文件对象来说,它有一个read( )方法,返回其内容。但是,许多对象,只要有read( )方法,都可

以被看做" file-like object ",即你不一定要传入真正的文件对象,可以传入任何实现了read( )方法的对象。

>>> 动态语言的鸭子类型特点决定了继承不像静态语言那样是必须的。

 

 

获取对象信息

判断对象类型:type( )函数、isinstance( )函数

》》》总是优先使用isinstance()判断类型,可以将指定类型及其子类“一网打尽”。

isinstance( ):1,判断的是一个对象是否是该类型本身,或者位于该类型的父继承链上。2,并且还可以判断一个变量是否是某些类型中的一种(有例子)。

print(isinstance([1,2,3,4],(list,tuple)))
print(isinstance((1,2,3,4),(list,tuple)))
print(isinstance([1,2,3,4],(int,str)))

 

 type( )函数如何判断一个对象是否是函数? 方法:使用types 模块中定义的常量。

import types
def fn():
    pass

print("是否是函数类型:",type(fn)==types.FunctionType)  #判断是否是函数类型
print("是否是绝对值类型:",type(abs)==types.BuiltinFunctionType)
print("是否是lambda匿名函数类型:",type(lambda x: x)==types.LambdaType)  #判断是否是lambda匿名函数类型
print("是否是生成器类型:",type(x for x in range(10))==types.GeneratorType)  #判断是否是生成器类型

 ********************************************************

dir( )函数

功能:可以获得一个对象的所有属性和方法,它返回一个包含字符串的list列表。

在Python中,如果你调用len( )函数试图获取一个对象的长度,实际上,在len( )函数内部,它自动去调用该对象的__len__(  )方法。(这句话非常重要  好好理解)

******************************************************************

hasattr,  setattr,  getattr :获取对象的属性或方法

对于属性,getattr返回的是属性值;对于方法,getattr返回的是方法的内存地址。

getattr(  )如果试图获取不存在的属性,会抛出AttributeError的错误;可以传入一个default参数,如果属性不存在,就返回默认值。

hasattr(obj,x)   #判断对象是否有属性x
setattr(x,y,z)   #为对象x的属性y设置值z
getattr(obj,x,default)   #获取对象的某个属性值value,default可以省略

 

小结

一个正确的用法的例子如下:

def readImage(fp):
    if hasattr(fp, \'read\'):
        return readData(fp)
    return None

假设我们希望从文件流fp中读取图像,我们首先要判断该fp对象是否存在read方法,如果存在,则该对象是一个流,如果不存在,则无法读取。hasattr( )就派上了用场。

请注意,在Python这类动态语言中,根据鸭子类型,有read(  )方法,不代表该fp对象就是一个文件流,它也可能是网络流或者是内存中的一个字节流,但只要read( )方法返回的是有效的图像数据,就不影响读取图像的功能。

 

 

 

实例属性和类属性

实例属性属于各个实例所有,互不干扰;

类属性属于类所有,但所有实例都可以访问。

不要对实例属性和类属性使用相同的名字,因为相同名称的实例属性将屏蔽掉类属性,但是当你删除实例属性后,再使用相同的名称,访问到的将是类属性。

 

以上是关于Python -- 面向对象编程类和实例访问限制的主要内容,如果未能解决你的问题,请参考以下文章

面向对象编程(python)和部分面向对象高级编程

Python 面向对象编程——访问限制

Python中面向对象(学习笔记)

python面向对象编程

python学习day11 面向对象编程 类和实例

Python入门-6面向对象编程:02类的定义-类和对象的关系-构造函数-实例属性-实例方法