面向对象之封装 及@property装饰器使用

目录

  • 封装

    • 1.封装的定义
    • 2.封装的目的:
    • 3.封装的三种方式
    • 4.封装的优点
    • 5.访问限制(封装)
  • @property 装饰器

封装

1.封装的定义

? 将复杂的丑陋的,隐私的细节隐藏到内部,对外提供简单的使用接口, 对外隐藏内部实现细节,并提供访问的接口;

2.封装的目的:

     封装的目的: 面向对象的核心是对象二字,精髓在于整合,封装的目的其实就是把一堆数据属性和方法属性整合到对象中,我们可以把对象比喻成一个容器,其实就是为了把数据存入一个容器中。存的目的就是为了取的,那封装到对象中的好处就是可以通过 "对象.属性" 的方式把属性取出来。 

3.封装的三种方式

? 1.publish: 对所有人公开权限,在外部哪里都可以访问。

? 2.protected: 这种只有在当前类,或者是继承它的 “子类” (派生类) 可以访问。

? 3.private: 这种封装相当于属性的私有化,让任何人都不可以访问。但是python没有这种强 制限制的机制,只能通过访问限制机制去实现表面上的属性私有化

4.封装的优点

? 1.良好的封装能够减少耦合。

? 2.类内部的结构可以自由修改。

? 3.可以对成员变量进行更精确的控制。

? 4.隐藏信息,实现细节。

# 不封装
def login(name, pwd, root):
    if name == 'tank' and pwd == '123' and root == 'vip':
        print('登录成功!')
    else:
        print('登录失败!')

login('tank', '123', 'vip')

# 封装
class User:
    def __init__(self, name, pwd, root):
        self.__name = name
        self.__pwd = pwd
        self.__root = root

    def change_pwd(self, new_pwd):
        self.__pwd = new_pwd

    @property
    def get_name(self):
        return self.__name

    @property
    def get_pwd(self):
        return self.__pwd

    @property
    def get_root(self):
        return self.__root

user_obj = User('tank', '123', 'vip')

def login(obj):
    if obj.get_name == 'nim' and obj.get_pwd == '123' and obj.get_root == 'vip':
        print('登录成功!')

    else:
        print('登录失败')

user_obj.change_pwd('321')
login(user_obj)

5.访问限制(封装)

? 隐藏对象的属性和实现细节,仅对外提供公共访问方式。
? 目的:
? 1.为了保证 关键数据的安全性
? 2.对外部隐藏实现细节,隔离复杂度

1)私有变量

#其实这仅仅这是一种变形操作
#类中所有双下划线开头的名称如__x都会自动变形成:_类名__x的形式:

class A:
    __N=0 #类的数据属性就应该是共享的,但是语法上是可以把类的数据属性设置成私有的如__N,会变形为_A__N
    def __init__(self):
        self.__X=10 #变形为self._A__X
    def __foo(self): #变形为_A__foo
        print('from A')
    def bar(self):
        self.__foo() #只有在类内部才可以通过__foo的形式访问到.

#A._A__N是可以访问到的,即这种操作并不是严格意义上的限制外部访问,仅仅只是一种语法意义上的变形

这种变形的特点:

? 1.类中定义的__x只能在内部使用,如 self.__x,引用的就是形变的结果;

? 2.这种变形其实正是针对外部的变形,在外部是无法通过__x这个名字访问到的;

? 3.在子类定义的__x不会覆盖在父类定义的 __x,因为子类中变形成了:_子类名__x,而父类中变形成了:_父类名__x,即双下划线开头的属性在继承给子类时,子类时无法覆盖的。

这种变形需要注意的问题是:

***1.这种机制也并没有真正意义上限制我们从外部直接访问属性,知道了类名和属性名就可以拼出名字:_类名__属性,然后就可以访问了,如a._A__N***

2.变形的过程只在类的内部生效,在定义后的赋值操作,不会变形

>>> a=A()
>>> a.__dict__
{'_A__X':10}
>>> a.__Y = 1
>>> a.__dict__
{'_A__X':10,'__Y':1}

2)私有方法

? 在继承中,父类如果不想让子类覆盖自己的方法,可以将方法定义成私有的。

# 正常情况
class A:
   def fa(self):
       print('from A')
   def test(self):
       print(self)  # <__main__.B object at 0x00000000023A6FD0>
       self.fa()

class B(A):
   def fa(self):
       print('from B')

b=B()
b.test()

# 把fa定义成私有的,即__fa
class A:
     def __fa(self): #在定义时就变形为_A__fa
         print('from A')
     def test(self):
         print(self) # <__main__.B object at 0x00000000023A6FD0>
         self.__fa() #只会与自己所在的类为准,即调用_A__fa

class B(A):
     def __fa(self):
         print('from B')

b=B()
b.test()

