1. 程式人生 > >Python——詳解__str__, __repr__和__format__

Python——詳解__str__, __repr__和__format__

本文始發於個人公眾號:TechFlow,原創不易,求個關注


今天是Python專題的第10篇文章,我們來聊聊Python當中的類。

列印例項

我們先從類和物件當中最簡單的列印輸出開始講起,列印一個例項是一個非常不起眼的應用,但是在實際的程式設計當中卻非常重要。原因也很簡單,因為我們debug的時候往往會想看下某個類當中的內容是不是符合我們的預期。但是我們直接print輸出的話,只會得到一個地址。

我們來看一個例子:

class point:
    def __init__(self, x, y):
        self.x = x
        self.y = y


if __name__ == "__main__":
    p = point(3, 4)
    print(p)

在這段程式碼當中我們定義了一個簡單的類,它當中有x和y兩個元素,但是如果我們直接執行的話,螢幕上會輸出這樣一個結果:

<__main__.point object at 0x10a18c210>

這個是直譯器在執行的時候這個例項的一些相關資訊,但是對於我們來說幾乎沒有參考意義,我們想要的是這個例項當中具體的值,而不是一個記憶體當中的地址。

想要實現這個功能,我們有很多方法,下面我們一一來看。

__str__方法

__str__方法大家應該都不陌生,它類似於Java當中的toString方法,可以根據我們的需要返回例項轉化成字串之後的結果。

比如,我們可以在類當中過載這個方法,就可以根據我們的需要輸出結果了:

class point:
    def __init__(self, x, y):
        self.x = x
        self.y = y

    def __str__(self):
        return 'x: %s, y: %s' % (self.x, self.y)

當我們執行它,得到的結果會是:

x: 3, y: 4

__str__和__init__, __len__很多函式一樣是Python中的特殊函式,在我們建立類的時候,系統會我們隱式創造許多這樣的特殊函式。我們可以根據需要過載其中的一部分完成我們想要的功能。比如如果我們寫的是一棵二叉樹的類,我們還可以在__str__函式當中進行遞迴遍歷所有的節點,打印出完整的樹來。

__repr__方法

你也許可能也聽說過__repr__函式,它也可以實現根據我們的需要自定義輸出的功能。比如我們把上面的程式碼改下函式名,也可以得到一樣的結果。

class point:
    def __init__(self, x, y):
        self.x = x
        self.y = y

    def __repr__(self):
        return 'x: %s, y: %s' % (self.x, self.y)

我們執行它,同樣會得到:

x: 3, y: 4

這是為什麼呢,難道__repr__和__str__是一樣的嗎?如果是一樣的,Python的設計者幹嘛要保留兩個完全相同的函式呢,為什麼不去掉其中一個呢?

在分析原因之前,我們先來做一個實驗,如果我們兩個函式都過載,那麼當我們輸出的時候,程式執行的是哪一個呢?為了做好區分,我們把repr當中的輸出的格式稍微修改一下。

class point:
    def __init__(self, x, y):
        self.x = x
        self.y = y

    def __str__(self):
        return 'x: %s, y: %s' % (self.x, self.y)

    def __repr__(self):
        return '<point x: %s, y: %s>' % (self.x, self.y)

我們執行之後,會發現輸出的結果還是:

x: 3, y: 4

先彆著急下結論,我們再把這段程式碼拷貝到jupyter notebook當中,我們這次不通過列印輸出,而通過jupyter自帶的互動框輸出互動結果,我們再來看下:

奇怪,怎麼結果就變成了__repr__的結果了呢?

其實這正是反應了兩者的區別,如果簡單理解,這兩個函式都是將一個例項轉成字串。但是不同的是,兩者的使用場景不同,其中__str__更加側重展示。所以當我們print輸出給使用者或者使用str函式進行型別轉化的時候,Python都會預設優先呼叫__str__函式。而__repr__更側重於這個例項的報告,除了例項當中的內容之外,我們往往還會附上它的類相關的資訊,因為這些內容是給開發者看的。所以當我們在互動式視窗輸出的時候,它會優先呼叫__repr__。

理論上來說,對於一個合格的__repr__函式要能夠做到:

eval(repr(obj)) == obj

也就是說我們通過__repr__輸出的內容執行之後可以再還原得到這個例項本身,當然在一些場景下這個非常難以實現,所以我們退而求其次,保證__repr__當中輸出類和物件足夠多的資訊,方便開發者除錯和使用即可。

另外多說一句,repr是report的縮寫,所以它有一個報告的意思在裡面,而str就只是轉化成字串而已。這兩者還是有一定區別的。

format

Python當中最常用的輸出函式除了上面兩個之外,還有一個就是format。

比較簡單的用法就是通過{}代表變數,然後按照順序依次輸入:

除此之外,我們還可以進一步寫明花括號裡的變數名稱,進一步增加可讀性:

