Python面向对象二 封装与修饰器

1. 特殊方法

• 在类中可以定义一些特殊方法也称为魔术方法
• 特殊方法都是形如 xxx()这种形式
• 特殊方法不需要我们调用,特殊方法会在特定时候自动调用,在实例对象创建以后执行

# 类的基本结构
'''
class 类名([父类]):

    公共的属性...
    
    # 对象的初始化方法
    def __init__(self,.....):
        
        代码块
    
    # 其他的方法
    def m1(self,....):
       
       代码块
       
    def m2(self,....):
       
       代码块
    .......


'''


# 我们希望在创建对象时,必须设置name属性,如果不设置对象将无法创建

# 学习特殊方法
# 1.特殊方法什么时候调用
# 2.特殊方法有什么作用
class Person:

    def __init__(self,name):
        # 通过self向新创建的对象中初始化属性
        self.name = name
        
    def speak(self):

        print('大家好,我是%s'%self.name)


p1 = Person('葫芦娃') 
p2 = Person('钢铁侠')
p3 = Person('绿巨人')
p4 = Person('美国队长')
print(p1.name) # 葫芦娃
print(p2.name) # 钢铁侠
print(p3.name) # 绿巨人
print(p4.name) # 美国队长
p1.speak() # 大家好,我是葫芦娃
p1.__init__() # TypeError: __init__() missing 1 required positional argument: 'name'
# 手动听添加name属性
p1.name = '钢铁侠'
print(p1.name) # 钢铁侠

p2.name = '绿巨人'
p2.speak() # 大家好,我是绿巨人

# 创建流程
# p1 = Person()
# 1.创建一个变量
# 2.在内存中创建一个新的对象
# 3.执行类中的代码块(只执行一次)
# 4.__init__(self)方法执行了

2. 封装

2.1 封装的引入

首先来看一个案例

# 练习 尝试定义一个车类
'''
属性:name color
方法:run() laba()

'''


class Car():

    def __init__(self,name,color):

        self.name = name
        self.color = color

    def run(self):

        print('汽车开始跑了')

    def laba(self):

        print('%s 滴滴滴滴'%self.name)


c = Car('大奔','白色')
print(c.name,c.color) # 大奔 白色
c.run() # 汽车开始跑了
c.name = '法拉利'
c.color = '黑不溜秋'
c.laba() # 法拉利 滴滴滴滴
print(c.name,c.color) # 法拉利 黑不溜秋

目前我们可以通过 对象.属性 的方式来修改属性值。这种方式导致对象中的属性值可以随意修改,现在我们需要一种方式来增强数据的安全性
出现封装的原因:我们需要一种方式来增强数据的安全性
1.属性不能随意改(我让你改你才能改,不让你改你就别改)
2.属性不能改为任意值

2.2 封装实例一

封装是面向对象的三大特性之一
• 封装是指隐藏对象中一些不希望被外部所访问到的属性或方法
• 我们也可以提供给一个getter()和setter()方法是外部可以访问到属性
getter() 获取对象中指定的属性
setter() 用来设置对象指定的属性

# 封装是面向对象的三大特性之一
# 封装指的是隐藏对象中一些不希望外部访问的属性或方法
# 如何隐藏对象的属性
# 将对象的属性名修改成一个外部不知道的名字

# 如何修改(获取)对象中的属性
# 需要提供一个getter()和setter()方法是外部可以访问属性

# 3.使用setter()方法设置属性,可以增加数据验证,确保数据的正确性
class Dog:

    def __init__(self,name,age):

        self.hidden_name = name
        self.hidden_age = age
    def speak(self):

        print('大家好,我是%s'%self.hidden_name)

    def get_age(self):

        # get_name()用来获取对象的name属性
        print('用户获取了属性')
        return self.hidden_age

    def set_age(self,age):
        # set_name()用来设置对象的name属性
        print('用户修改了属性')
        if age > 0:
            self.hidden_age = age

d = Dog('德牧',5)
d.speak() # 大家好,我是德牧
d.set_name('二哈')
print(d.get_name()) # 二哈
d.hidden_name = '泰迪'
d.speak() # 大家好,我是泰迪
d.set_age(8)
print(d.get_age()) # 8

使用封装,确实增加了类的定义的复杂程度,但是它也确保了数据的安全
• 1. 隐藏属性名,使调用这无法随意的修改对象中的属性
• 2. 增加了getter()和setter()方法,很好控制属性是否是只读的
• 3. 使用setter()设置属性,可以在呢及数据的验证
• 4. 使用getter()方法获取属性,使用setter()方法设置属性可以在读取属性和修改属性的同时做一些其他的处理

2.3 封装实例二

  1. 可以为对象的属性使用双下划线开头 __xxx。:
    双下划线开头的属性,是对象的隐藏属性,隐藏属性只能在类的内部访问,无法通过对象访问, 其实隐藏属性只不过是Python自动为属性改了一个名字 --> _类名__属性名 例如 __name -> _Person__name

class Person:

    def __init__(self,name):

        self.__name = name

    def get_name(self):

        return self.__name

    def set_name(self,name):

        self.__name = name

p = Person('葫芦娃')

print(p.__name) # 'Person' object has no attribute '__name'
print(p._Person__name) # 葫芦娃

p.__name = '钢铁侠' # p.__name 方式不可更改属性name
print(p.get_name()) # 葫芦娃 

p.set_name('钢铁侠')
print(p.get_name()) # 钢铁侠

p._Person__name = '绿巨人' # p._Person__name 方式可更改属性
print(p.get_name()) # 绿巨人

  1. 可以为对象的属性使用单下划线开头 _xxx。
    上面的方式实际上依然可以在外部访问,所以这种方式我们一般不用。一般我们会将一些私有属性以_开头
    一般情况下,使用_开头的属性都是私有属性,没有特殊情况下不要修改私有属性
# 一般情况下,使用_开头的属性都是私有属性,没有特殊情况不要修改私有属性
class Person:

    def __init__(self,name):

        self._name = name

    def get_name(self):

        return self._name

    def set_name(self,name):

        self._name = name

p = Person('葫芦娃')

p.set_name('钢铁侠')
print(p.get_name())
print(p._name)

3. property装饰器

• 我们可以使用@property装饰器来创建只读属性,@property装饰器会将方法转换为相同名称的只读属性,可以与所定义的属性配合使用,这样可以防止属性被修改

class Person:

    def __init__(self,name):

        self._name = name
    # property装饰器 用来将一个方法转换为对象的属性
    @property
    def name(self):
        print('getter方法执行了')
        # 这是一个getter()方法
        return self._name

    # setter方法的装饰器 @属性名.setter
    @name.setter
    def name(self,name):

        self._name = name



p = Person('葫芦娃')
print(p.name)
# getter方法执行了
# 葫芦娃
p.name = '钢铁侠'
print(p.name)
# getter方法执行了
# 钢铁侠
  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值