Python面向对象编程(Object Oriented Programming)之类定义及继承

面向对象编程

面向过程编程和对象编程都是程序设计思想,面向过程程序设计是将需要实现的功能流程化列出来,让程序依次按照命令执行。跟面向过程程序设计不同,面向对象编程是将现实中的个体的状态和行为抽象化成对象的属性和方法,通过对象间的方法进行信息处理和通信模拟现实世界中的实体交互,高度抽象的设计思想可以模拟现实中的情景和需求,更方便地面对大型程序设计的需要。
Python是一门面向对象编程的语言,其将所有的数据类型都视为对象。其中最重要的概念是类和实例。类是为某一个现实对象创建出的模板,其包含了该对象所拥有的属性和方法,通过实例化对类进行个性化设定便创造出实例。

1.Class 和 Instance

类是作为对象的模板,可以在里面设定公共属性(类属性)、实例属性(对象属性)、构造器和方法

class Dog(object):
    #类属性默认用大写命名
    CATEGORY = '犬属'
    #构造器由__init__命名,括号内包括固定参数self和实例属性
    def __init__(self,name,ages):
        self.name = name
        self.__ages = ages
    #方法的括号第一个参数一定是self,可加其他参数
    def run(self,state):
        print("狗儿跑得"+state+",我叫"+self.name)
    #将属性复制功能封装在函数中提供了一定的安全性,并且可以在赋值中添加校准及其他功能。
    def setAges(self,ages):
        if type(ages) == int:
            self.__ages = ages
        else:
            print("你输入的不是数字,不能充当年龄")
    def getAges(self):
        return self.__ages
    
dog = Dog("旺财",12)
print(dog.CATEGORY)
dog.run("慢")
dog.setAges(10)
print(dog.getAges())

结果为:

>犬属
>狗儿跑得慢,我叫旺财
>10

在上面的实例属性设定中出现了__ages这样子以双下划线开头的属性定义,这标志着是private变量,不准许通过实例.变量的方式来读取,这种保护机制便确保了外部程序没办法随意修改对象内部的状态。其实本质上解释器会将该变量名转化为_Dog__ages在其前面添加上_类名

dog.__ages = 1
# 仅仅是为实例创建了一个新的属性而已,动态语言确实可以运行时添加属性
print(dog.getAges())
print(dog.__ages)
# __ages在实例化时已经变成了_Dog__ages
dog._Dog__ages = 13
print(dog.getAges())

结果是:

>10
>1
>13

注意:有些时候在类定义中出现了了一个下划线开的实例变量名,这样的实例变量在程序层面上是可以正常访问的,但是按照约定俗成的规定,这种命名形式是告诉使用者将其看成私有变量,不要从外面进行访问。

2.类属性和实例属性

类属性是指该变量归类所有,所有新创建的实例都可以访问得到,但只能通过类名索引的方式来修改勒边来那个。Python可以在类定义中用self定义实例变量,作为动态语言也可以在程序执行过程中通过实例直接创建实例变量。

>>> class Dog():
    name = "旺财"

>>> dog = Dog()
# 访问类变量
>>> dog.name
'旺财'
# 绑定新的实例属性
>>> dog.ages = 12
>>> dog.ages
12
# 实例属性覆盖类属性
>>> dog.name = "耿狗"
>>> dog.name
'耿狗'
# 删除实例属性,重新可以访问类属性
>>> del dog.name
>>> dog.name
'旺财'
# 重新定义类属性
>>> Dog.name = "lal"
>>> Dog.name
'lal'
# 测试是不是真正删除类属性
>>> del Dog.name
>>> Dog.name
Traceback (most recent call last):
  File "<pyshell#23>", line 1, in <module>
    Dog.name
AttributeError: type object 'Dog' has no attribute 'name'

python作为动态语言可以在运行过程中为实例绑定新的实例属性(对类和对其他实例无效),如上面所示为Dog实例dog绑定了新的实例属性ages。有时候我们可能希望限制绑定的属性,python提供了slots特殊类属性来限定可动态绑定的实例属性,在类定义的时候将可能需要动态绑定的实例属性名创建成一个元组赋给slots类属性,绑定除了指定属性外的实例属性将会出现错误。

class Dog(object):
    __slots__ = ('name','ages')

dog = Dog()
dog.color = yellow

结果为:

Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'Dog' object has no attribute 'color'

3.枚举类与元类

在python中可以用大写字母来设定常量,表明不可修改,但是这并不是一种语言机制而是一种自觉机制,有被篡改的风险。在python3.4中添加了枚举类便于大小一定的特殊类的定义,如季节、工作日等。
枚举类有两种不同的设定形式,一种便是按照类定义的形式继承Enum类,可以在类中自定义的设定每一个成员的具体值,或者可以用Enum函数直接对每一个成员自动编排值。

from enum import Enum,unique

# 用@unique使枚举类成员的值都是唯一的
@unique
class Season(Enum):
    spring = 1
    sunner = 2
    autumn = 3
    winter = 4
# 可以用Eunm函数直接定义
Season = Enum('Season',('spring','sunner','autumn','winter'))

其中以枚举类名.成员名的形式来索引每一个成员,每一个成员都有其名字和值,可以调用枚举类内部的members变量来获取包含全部成员情况的字典

spring = Season.spring
print(spring.name)
print(spring.value)
for name, member in Season.__members__.items():
    print(name, '=>', member, ',', member.value)

输出:

>spring
>1
>Season.sunner
>spring => Season.spring , 1
>sunner => Season.sunner , 2
>autumn => Season.autumn , 3
>winter => Season.winter , 4
>Season.spring
>Season.sunner
>Season.autumn
>Season.winter

4.继承与多态

上面举的例子Dog类继承了object类,故object称为Dog类的Super class,Dog类为Subclass,子类通过继承获得了父类的全部成分,并且拥有重新复写成分的能力。

class WangCai(Dog):
    def __init__(self,state,name,ages):
        self.state = state
        self.name = name
        self.ages = ages
    def run(self):
        print("我是子类的旺财")
        super().run(self.state)

wangcai = WangCai("慢","旺财",12)
wangcai.run()

结果为:

>我是子类的旺财
>狗儿跑得慢,我叫旺财

多态一般是指在静态类型语言中的子类可以传入需要父类做参数的函数,在运行的时候调用子类的相关内容,包含了编译时类型和运行时类型的概念。这样一来原先为父类编写的方法也可以操作子类,而不需要从新定义新的函数,大大降低了开发成本。但是其实像python这种动态类型的语言,设定的参数并不需要提前设定类型,参数引用的对象是具体运行到的时候才定义的,因此在实际操作时可以灵活很多。只要满足函数中所需的调用的成分,就可以充当参数传入并在函数内部进行调用。这就是动态语言的“鸭子类型”,它并不要求严格的继承体系,一个对象只要“看起来像鸭子,走起路来像鸭子”,那它就可以被看做是鸭子。

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

推荐阅读更多精彩内容