python - 自定义 class

本文介绍了Python中的类定义、方法调用、私有成员的实现(尽管名义上不可见但可通过特殊命名绕过)、继承机制,包括指定单一父类、判断类关系、多重继承以及对象类型检查。
摘要由CSDN通过智能技术生成

1 创建自定义类

下面定义一个 Actor 类:

class Actor:
    def set_name(self, name):
        self.name = name
    def get_name(self):
        return self.name
    def greet(self):
        print(f'大家好,我是演员{self.name}')

尝试使用 Actor 类:

>>> stephen_chow = Actor()
>>> stephen_chow.set_name('周星驰')
>>> stephen_chow.greet()
大家好,我是演员周星驰

其中,self 代表了对象本身,类似于 Java、c# 里的 this

不是所有方法都需要定义 self 参数的,例如:

class Hello:
    def say_hello():
        print('Hello Python!')

say_hello 方法没有定义 self 参数,这是,如果我们使用对象调用 say_hello 方法,则会抛出 TypeError: say_hello() takes 0 positional arguments but 1 was given 异常:

>>> hello = Hello()
>>> hello.say_hello()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: say_hello() takes 0 positional arguments but 1 was given

因为,使用对象调用方法时,对象本身默认会传入方法,即 self。但,如果我们直接使用 class 调用方法,则能调用成功:

>>> Hello.say_hello()
Hello Python!

这跟 Java 的 static 方法类似。

2 隐藏类成员

Python 没有为类的私有成员提供直接支持,若我们要让方法和属性设置为私有,则需要让方法和属性名称以两个下划线 __ 开头:

class PrivateMemberDemo:
    # public 的属性
    name = 'public name property'
    # private 的属性
    __name = 'private name property'
    
    def accessible(self):
        print(f'The value of name is "{self.name}"')
    def __inaccessible(self):
        print(f'The value of __name is "{self.__name}"')

上面的类定义了两个私有成员,属性__name 和 方法__inaccessible,当尝试访问它们的时候,会报 AttributeError 异常

>>> demo = PrivateMemberDemo()
>>> demo.name
'public name property'
>>> demo.__name
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'PrivateMemberDemo' object has no attribute '__name'
>>> demo.accessible()
The value of name is "public name property"
>>> demo.__inaccessible()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'PrivateMemberDemo' object has no attribute '__inaccessible'

但其实这只是一个掩眼法,Python 编译器为 __ 开头的成员变量进行了重新命名,命名规则是 _{类名}{方法名}

使用 dir(obj) 查看对象里有哪些属性和方法:

>>> dir(demo)
['_PrivateMemberDemo__inaccessible', '_PrivateMemberDemo__name', '__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'accessible', 'name']

我们会发现 __name__inaccessible 经编译器处理后,变成了 _PrivateMemberDemo__name_PrivateMemberDemo__inaccessible,因此,我们直接访问 _PrivateMemberDemo__name_PrivateMemberDemo__inaccessible 还是可以的:

>>> demo._PrivateMemberDemo__name
'private name property'
>>> demo._PrivateMemberDemo__inaccessible()
The value of __name is "private name property"

也就是说,我们无法阻止老司机访问对象的私有方法和属性。

3 继承

3.1 指定父类

Python 也是支持继承的,语法为 class ChildClass(ParentClass)

class School:
    def init(self):
        self.courses = []
    def output_courses(self):
        print('进行一下课程:')
        for course in self.courses:
            print(course)

class PrimarySchool(School):
    # 重写父类的 init 方法
    def init(self):
        self.courses = ['语文', '数学', '英语']

class MiddleSchool(School):
    # 重写父类的 init 方法
    def init(self):
        self.courses = ['语文', '数学', '英语', '地理', '物理', '化学', '历史', '生物']

上面的例子中,定义父类 School,子类 PrimarySchoolMiddleSchool,其中子类中分别重写了父类的 init 方法:

>>> primary_school = PrimarySchool()
>>> primary_school.init()
>>> primary_school.output_courses()
进行一下课程:
语文
数学
英语
>>>
>>>
>>>
>>>
>>>
>>> middle_school = MiddleSchool()
>>> middle_school.init()
>>> middle_school.output_courses()
进行一下课程:
语文
数学
英语
地理
物理
化学
历史
生物

3.2 判断类的关系

要确定一个类是否另一个类的子类,可使用 Python 的内置方法 issubclass

>>> issubclass(MiddleSchool, School)
True
>>> issubclass(PrimarySchool, School)
True
>>> issubclass(MiddleSchool, PrimarySchool)
False

若我们想要知道一个类的父类是什么,可以使用类的属性 __bases__

>>> MiddleSchool.__bases__
(<class '__main__.School'>,)
>>> PrimarySchool.__bases__
(<class '__main__.School'>,)

3.3 判断对象与类的关系

要确定一个对象是否为指定类的实例,可使 Python 的内置方法 isinstance:

>>> isinstance(primary_school, PrimarySchool)
True
>>> isinstance(primary_school, School)
True
>>> isinstance(primary_school, MiddleSchool)
False
>>> isinstance(middle_school, MiddleSchool)
True
>>> isinstance(middle_school, School)
True
>>> isinstance(middle_school, PrimarySchool)
False

要知道一个对象属于哪一个类,可使用对象的属性 __class__

>>> middle_school.__class__
<class '__main__.MiddleSchool'>

3.4 多父类

我们可以为一个类指定多个父类,下面定义代表智能手机的类 SmarkPhone,集成了通信、听音乐 和 玩游戏 的功能 SmarkPhone

class Phone:
    def call(self):
        print('正在呼出电话...')

class Mp3:
    def listen_music(self):
        print('播放张学友的李香兰')

class PlayStation:
    def play(self):
        print('正在玩生化危机4重制版')

class SmarkPhone(Phone, Mp3, PlayStation):
    pass

通过 SmarkPhone 对象,可以调用 callplaylisten_music 方法:

>>> phone = SmarkPhone()
>>> phone.call()
正在呼出电话...
>>> phone.listen_music()
播放张学友的李香兰
>>> phone.play()
正在玩生化危机4重制版

使用多重继承时,需要注意,如果多个父类定义同名方法,那么,在父类列表中,位于前面的父类的方法会覆盖位于后面的父类的方法:

class ResidentEvil:
    def play(self):
        print('正在玩生化危机')

class ResidentEvil2nd:
    def play(self):
        print('正在玩生化危机2')

class ResidentEvilCollection(ResidentEvil2nd, ResidentEvil):
    pass

定义对象,并调用 play 方法,将会调用了 ResidentEvil2ndplay 方法:

>>> games = ResidentEvilCollection()
>>> games.play()
正在玩生化危机2
  • 8
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值