抵御代碼復雜性使python函數(shù)更加Pythonic技巧示例詳解
掌握設計優(yōu)雅返回結(jié)果的函數(shù)的藝術(shù)
在開發(fā)解決方案時,我們傾向于將復雜的實際問題提煉為更小、更易于管理的子問題,然后使用函數(shù)來解決這些問題。函數(shù)是冗余代碼的克星,也是我們抵御代碼復雜性的最強防線。
大多數(shù)函數(shù)在編寫過程中,關(guān)鍵是其返回值。函數(shù)產(chǎn)生結(jié)果的方式會極大地影響用戶調(diào)用函數(shù)時的體驗。掌握設計優(yōu)雅返回結(jié)果的函數(shù)的藝術(shù)是制作高質(zhì)量函數(shù)的基礎。
不要返回多種類型
Python 是如此的靈活,以至于我們可以很容易地做到在其他語言中很難做到的事情。例如:讓函數(shù)同時返回不同類型的結(jié)果。就像下面這樣:
def get_users(user_id=None):
if user_id is not None:
return User.get(user_id)
else:
return User.filter(is_active=True)
# Return single user
get_users(user_id=1)
# Return all users
get_users()在上面的代碼片段中,當我們需要獲取單個用戶時,我們傳遞一個 "user_id " 參數(shù),否則,如果我們傳遞一個 None 值,它將返回所有活躍用戶的列表。乍一看,這種設計似乎很合理。
但是,編寫類似功能強大的函數(shù)并不是一件好事。這是因為優(yōu)秀的函數(shù)必須具有 單一職責。所謂 "單一職責",是指一個函數(shù)只做好一件事,而且目的明確。這樣的函數(shù)將來也不太可能隨著需求的變化而修改,同時也非常方便編寫單元測試。
返回多種類型的函數(shù)違反了 "單一職責 "原則。一個好的函數(shù)應始終提供一個穩(wěn)定的返回值,以盡量減少調(diào)用者的處理成本。就像上面的例子,我們應該編寫兩個獨立的函數(shù) get_active_users() 和 get_user_by_id(user_id)。
使用類型提示定義返回類型
使用類型提示定義返回類型和顯式參數(shù)聲明。這樣,集成開發(fā)環(huán)境就能幫助您進行自動補全和類型檢查,從而在編輯的時候就發(fā)現(xiàn)錯誤。
例如:
def say_hello(name: str) -> str:
return "Hello, " + name-> 語法表示 say_hello() 函數(shù)將返回一個字符串。
使用部分函數(shù)構(gòu)造新函數(shù)
假設在這種情況下,您的代碼中有一個帶有很多參數(shù)的函數(shù) A,它非常適用。另一個函數(shù) B 調(diào)用 A 做一些工作,就像下面這樣:
def add(x, y):
return x + y
def sum(value):
# Calling add
return add(100, value)在上述示例中,我們可以使用 functools 模塊中的 partial() 函數(shù)來簡化它。
import functools sum = functools.partial(add, 100) sum(200) # Output is 300
partial(func,*args,**kwargs) 以傳入的函數(shù)為基礎,使用變量參數(shù)構(gòu)造一個新函數(shù)。在合并當前調(diào)用參數(shù)和構(gòu)造參數(shù)后,對新函數(shù)的所有調(diào)用都將委托給原始函數(shù)。
因此,在使用部分函數(shù)時,可以將上面的求和函數(shù)定義修改為單行表達式,這樣會更加簡潔和直接。
拋出異常而不是返回錯誤
有時,您可能需要編寫同時返回結(jié)果和錯誤信息的函數(shù):
def create_user(name):
if len(name) > MAX_LENGTH_OF_NAME:
return None, 'name of user is too long'
if len(CURRENT_USERS) > MAX_USERS_QUOTA:
return None, 'too many users'
return User(name=name), ''def create_from_input():
name = input()
user, err_msg = create_user(name)
if err_msg:
print(f'create user failed: {err_msg}')
else:
print(f'user<{name}> created')在上例中,create_user 函數(shù)的作用是創(chuàng)建一個新的用戶對象。同時,為了在發(fā)生錯誤時向調(diào)用者提供錯誤詳細信息,它利用了多返回值特性,將錯誤信息作為第二個結(jié)果返回。
但在 Python 中,這并不是解決此類問題的最佳方法。因為這種做法會增加調(diào)用者處理錯誤的成本,尤其是當許多函數(shù)都遵循這種規(guī)范,并且存在多層調(diào)用時。
在這種情況下,使用異常來處理錯誤過程是更習以為常的做法。因此,上述代碼可以重寫為:
class CreateUserError(Exception):
"""Exception for user creation failure"""
pass
def create_user(name):
"""Create new user
:raises: CreateUserError
"""
if len(name) > MAX_LENGTH_OF_NAME:
raise CreateUserError('name of user is too long')
if len(CURRENT_USERS) > MAX_USERS_QUOTA:
raise CreateUserError('Too many users')
return User(name=name)
def create_for_input():
name = input()
try:
user = create_user(name)
except CreateUserError as e:
print(f'create user failed: {e}')
else:
print(f'user<{name}> created')在使用拋出異常而不是返回結(jié)果、錯誤信息后,整個錯誤處理過程乍一看變化不大,但實際上在一些細節(jié)上有很大不同:
• 新版函數(shù)的返回值類型更加穩(wěn)定,它將始終只返回用戶類型或拋出異常。
• 異常與返回值的不同之處在于,異常在被捕獲之前會不斷向調(diào)用棧的上層報告。因此,create_user 的一級調(diào)用者可以完全省略異常處理,將其留給上層處理。
盡量少返回 None
None 常被用來表示應該存在但缺少的東西,在 Python 中是獨一無二的。由于 None 獨特的虛無主義特質(zhì),它經(jīng)常被用作函數(shù)返回值。
當我們使用 None 作為函數(shù)的返回值時,通常有以下三種情況。
• 作為操作類函數(shù)的默認返回值:當操作類函數(shù)不需要任何返回值時,通常會返回 None。此外,對于沒有任何返回語句的函數(shù),None 也是默認返回值。例如,list.append()。
• 作為某個可能不存在的預期值:在 Python 標準庫中,正則表達式模塊下的函數(shù) re 就屬于這一類。例如,re.search 和 re.match。
• 作為代表錯誤結(jié)果的值:有時,當函數(shù)調(diào)用失敗時,我們經(jīng)常使用 None 作為默認返回值。如果是這種情況,請確保您的函數(shù)名稱更有意義,例如 create_user_or_none()。
限制遞歸的使用
當函數(shù)返回調(diào)用自身時,這就是遞歸。遞歸在某些情況下是非常有用的編程技巧,但 Python 對遞歸的支持非常有限。
Python 語言不支持尾部遞歸優(yōu)化。此外,Python 對遞歸級別的最大數(shù)量也有嚴格限制。所以要盡可能少寫遞歸。如果您想用遞歸來解決問題,首先要考慮是否可以用循環(huán)來輕松替代遞歸。如果答案是肯定的,那就用循環(huán)重寫。如果絕對必須使用遞歸,請考慮以下幾點:
• 確保遞歸層小于 sys.getrecursionlimit()。
• 盡可能使用緩存,例如 functools.lru_cache。
以上就是抵御代碼復雜性使python函數(shù)更加Pythonic技巧示例詳解的詳細內(nèi)容,更多關(guān)于python函數(shù)Pythonic的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
用Python實現(xiàn)一個打字速度測試工具來測試你的手速
有很多小伙伴們都苦惱自己手速不夠,今天特地整理了這篇文章,教你用Python實現(xiàn)一個打字測試工具來測試你的打字速度,文中有非常詳細的代碼示例,對想練手速的小伙伴們很有用哦,需要的朋友可以參考下2021-05-05
Python?創(chuàng)建或讀取?Excel?文件的操作代碼
Excel是一種常用的電子表格軟件,廣泛應用于金融、商業(yè)和教育等領(lǐng)域,本文介紹Python?創(chuàng)建或讀取?Excel?文件的操作代碼,感興趣的朋友一起看看吧2023-09-09
Python實現(xiàn)音頻添加數(shù)字水印的示例詳解
數(shù)字水印技術(shù)可以將隱藏信息嵌入到音頻文件中而不明顯影響音頻質(zhì)量,下面小編將介紹幾種在Python中實現(xiàn)音頻數(shù)字水印的方法,希望對大家有所幫助2025-04-04
對python使用telnet實現(xiàn)弱密碼登錄的方法詳解
今天小編就為大家分享一篇對python使用telnet實現(xiàn)弱密碼登錄的方法詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-01-01
python中Matplotlib實現(xiàn)繪制3D圖的示例代碼
本篇文章主要介紹了python中Matplotlib實現(xiàn)繪制3D圖的示例代碼,具有一定的參考價值,有興趣的可以了解一下2017-09-09
python實現(xiàn)apahce網(wǎng)站日志分析示例
這篇文章主要介紹了python實現(xiàn)apahce網(wǎng)站日志分析示例,需要的朋友可以參考下2014-04-04