format的功能遠不止如此,它還支援許多引數,類似於C語言當中的printf,可以通過不同的引數做到各種各樣的輸出。比如控制小數點後面保留的位數,或者是轉化成百分數、科學記數法、左右對齊等功能。這裡不一一列舉了,大家用到的時候再查詢即可。

我們當然可以使用format重新__repr__和__str__當中的邏輯,但這並不能體現它的強大。因為在Python當中,也為類提供了__format__這個特殊函式,通過重寫__format__和使用format,我們可以做到更牛的功能。

format聯合__format__

我們可以在類當中過載__format__函式,這樣我們就可以在外部直接通過format函式來呼叫物件,輸出我們想要的結果。

我們來看程式碼:

class point:
    def __init__(self, x, y):
        self.x = x
        self.y = y

    def __str__(self):
        return 'x: %s, y: %s' % (self.x, self.y)

    def __format__(self, code):
        return 'x: {x}, y: {y}'.format(x = self.x, y = self.y)

我們把剛才的__repr__改成了__format__,但是需要注意一個細節,我們多加了一個引數code,這是由於format當中支援通過引數來對處理邏輯進行配置的功能,所以我們必須要在介面處多加一個引數。加好了以後,我們就可以直接呼叫format(p)了。

到這裡還沒有結束,在有些場景當中,對於同一個物件我們可能有多種輸出的格式。比如點,在有些場景下我們可能希望輸出(x, y),有時候我們又希望輸出x: 3, y: 4,可能還有些場景當中,我們希望輸出<x, y>。

我們針對這麼多場景,如果各自實現不同的介面會非常麻煩。這個時候利用__format__當中的這個引數,就可以大大簡化這個過程,我們來看程式碼:

formats = {
    'normal': 'x: {p.x}, y: {p.y}',
    'point' : '({p.x}, {p.y})',
    'prot': '<{p.x}, {p.y}>'
}

class point:
    def __init__(self, x, y):
        self.x = x
        self.y = y

    def __str__(self):
        return 'x: %s, y: %s' % (self.x, self.y)

    def __format__(self, code):
        return formats[code].format(p=self)

我們在呼叫的時候就可以通過引數來控制我們究竟使用哪一種格式來格式化物件了:

也就是說通過過載__format__方法,我們把原本固定的格式化的邏輯做成了可配置的。這樣大大增加了我們使用過程當中的靈活性,這種靈活性在一些問題場景當中可以大大簡化和簡潔我們的程式碼。對於Python這門語言來說,我個人感覺實現功能只是其中很小的一個部分,把程式碼寫得簡潔美觀,才是其中的大頭。這也是為什麼很多人都說Python易學難精的原因。

今天的文章就是這些,如果覺得有所收穫,請順手點個關注或者轉發吧,你們的舉手之勞對我來說很重要。

相關推薦

Python——__str__, __repr____format__

本文始發於個人公眾號:TechFlow,原創不易,求個關注 今天是Python專題的第10篇文章,我們來聊聊Python當中的類。 列印例項 我們先從類和物件當中最簡單的列印輸出開始講起,列印一個例項是一個非常不起眼的應用,但是在實際的程式設計當中卻非常重要。原因也很簡單,因為我們debug的時候往往會想看

Python-Flask框架之——圖書管理系統 , 附源碼效果圖 !

數據庫 mysql AS image http 分享圖片 bubuko 書籍 sql 該圖書管理系統要實現的功能: 1. 可以通過添加窗口添加書籍或作者, 如果要添加的作者和書籍已存在於書架上, 則給出相應的提示. 2. 如果要添加的作者存在, 而要添加的書籍書架上沒有,

梳理Python 框架之中介軟體(用途機制)

什麼是中介軟體? 中介軟體是一個Python程式設計師用來處理Django的請求和響應的框架級別的鉤子,它是一個輕量,低級別的外掛系統,用於全域性範圍內改變Django的輸入,輸出。每個中介軟體元件都負責做一些特定的功能。 說的直白一點是中介軟體就是幫我們程式設計

Python 拓展之深拷貝淺拷貝!

首先我在這介紹兩個新的小知識,要在下面用到。一個是函式 id() ,另一個是運算子 is。id() 函式就是返回物件的記憶體地址;is 是比較兩個變數的物件引用是否指向同一個物件,在這裡請不要和 == 混了,== 是比較兩個變數的值是否相等。 >>> a = [1,2,3] &

Python 拓展之深拷貝淺拷貝(轉載)

總結: 無拷貝--則為引用 淺拷貝--列表依舊為引用 深拷貝--完全拷貝,列表不在為引用 轉載地址:https://www.cnblogs.com/Rocky0429/p/10088657.html 正式開始 首先我在這介紹兩個新的小知識,要在下面用到。一個是函

Python 拓展之深拷貝淺拷貝

正式開始 首先我在這介紹兩個新的小知識,要在下面用到。一個是函式 id() ,另一個是運算子 is。id() 函式就是返回物件的記憶體地址;is 是比較兩個變數的物件引用是否指向同一個物件,在這裡請不要和 == 混了,== 是比較兩個變數的值是否相等。 >