示例1:

#222  我们通过对__id_number进行修改,来对比外部是否能对__开头的属性进行修改.# p.show_id()  #12312  内部值并没有发生变化,所以通过__开头进行封装,外部通过普通方法并不能对内部的值进行修改.

示例2:

  # 通过__方法名()的方式,将内部内部具体实现细节封装起来,只给外部暴露一个简单的开机(open函数)接口
  class PC:
     def __init__(self,price,color,kind):
         self.color=color
         self.price=price
         self.kind=kind
     def open(self):
         print('接通电源')
         self.__check_device()
         print('载入内核')
         print('初始化内核')
         self.__start_service()
         print('启动GUI')
         self.__login()

     def __check_device(self):
         print('检测硬件1')
         print('检测硬件2')
         print('检测硬件3')
         print('检测硬件4')
     def __start_service(self):
         print('启动服务1')
         print('启动服务2')
         print('启动服务3')
     def __login(self):
         print('login..')

 pc=PC(6000,'white','lenvo')
 pc.open()
 '''
 接通电源
 检测硬件1
 检测硬件2
 检测硬件3
 检测硬件4
 载入内核
 初始化内核
 启动服务1
 启动服务2
 启动服务3
 启动GUI
 login..

 '''

示例3:

# e.g.1
class Downloader:
    def __init__(self,filename,url,buffer_size):
        self.filename=filename
        self.url=url
        self.__buffer_size=buffer_size

    def start_download(self):
        if self.__buffer_size<=1024*1024:
            print('开始下载...')
        else:
            print('内存不足..')

d=Downloader('葫芦娃','http://www.baidu.com',1024*1024)

d.buffer_size=1024*1024*10  #类中__开头的属性无法直接访问,所以无法修改。
d.start_download()
'''
开始下载...
'''

# e.g.2
class Downloader:
    def __init__(self,filename,url,buffer_size):
        self.filename=filename
        self.url=url
        self.__buffer_size=buffer_size

    def start_download(self):
        if self.__buffer_size<=1024*1024:
            print('开始下载...')
            print('当前缓冲区大小为%s'%self.__buffer_size)
        else:
            print('内存不足..')

    def set_buffer_size(self,size):
        self.__buffer_size=size
    def get_buffer_size(self):
        return self.__buffer_size

d=Downloader('葫芦娃','http://www.baidu.com',1024*1024)

#通过函数去修改内部封装的属性
d.set_buffer_size(1024*1024/2)

#通过函数访问内部封装的属性
print(d.get_buffer_size())
d.start_download()

'''
524288.0
开始下载...
当前缓冲区大小为524288.0
'''

示例4

class Foo:
    __name = 'rose'  # 变形成 _Foo__name
    def __init__(self,sex):
        self.__sex=sex

    @property
    def __sing(self):  # 变形成_Foo__sing,外部可以通过变形后的名字从外部直接访问。
        print('我是隐藏属性,不懂其变形情况在外部是无法访问的!')

f = Foo('male')
print(f._Foo__name)
f._Foo__sing

'''
rose
我是隐藏属性,不懂其变形情况在外部是无法访问的!

'''

@property 装饰器

? @property 用来装饰类内部的函数:@property使对象调用某个方法时,将对象.方法()变成对 象.方法得形式,使对象点函数(方法)看起来像是点普通的数据属性一样,使得调用方式一 致。切记:装饰过后的函数名后就不要再加括号啦!!!

? @key.setter 该装饰器用在修改属性的方法上
? @key.deleter 该装饰器用在删除属性的方法上

? 注意:key是被property装饰的方法的名称,也就是属性的名称 ;内部会创建一个对象 变量名称就是函数名称,所以在使用setter和deleter时 必须保证使用对象去调用方法,所以是 key.setter和key.deletter。

示例1

class Foo:
    def __init__(self,val):
        self.__NAME=val #将所有的数据属性都隐藏起来

    @property
    def name(self):
        return self.__NAME #obj.name访问的是self.__NAME(这也是真实值的存放位置)

    @name.setter
    def name(self,value):
        if not isinstance(value,str):  #在设定值之前进行类型检查
            raise TypeError('%s must be str' %value)
        self.__NAME=value #通过类型检查后,将值value存放到真实的位置self.__NAME

    @name.deleter
    def name(self):
        raise TypeError('Can not delete')

f=Foo('tank')
# print(f.name)
# f.name = 10  # 抛出异常'TypeError: 10 must be str'
f.name = 'zhang'
print(f.__dict__)  # {'_Foo__NAME': 'zhang'}
# del f.name #抛出异常'TypeError: Can not delete'

示例2

class People:
    def __init__(self,name,weight,height):
        self.name=name
        self.weight=weight
        self.height=height
    @property
    def bmi(self):
        return self.weight / (self.height**2)

