天天看點

python 封裝

封裝

# 封裝是面向對象的三大特性之一
# 封裝指的是隐藏對象中一些不希望被外部所通路到的屬性或方法
# 如何隐藏一個對象中的屬性?
#   - 将對象的屬性名,修改為一個外部不知道的名字
# 如何擷取(修改)對象中的屬性?
#   - 需要提供一個getter和setter方法使外部可以通路到屬性
#   - getter 擷取對象中的指定屬性(get_屬性名)
#   - setter 用來設定對象的指定屬性(set_屬性名)
# 使用封裝,确實增加了類的定義的複雜程度,但是它也確定了資料的安全性
#   1.隐藏了屬性名,使調用者無法随意的修改對象中的屬性
#   2.增加了getter和setter方法,很好的控制的屬性是否是隻讀的
#       如果希望屬性是隻讀的,則可以直接去掉setter方法
#       如果希望屬性不能被外部通路,則可以直接去掉getter方法
#   3.使用setter方法設定屬性,可以增加資料的驗證,確定資料的值是正确的
#   4.使用getter方法擷取屬性,使用setter方法設定屬性
#       可以在讀取屬性和修改屬性的同時做一些其他的處理
#   5.使用getter方法可以表示一些計算的屬性

class Dog:
    '''
        表示狗的類
    '''
    def __init__(self , name , age):
        self.hidden_name = name
        self.hidden_age = age

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

    def get_name(self):
        '''
            get_name()用來擷取對象的name屬性
        '''    
        # print('使用者讀取了屬性')
        return self.hidden_name

    def set_name(self , name):
        # print('使用者修改了屬性')
        self.hidden_name = name

    def get_age(self):
        return self.hidden_age

    def set_age(self , age):
        if age > 0 :
            self.hidden_age = age    


d = Dog('旺财',8)

# d.say_hello()

# 調用setter來修改name屬性 
d.set_name('小黑')
d.set_age(-10)

# d.say_hello()
print(d.get_age())
           
class Rectangle:
    '''
        表示矩形的類
    '''
    def __init__(self,width,height):
        self.hidden_width = width
        self.hidden_height = height

    def get_width(self):
        return self.hidden_width

    def get_height(self):
        return self.hidden_height   

    def set_width(self , width):
        self.hidden_width = width 

    def set_height(self , height):
        self.hidden_height = height 

    def get_area(self):
        return self.hidden_width * self.hidden_height        

# r = Rectangle(5,2)  
# r.set_width(10)
# r.set_height(20)

# print(r.get_area())     
 
 
# 可以為對象的屬性使用雙下劃線開頭,__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) __開頭的屬性是隐藏屬性,無法通過對象通路
# p.__name = '豬八戒'
# print(p._Person__name)
# p._Person__name = '豬八戒'

# print(p.get_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)
           

裝飾器

class Person:
    def __init__(self,name,age):
        self._name = name
        self._age = age

    # property裝飾器,用來将一個get方法,轉換為對象的屬性
    # 添加為property裝飾器以後,我們就可以像調用屬性一樣使用get方法
    # 使用property裝飾的方法,必須和屬性名是一樣的
    @property    
    def name(self):
        print('get方法執行了~~~')
        return self._name

    # setter方法的裝飾器:@屬性名.setter
    @name.setter    
    def name(self , name):
        print('setter方法調用了')
        self._name = name        

    @property
    def age(self):
        return self._age

    @age.setter    
    def age(self , age):
        self._age = age   

        

p = Person('豬八戒',18)

p.name = '孫悟空'
p.age = 28

print(p.name,p.age)