Python——__slots__,property私有方法

本文始發於個人公眾號:TechFlow,原創不易,求個關注 今天是Python專題的第11篇文章,我們來聊聊面向物件的一些進階使用。 __slots__ 如果你看過github當中一些大牛的程式碼,你會發現很多大牛經常在類的頂部加上__slots__關鍵字。如果你足夠好奇,你可能會試著把這個關鍵字去掉再執行

筆記:MyBatis Mapper XML文件 - 映射參數

gin server 頂級 ctp columns ref acl 目標 對象傳遞 MyBatis 的真正強大在於它的映射語句,也是它的魔力所在。由於它的異常強大,映射器的 XML 文件就顯得相對簡單。如果拿它跟具有相同功能的 JDBC 代碼進行對比,你會立即發現省掉了將近

Spring Boot 配置文件:PropertiesYAML

列表 config 其他 操作系統 des num mat 變量 onf 一.配置文件的生效順序,會對值進行覆蓋: 1. @TestPropertySource 註解 2. 命令行參數 3. Java系統屬性(System.getProperties

懶漢模式餓漢模式以及他們的改進

下一步 例子 理解 創建對象 進行 有一個 這就是 cnblogs 多人 提到單例模式的話相信很多人都不會陌生,一般初級程序員也知道懶漢模式和餓漢模式。 那麽什麽是單例模式呢?我個人低的理解就是當用這個類的對象的時候就只能創建同一個對象。是你,是你,還是你! 而在單例模式中

Linux rpm 命令參數使用[介紹應用]

使用詳解 gtk ont 由於 ins toolbar root 重新整理 完成後 參考來源:http://www.cnblogs.com/xiaochaohuashengmi/archive/2011/10/08/2203153.html rpm 執行安裝包二進制包(Bi

MyBatis之Mapper XML 文件(二)-sql入參

java mybatis sql 參數 mapper sql這個元素可以被用來定義可重用的 SQL 代碼段,可以包含在其他語句中。它可以被靜態地(在加載參數) 參數化. 不同的屬性值通過包含的實例變化. 比如:<sql id="userColumns"> $

Net Core中數據庫事務隔離——以DapperMysql為例

事務 ring 增刪改 tostring 測試 stc efault 多個 log Net Core中數據庫事務隔離詳解——以Dapper和Mysql為例 事務隔離級別 準備工作 Read uncommitted 讀未提交 Read committed 讀取提交內

Linux守護進程(init.dxinetd)

ger 發行版 數據報 支持 字符串 rdp 任務 超級 客戶機 一 Linux守護進程 Linux 服務器在啟動時需要啟動很多系統服務,它們向本地和網絡用戶提供了Linux的系統功能接口,直接面向應用程序和用戶。提供這些服務的程序是由運行在後臺的守護進程來執行的。守護進

【轉】Linux rpm 命令參數使用[介紹應用]

binary 包管理 samba cpio 詳解 -- hash pos 升級 RPM是RedHat Package Manager(RedHat軟件包管理工具)類似Windows裏面的“添加/刪除程序” rpm 執行安裝包二進制包(Binary)以及源代碼包(So

C#特性反射(三)

typeinfo ref 都是 system.in 全局 color com 依然 程序   類型信息(Type Information)用來表示類型聲明的信息,通過抽象基類System.Type的實例存儲這些信息,當使用反射時,CLR獲取指定類型的Type對象,通過這個對

Redis全方位--磁碟持久化容災備份

序言   在上一篇部落格中,部落格介紹了redis的資料型別使用場景和redis分散式鎖的正確姿勢。我們知道一旦Redis重啟,存在redis裡面的資料就會全部丟失。所以這篇部落格中向大家介紹Redis的磁碟持久化。   REDIS持久化   以每隔一段時間對redis進行快照的方

Linux用戶搶占內核搶占(概念, 實現觸發時機)--Linux進程的管理與調度(二十)

amp 3.1 not 職責 mon 顯式 default hust ron 1 非搶占式和可搶占式內核 為了簡化問題,我使用嵌入式實時系統uC/OS作為例子 首先要指出的是,uC/OS只有內核態,沒有用戶態,這和Linux不一樣 多任務系統中, 內核負責管理各個任務, 或

一文卷積逆卷積

文章目錄 一文詳解卷積和逆卷積 卷積運算 單通道 多通道 卷積運算的引數計算 逆卷積 卷積運算的矩陣實現 參考資料 一文詳解卷積和逆卷積 卷積神經

C#委託事件(二)

  一、當我們使用關鍵字delegate宣告一個自定義委託型別時,實際上是聲明瞭一個該名稱的類型別,繼承自抽象類System.MulticastDelegate,還包含例項方法Invoke、BeginInvoke、EndInvoke:   public delegate void MyDelegate