p1=People('egon',75,1.85)
print(p1.bmi)  #直接实例对象 点 方法名称就可以了

示例3

class Goods:

    def __init__(self):
        # 原价
        self.original_price = 100
        # 折扣
        self.discount = 0.8

    @property
    def price(self):
        # 实际价格 = 原价 * 折扣
        new_price = self.original_price * self.discount
        return new_price

    @price.setter
    def price(self, value):
        self.original_price = value

    @price.deleter
    def price(self):
        del self.original_price

obj = Goods()
obj.price         # 获取商品价格
obj.price = 200   # 修改商品原价
print(obj.price)
del obj.price     # 删除商品原价

原文地址:https://www.cnblogs.com/zhangchaocoming/p/11665888.html

时间: 2024-10-08 07:26:56

面向对象之封装 及@property装饰器使用的相关文章

python面向对象:组合、封装、property装饰器、多态

一.组合二.封装三.property装饰器四.多态 一.组合 ''' 1. 什么是组合 一个对象的属性是来自于另外一个类的对象,称之为组合 2. 为何用组合 组合也是用来解决类与类代码冗余的问题 3. 如何用组合 ''' # class Foo: # aaa=1111 # def __init__(self,x,y): # self.x=x # self.y=y # # def func1(self): # print('Foo内的功能') # # # class Bar: # bbb=2222

面向对象——property装饰器

property装饰器 property装饰器的作用,其实就是将将函数属性伪装成为属性的的装饰器 class People: def __init__(self,name,weight,height): self.name = name self.weight =weight self.height=height @property def bmi(self): return self.weight/(self.height**2) msj = People('msj',82,1.84) pri

面向对象编程之property装饰器

一.与函数装饰器的相似点 1.都是可调用对象 2.在不修改装饰对象的源代码和调用方式的前提下,添加了新功能 二.property装饰器的独特用法:不管property装饰器新增的是什么功能,结果都是给装饰对象伪造了了一个数据属性. 三.具体用法:将对象数据属性的查/改/删的操作按照,封装===>定义接口===>伪装接口,的顺序装饰,最后的结果,使用者看来似乎还是直接对对象数据属性的操作方法,其实已被接口内的功能进行了限制. class Human: def __init__(self, nam

一段小代码说明@property装饰器的用法

#coding:utf-8 """ 一段小代码说明@property装饰器的用法.__name是私有变量 外部不能通过foo.__name访问,但可以通过foo._Foo__name访问. 如果想通过点运算符设置和访问实例属性怎么办,@property派上用途了, 先将一个方法头上加上@property,这个方法就变成实例属性了,再加上这个@get_name.setter 装饰器(@属性名.setter)就可以通过赋值来设置属性. python北京周末培训班 https://

python staticmethod,classmethod方法的使用和区别以及property装饰器的作用

class Kls(object): def __init__(self, data): self.data = data def printd(self): print(self.data) @staticmethod def smethod(*arg): print('Static:', arg) @classmethod def cmethod(*arg): print('Class:', arg) >>> ik = Kls(23) >>> ik.printd()

python @property装饰器

@property装饰器 @property装饰器就是负责把一个方法变成属性调用把一个getter方法变成属性,只需要加上@property就可以了,此时,@property本身又创建了另一个装饰器@score.setter,负责把一个setter方法变成属性赋值class Student(object): @property def score(self): return self._score @score.setter def score(self, value): if not isin

3.1.8 property装饰器

在类的方法上加上一行@property 装饰器,会使得用户调用该函数属性时,就像调用数据属性一样,不需要加上() 比如想获取一些名词,再加上括号,容易使调用者忘记.动词才加()调用方法. 如下示例: '''BMI指数(bmi是计算而来的,但很明显它听起来像是一个属性而非方法,如果我们将其做成一个属性,更便于理解) 成人的BMI数值: 过轻:低于18.5 正常:18.5-23.9 过重:24-27 肥胖:28-32 非常肥胖, 高于32 体质指数(BMI)=体重(kg)÷身高^2(m) EX:70

Python中,关于@property装饰器

1.为什么使用@property装饰器?br/>在类中,当我么不想在外界直接调用到类的属性,或者不想展示属性的真实内容时,可以用到@property.它规定了我们直接用 对象名.属性名 获取对象属性时并不会直接取得对象的属性,而是通过调用@property装饰过的属性函数来给调用者反馈. 2.我们为什么不使用特定的方法来进行上面的操作?原因是因为太繁琐.例: class Person: def __init__(self, username, password) -> None: self.u

@property 装饰器

property() 函数作用于新式类,返回属性值. class C(object): def __init__(self): self._x = None def getx(self): print('get') return self._x def setx(self, value): print('set') self._x = value def delx(self): print('del') del self._x x = property(getx, setx, delx, "I'