Python學習:17.Python面向對象(四、屬性(特性),成員修飾符,類的特殊成員)

来源:https://www.cnblogs.com/liudi2017/archive/2018/07/28/9383838.html
-Advertisement-
Play Games

一、屬性(特性) 普通方法去執行的時候,後面需要加括弧,特性方法執行的時候和靜態欄位一樣不需要不需要加括弧. 特性方法不和欄位同名. 特性方法不能傳參數. 在我們定義資料庫欄位類的時候,往往需要對其中的類屬性做一些限制,一般用get和set方法來寫,那在python中,我們該怎麼做能夠少寫代碼,又能 ...


一、屬性(特性)

普通方法去執行的時候,後面需要加括弧,特性方法執行的時候和靜態欄位一樣不需要不需要加括弧.

特性方法不和欄位同名.

特性方法不能傳參數.

在我們定義資料庫欄位類的時候,往往需要對其中的類屬性做一些限制,一般用get和set方法來寫,那在python中,我們該怎麼做能夠少寫代碼,又能優雅的實現想要的限制,減少錯誤的發生呢,這時候就需要我們的@property.

獲取特性

class Foo:

    def __init__(self,name):
        self.name = name

    # 普通方法
    def start(self):
        temp = '%s sb' %self.name
        return temp

    # 特性,將方法的執行方式變為和欄位一樣
    @property
    def end(self):
        temp = '%s gd'%self.name
        return temp

obj = Foo('alexsel')
ret1 = obj.start()
ret2 = obj.end
print(ret2)
print(obj.name)


輸出結果:
alexsel gd
alexsel

設置特性

設置特性方法的時候,所需要加的裝飾器名字規則是,你所設置特性方法名字點setter(例如:@end.setter)

class Foo:

    def __init__(self,name):
        self.name = name

    # 普通方法
    def start(self):
        temp = '%s sb' %self.name
        return temp

    # 特性,將方法的執行方式變為和欄位一樣
    @property
    def end(self):
        temp = '%s gd'%self.name
        return temp

    # 如果需要使用設置特性的的方法,就需要這個點前面名字和所要設置特性的方法同名,就像這裡的end
    @end.setter
    def end(self,value):
        print(value)
        self.name = value

obj = Foo('alexsel')
#獲取特性,獲取特性的時候,拿到的是@Property的返回值
ret2 = obj.end
print(ret2)
#設置特性,設置特性的時候,會執行@end.setter下麵的方法,這個'aaa'就傳遞給value
obj.end = 'aaa'
ret1 = obj.end
print(ret1)


輸出結果:
alexsel gd
aaa
aaa gd

這個特性在python中不是特別常用,因為我們一般可以使用普通的方法代替特性方法。

 

二、成員修飾符

 首先介紹‘__’,這個在命名之前添加就會變成私有的,只有在類的內部才能訪問。

class Foo:

    book = 'alexsel'
    #命名的時候前面添加__,只有在類的內部才能訪問,在外部無法訪問
    __book = 'book'

    def __init__(self):
        self.__name = 'alexsel'


    def start(self):
        print(Foo.__book)
        print(self.__name)


    def __end(self):
        print('__end')


obj = Foo()
print(obj.book)
# print(obj.__book) #這種無法拿到欄位
#在外部也無法調用類私有方法
#obj.__end()
#通過內部方法調用拿到欄位
obj.start()


輸出結果:
alexsel
book
alexsel

私有的屬性只有自己可以訪問,當某個類繼承這個類之後,也無法訪問父類的私有屬性。

class Foo:

    book = 'alexsel'
    __book = 'book'

    def __init__(self):
        self.__name = 'alexsel'


    def start(self):
        print(Foo.__book)
        print(self.__name)
        self.__end()


    def __end(self):
        print('__end')


class Bar(Foo):
    def start(self):
        print(self.__name)#子類繼承父類,也無法調用父類的私有屬性


obj = Bar()
obj.start()

輸出結果:
報錯

靜態方法也是如此

class Foo:

    book = 'alexsel'
    __book = 'book'

    def __init__(self):
        self.__name = 'alexsel'


    def start(self):
        Foo.__add() #內部調用私有靜態方法


    def __end(self):
        print('__end')


    @staticmethod
    def __add():
        print('add')


obj = Foo()
obj.start()


輸出結果:
add

python成員修飾符有兩種,一種是共有的,一種是私有的,共有的都可以訪問,私有的只有自己可以訪問,或者在外部間接訪問。

但是還有一種強行訪問私有屬性的方法,如下:

class Foo:

    book = 'alexsel'
    __book = 'book'

    def __init__(self):
        self.__name = 'alexsel'


    def start(self):
        Foo.__add() #內部調用私有靜態方法


    def __end(self):
        print('__end')


    @staticmethod
    def __add():
        print('add')


obj = Foo()
print(obj._Foo__book)


輸出結果:
book

雖然可以使用,但是不推薦這種使用方法。

三、類的特殊成員

1.call

__call__()的作用是使實例能夠像函數一樣被調用,同時不影響實例本身的生命周期(__call__()不影響一個實例的構造和析構)。

 __call__ 方法的執行是由對象後加括弧觸發的,即:對象() 或者 類()()。

class Foo:

    def __init__(self):
        print( Foo:      def 'init')

    def __call__(self, *args, **kwargs):
        print('call')
        return 1
r = Foo()
r()
r = Foo()()#加第一個括弧執行__init__,執行完__init__,獲取到一個對象,對象加一個括弧就是執行__call__,拿到返回值
print(r)


