Python中面向对象的讲解(3)
1.私有屬性和私有方法
封裝的意義:
將屬性和方法放到一起做為一個(gè)整體,然后通過實(shí)例化對(duì)象來處理;
隱藏內(nèi)部實(shí)現(xiàn)細(xì)節(jié),只需要和對(duì)象及其屬性和方法交互就可以了;
對(duì)類的屬性和方法增加 訪問權(quán)限控制。
私有權(quán)限:在屬性名和方法名 前面 加上兩個(gè)下劃線 __
類的私有屬性 和 私有方法,都不能通過對(duì)象直接訪問,但是可以在本類內(nèi)部訪問;
類的私有屬性 和 私有方法,都不會(huì)被子類繼承,子類也無法訪問;
私有屬性 和 私有方法 往往用來處理類的內(nèi)部事情,不通過對(duì)象處理,起到安全作用。
私有屬性:
class Person(object):def __init__(self,name,age):self.name=nameself.__age=agep=Person('fei',22) print(p.name) print(p.__age)
私有方法:
類部調(diào)用私有屬性和私有方法
子類不能繼承父類私有屬性和方法
1). 私有屬性,可以在類內(nèi)部通過self調(diào)用,但不能通過對(duì)象訪問
2). 私有方法,可以在類內(nèi)部通過self調(diào)用,但不能通過對(duì)象訪問
3). 對(duì)象不能訪問私有權(quán)限的屬性和方法
4). 子類不能繼承父類私有權(quán)限的屬性和方法
5). Python中沒有像C++中 public 和 private, protected 這些關(guān)鍵字來區(qū)別公有屬性和私有屬性。
6). Python是以屬性命名方式來區(qū)分,如果在屬性和方法名前面加了2個(gè)下劃線’__’,則表明該屬性和方法是私有權(quán)限,否則為公有權(quán)限。
2.修改私有屬性的值
class Person(object):def __init__(self):self.__age=24def set_age(self,age):self.__age=agedef get_age(self):return self.__agep=Person() p.set_age(28) print(p.get_age())現(xiàn)代軟件開發(fā)中,通常會(huì)定義get_xxx()方法和set_xxx()方法來獲取和修改私有屬性值
get_xxx()方法–>返回私有屬性的值
set_xxx()方法–>接收參數(shù),修改私有屬性的值
對(duì)象不能訪問私有權(quán)限的屬性和方法,可以通過訪問公有方法set_money()來修改私有屬性的值,可以通過訪問公有方法get_money()來獲取私有屬性的值
3.類屬性和實(shí)例屬性
類屬性就是類對(duì)象所擁有的屬性,它被所有類對(duì)象的實(shí)例對(duì)象所共有,在內(nèi)存中只存在一個(gè)副本,這個(gè)和C++中類的靜態(tài)成員變量有點(diǎn)類似。對(duì)于公有的類屬性,可以通過類或者實(shí)例對(duì)象訪問
實(shí)例屬性只能通過對(duì)象來調(diào)用,類不能調(diào)用
類屬性:
class People(object):name = 'Tom' # 公有的類屬性__age = 12 # 私有的類屬性p = People()print(p.name) # 正確 print(People.name) # 正確 print(p.__age) # 錯(cuò)誤,不能在類外通過實(shí)例對(duì)象訪問私有的類屬性 print(People.__age) # 錯(cuò)誤,不能在類外通過類對(duì)象訪問私有的類屬性可以通過類或者實(shí)例對(duì)象調(diào)用
實(shí)例屬性(對(duì)象屬性)
class People(object):address = '山東' # 類屬性def __init__(self):self.name = 'xiaowang' # 實(shí)例屬性self.age = 20 # 實(shí)例屬性 p = People() p.age = 12 # 實(shí)例屬性 print(p.address) # 正確 print(p.name) # 正確 print(p.age) # 正確 print(People.address) # 正確 print(People.name) # 錯(cuò)誤 print(People.age) # 錯(cuò)誤可以通過實(shí)例化對(duì)象調(diào)用,類不能調(diào)用
通過實(shí)例(對(duì)象)去修改類屬性
class People(object):country = 'china' #類屬性 print(People.country) p = People() print(p.country) p.country = 'japan' print(p.country) # 實(shí)例屬性會(huì)屏蔽掉同名的類屬性 print(People.country) del p.country # 刪除實(shí)例屬性 print(p.country)對(duì)象修改類屬性,只對(duì)本對(duì)象有效果,對(duì)別的
對(duì)象沒有影響
4.類方法和靜態(tài)方法
類方法:
類方法是類對(duì)象所擁有的方法,需要用修飾器@classmethod來標(biāo)識(shí)其為類方法,對(duì)于類方法,第一個(gè)參數(shù)必須是類對(duì)象,一般以cls作為第一個(gè)參數(shù)(當(dāng)然可以用其他名稱的變量作為其第一個(gè)參數(shù),但是大部分人都習(xí)慣以’cls’作為第一個(gè)參數(shù)的名字,就最好用’cls’了),能夠通過實(shí)例對(duì)象和類對(duì)象去訪問。
class People(object):# 類屬性age =18#類方法,用classmethod來進(jìn)行修飾@classmethoddef get_country(cls):return cls.age p = People() print(p.get_country()) #可以用過實(shí)例對(duì)象引用 print(People.get_country()) #可以通過類對(duì)象引用 class People(object):# 類屬性age= 18#類方法,用classmethod來進(jìn)行修飾@classmethoddef get_country(cls):return cls.age@classmethoddef set_country(cls,age):cls.age = age p = People() print(p.get_country()) #可以用過實(shí)例對(duì)象訪問 print(People.get_country()) #可以通過類訪問 p.set_country(23) print(p.get_country()) print(People.get_country()) p1 = People() print(p1.get_country())結(jié)果顯示在用類方法對(duì)類屬性修改之后,通過類對(duì)象和實(shí)例對(duì)象訪問都發(fā)生了改變(全部改變)
靜態(tài)方法:
需要通過修飾器@staticmethod來進(jìn)行修飾,靜態(tài)方法不需要多定義參數(shù),可以通過對(duì)象和類來訪問。
class People(object):country = 'china'@staticmethod#靜態(tài)方法def get_country():return People.country p = People() # 通過對(duì)象訪問靜態(tài)方法 print(p.get_country()) # 通過類訪問靜態(tài)方法 print(People.get_country())靜態(tài)方法中不需要額外定義參數(shù),因此在靜態(tài)方法中引用類屬性的話,必須通過類實(shí)例對(duì)象來引用,調(diào)用靜態(tài)方法可以通過對(duì)象或者類調(diào)用
實(shí)例方法
實(shí)例方法中的第一個(gè)參數(shù)是self,只能通過對(duì)象來訪問。
class People(object):def selfmethod(self):print("我是實(shí)例方法") p = People() p.selfmethod() #People.selfmethod() #報(bào)錯(cuò)實(shí)例方法中需要self參數(shù),因此調(diào)用實(shí)例方法只能通過實(shí)例對(duì)象調(diào)用 也可以通過類調(diào)用但是一般不這樣用
總結(jié):
類方法使用@classmethod裝飾,第一個(gè)參數(shù)為類(cls),調(diào)用時(shí)可以通過類的實(shí)例或者類本身來調(diào)用。
實(shí)例方法定義時(shí)第一個(gè)參數(shù)為類的一個(gè)實(shí)例(self),調(diào)用時(shí)必須通過實(shí)例調(diào)用。
靜態(tài)方法使用@staticmethod裝飾,調(diào)用時(shí)可以使用類的實(shí)例或者類本身來調(diào)用。
5.__new__方法
__new__和__init__的作用
class A(object):def __init__(self):print('init方法')def __new__(cls, *args, **kwargs):print('new方法')return object.__new__(cls) a=A()
總結(jié):
1). __new__至少要有一個(gè)參數(shù)cls,代表要實(shí)例化的類,此參數(shù)在實(shí)例化時(shí)由Python解釋器自動(dòng)提供
2). __new__必須要有返回值,返回實(shí)例化出來的實(shí)例,這點(diǎn)在自己實(shí)現(xiàn)__new__時(shí)要特別注意,可以return父類__new__出來的實(shí)例,或者直接是object的__new__出來的實(shí)例
3). __init__有一個(gè)參數(shù)self,就是這個(gè)__new__返回的實(shí)例,__init__在__new__的基礎(chǔ)上可以完成一些其它初始化的動(dòng)作,__init__不需要返回值
4). 我們可以將類比作制造商,__new__方法就是前期的原材料購買環(huán)節(jié),__init__方法就是在有原材料的基礎(chǔ)上,加工,初始化商品環(huán)節(jié)
6.單例模式
單例模式:永遠(yuǎn)用一個(gè)對(duì)象得實(shí)例,避免新建太多實(shí)例浪費(fèi)資源
實(shí)質(zhì):使用__new__方法新建類對(duì)象時(shí)先判斷是否已經(jīng)建立過,如果建過就使用已有的對(duì)象
class Foo(object):instance = Nonedef __init__(self):self.name = 'alex'def __new__(cls):if Foo.instance:return Foo.instanceelse:Foo.instance = object.__new__(cls)return Foo.instance obj1 = Foo() obj2 = Foo() print(obj1,obj2)
福利
總結(jié)
以上是生活随笔為你收集整理的Python中面向对象的讲解(3)的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Python中面向对象的讲解(2)
- 下一篇: Python单元测试之unittest