輸出結果:
init
call
init
call
1

類後面添加括弧執行__init__方法,對象後面加括弧執行__call__方法。

2.getitem,setitem,delitem

用於索引操作,如字典。以上分別表示獲取、設置、刪除數據。

首先得實例是字典類型的操作。

class Foo:

    def __init__(self):
        print('init')

    def __call__(self, *args, **kwargs):
        print('call')
        return 1

    def __getitem__(self, item):
        print(item)

    def __setitem__(self, key, value):
        print(key,value)

    def __delitem__(self, key):
        print(key)

r = Foo()
r() #調用__call__方法
r['k1']#使用中括弧傳參數的時候,預設使用過的是__getitem__方法
r['xxx'] = 123#這裡的xxx傳給__setiem__的key,123傳給__setitem__的value。
del r['xxx']    #刪除的時候,調用的是__delitem__方法


輸出結果:
init
call
k1

接下來是切片類型的操作,在切片操作的時候,在2.x版本中,執行的是__getslice_,__setslice__,delslice__方法,在3.x中的版本執行的是還是__getitem__,__setitem__,__delitem__方法。

class Foo:

    def __init__(self):
        print('init')

    def __call__(self, *args, **kwargs):
        print('call')
        return 1

    def __getitem__(self, item):
        print(item,type(item),'__getitem__')

    def __setitem__(self, key, value):
        print(key,value)

    def __delitem__(self, key):
        print(key)

r = Foo()
#使用切片的時候,在2.x版本中,調用的是__getslice__方法,在3.x中調用的是__getitem__方法。
r[1:3]
r[1:3] = [11,22,33] #這裡執行__setitem__方法
del r[1:3]  #在這裡執行__delitem__方法

 3.dict

dict是用來存儲對象屬性的一個字典,其鍵為屬性名,值為屬性的值。

class Foo:
    """
    我是一個註釋
    """

    book = 'alexsel'
    __book = 'book'

    def __init__(self):
        self.__name = 'alexsel'


    def start(self):
        Foo.__add() #內部調用私有靜態方法


    def __end(self):
        print('__end')


    @staticmethod
    def __add():
        print('add')


obj = Foo()
print(obj.__dict__) #獲取對象裡面所有欄位
print(Foo.__dict__)

輸出結果:
{'_Foo__name': 'alexsel'}
{'book': 'alexsel', '_Foo__book': 'book', '__init__': <function Foo.__init__ at 0x00000000027CF950>, '__dict__': <attribute '__dict__' of 'Foo' objects>, 'start': <function Foo.start at 0x00000000027EBB70>, '_Foo__add': <staticmethod object at 0x00000000027B6390>, '__weakref__': <attribute '__weakref__' of 'Foo' objects>, '_Foo__end': <function Foo.__end at 0x00000000027EBBF8>, '__doc__': '\n    我是一個註釋\n    ', '__module__': '__main__'}

4.__iter__

類的迭代器可以使用for迴圈迭代類。

如果創建的對象可以被迭代,在類的內部就執行了__iter__方法。

class Foo:
    def __iter__(self):
        yield 1
        yield 2
        yield 3
        yield 4


obj = Foo()
#如果執行for對象時,自動會執行對象的iter方法
for i in obj:
    print(i)



輸出結果:
1
2
3
4

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在使用 RabbitMQ 的時候,作為消息發送方希望杜絕任何消息丟失或者投遞失敗場景。RabbitMQ 為我們提供了兩個選項用來控制消息的投遞可靠性模式。 rabbitmq 整個消息投遞的路徑為: producer->rabbitmq broker cluster->exchange->que... ...
  • 對於我們開發者來說,設計與實現REST API似乎已經成為了我們的日常生活。API現在已經成為了系統間互通的預設方式。AMAZON就是一個有效的使用API進行系統間溝通的最好的例子。在這篇文章中,我將重點討論如何幫助你設計更好的API並避免一些常見的誤區。 ...
  • 首先點擊File-àNew-àWeb [roject-à在Projcet Name里寫項目名-à點擊finish-à會出來一個框,選擇NO,一個javaweb項目就創建好了。具體請看下圖! 配置伺服器連接: 找到Server-à下麵的Tomcat 7.x點擊右鍵-à點擊Configure Serve ...
  • 想著寫一篇hibernate的博文,於是準備從頭開始,從官網下了最新的穩定版本來做講述。 結果利用hibernate自動建表的時候發生下麵這個問題。 我很納悶,之前用低版本一點的沒有發生這個問題啊。 於是,我把必要文件都拷到之前那個hibernate版本是5.0.7的工程中,結果並沒有發生問題。 所... ...
  • [TOC] 翻譯自《Demo Week: Tidy Forecasting with sweep》 原文鏈接:www.business science.io/code tools/2017/10/25/demo_week_sweep.html 時間序列分析工具箱——sweep 的用途 正如 包之於 ...
  • Python名片管理系統 找了很久的 bug,最後發現是 input 沒轉換類型…… ...
  • <! done 本節內容 Python介紹 發展史 Python 2 or 3? 安裝 Hello World程式 變數 用戶輸入 模塊初識 .pyc是個什麼鬼? 數據類型初識 數據運算 表達式if ...else語句 表達式for 迴圈 break and continue  表達式wh ...
  • GameplayKit是一個面向對象的框架,為構建游戲提供基礎工具和技術。 GameplayKit包含用於設計具有功能性,可重用架構的游戲的工具,以及用於構建和增強諸如角色移動和對手行為的游戲玩法特征的技術。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...