欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

分析在Python中何種情況下需要使用斷言

 更新時間:2015年04月01日 11:57:25   投稿:goldensun  
這篇文章主要介紹了分析在Python中何種情況下需要使用斷言,以避免在斷言使用中經(jīng)??赡芘龅降腻e誤,作者給出了具體代碼示例,需要的朋友可以參考下

這個問題是如何在一些場景下使用斷言表達(dá)式,通常會有人誤用它,所以我決定寫一篇文章來說明何時使用斷言,什么時候不用。

為那些還不清楚它的人,Python的assert是用來檢查一個條件,如果它為真,就不做任何事。如果它為假,則會拋出AssertError并且包含錯誤信息。例如:
 

py> x = 23
py> assert x > 0, "x is not zero or negative"
py> assert x%2 == 0, "x is not an even number"
Traceback (most recent call last):
File "", line 1, in
AssertionError: x is not an even number

很多人用assert作為一個很快和容易的方法來在參數(shù)錯誤的時候拋出異常。但這樣做是錯的,非常錯誤,有兩個原因。首先AssertError不是在測試參數(shù)時應(yīng)該拋出的錯誤。你不應(yīng)該像這樣寫代碼:
 

if not isinstance(x, int):
raise AssertionError("not an int")

你應(yīng)該拋出TypeError的錯誤,assert會拋出錯誤的異常。

但是,更危險的是,有一個關(guān)于assert的困擾:它可以被編譯好然后從來不執(zhí)行,如果你用 –O 或 –oo 選項(xiàng)運(yùn)行Python,結(jié)果不保證assert表達(dá)式會運(yùn)行到。當(dāng)適當(dāng)?shù)氖褂胊ssert時,這是未來,但是當(dāng)assert不恰當(dāng)?shù)氖褂脮r,它會讓代碼用-O執(zhí)行時出錯。

那什么時候應(yīng)該使用assert?沒有特定的規(guī)則,斷言應(yīng)該用于:

  •     防御型的編程
  •     運(yùn)行時檢查程序邏輯
  •     檢查約定
  •     程序常量
  •     檢查文檔

(在測試代碼的時候使用斷言也是可接受的,是一種很方便的單元測試方法,你接受這些測試在用-O標(biāo)志運(yùn)行時不會做任何事。我有時在代碼里使用assert False來標(biāo)記沒有寫完的代碼分支,我希望這些代碼運(yùn)行失敗。盡管拋出NotImplementedError可能會更好。)

關(guān)于斷言的意見有很多,因?yàn)樗艽_保代碼的正確性。如果你確定代碼是正確的,那么就沒有用斷言的必要了,因?yàn)樗麄儚膩聿粫\(yùn)行失敗,你可以直接移除這些斷言。如果你確定檢查會失敗,那么如果你不用斷言,代碼就會通過編譯并忽略你的檢查。

在以上兩種情況下會很有意思,當(dāng)你比較肯定代碼但是不是絕對肯定時??赡苣銜e過一些非常古怪的情況。在這個情況下,額外的運(yùn)行時檢查能幫你確保任何錯誤都會盡早地被捕捉到。

另一個好的使用斷言的方式是檢查程序的不變量。一個不變量是一些你需要依賴它為真的情況,除非一個bug導(dǎo)致它為假。如果有bug,最好能夠盡早發(fā)現(xiàn),所以我們?yōu)樗M(jìn)行一個測試,但是又不想減慢代碼運(yùn)行速度。所以就用斷言,因?yàn)樗茉陂_發(fā)時打開,在產(chǎn)品階段關(guān)閉。

一個非變量的例子可能是,如果你的函數(shù)希望在它開始時有數(shù)據(jù)庫的連接,并且承諾在它返回的時候仍然保持連接,這就是函數(shù)的不變量:
 

def some_function(arg):
  assert not DB.closed()
  ... # code goes here
  assert not DB.closed()
  return result

斷言本身就是很好的注釋,勝過你直接寫注釋:

# when we reach here, we know that n > 2

你可以通過添加斷言來確保它:

assert n > 2

斷言也是一種防御型編程。你不是讓你的代碼防御現(xiàn)在的錯誤,而是防止在代碼修改后引發(fā)的錯誤。理想情況下,單元測試可以完成這樣的工作,可是需要面對的現(xiàn)實(shí)是,它們通常是沒有完成的。人們可能在提交代碼前會忘了運(yùn)行測試代碼。有一個內(nèi)部檢查是另一個阻擋錯誤的防線,尤其是那些不明顯的錯誤,卻導(dǎo)致了代碼出問題并且返回錯誤的結(jié)果。

加入你有一些if…elif 的語句塊,你知道在這之前一些需要有一些值:
 

# target is expected to be one of x, y, or z, and nothing else.
if target == x:
  run_x_code()
elif target == y:
  run_y_code()
else:
  run_z_code()

假設(shè)代碼現(xiàn)在是完全正確的。但它會一直是正確的嗎?依賴的修改,代碼的修改。如果依賴修改成 target = w 會發(fā)生什么,會關(guān)系到run_w_code函數(shù)嗎?如果我們改變了代碼,但沒有修改這里的代碼,可能會導(dǎo)致錯誤的調(diào)用 run_z_code 函數(shù)并引發(fā)錯誤。用防御型的方法來寫代碼會很好,它能讓代碼運(yùn)行正確,或者立馬執(zhí)行錯誤,即使你在未來對它進(jìn)行了修改。

在代碼開頭的注釋很好的一步,但是人們經(jīng)常懶得讀或者更新注釋。一旦發(fā)生這種情況,注釋會變得沒用。但有了斷言,我可以同時對代碼塊的假設(shè)書寫文檔,并且在它們違反的時候觸發(fā)一個干凈的錯誤
 

assert target in (x, y, z)
if target == x:
  run_x_code()
elif target == y:
  run_y_code()
else:
  assert target == z
  run_z_code()

這樣,斷言是一種防御型編程,同時也是一種文檔。我想到一個更好的方案:
 

if target == x:
  run_x_code()
elif target == y:
  run_y_code()
elif target == z:
  run_z_code()
else:
  # This can never happen. But just in case it does...
  raise RuntimeError("an unexpected error occurred")

按約定進(jìn)行設(shè)計(jì)是斷言的另一個好的用途。我們想象函數(shù)與調(diào)用者之間有個約定,比如下面的:

“如果你傳給我一個非空字符串,我保證傳會字符串的第一個字母并將其大寫?!?/p>

如果約定被函數(shù)或調(diào)用這破壞,代碼就會出問題。我們說函數(shù)有一些前置條件和后置條件,所以函數(shù)就會這么寫:
 

def first_upper(astring):
  assert isinstance(astring, str) and len(astring) > 0
  result = astring[0].upper()
  assert isinstance(result, str) and len(result) == 1
  assert result == result.upper()
  return result

按約定設(shè)計(jì)的目標(biāo)是為了正確的編程,前置條件和后置條件是需要保持的。這是斷言的典型應(yīng)用場景,因?yàn)橐坏┪覀儼l(fā)布了沒有問題的代碼到產(chǎn)品中,程序會是正確的,并且我們能安全的移除檢查。

下面是我建議的不要用斷言的場景:

  •     不要用它測試用戶提供的數(shù)據(jù)
  •     不要用斷言來檢查你覺得在你的程序的常規(guī)使用時會出錯的地方。斷言是用來檢查非常罕見的問題。你的用戶不應(yīng)該看到任何斷言錯誤,如果他們看到了,這是一個bug,修復(fù)它。
  •     有的情況下,不用斷言是因?yàn)樗染_的檢查要短,它不應(yīng)該是懶碼農(nóng)的偷懶方式。
  •     不要用它來檢查對公共庫的輸入?yún)?shù),因?yàn)樗荒芸刂普{(diào)用者,所以不能保證調(diào)用者會不會打破雙方的約定。
  •     不要為你覺得可以恢復(fù)的錯誤用斷言。換句話說,不用改在產(chǎn)品代碼里捕捉到斷言錯誤。
  •     不要用太多斷言以至于讓代碼很晦澀。

相關(guān)文章

  • Django學(xué)習(xí)筆記之Class-Based-View

    Django學(xué)習(xí)筆記之Class-Based-View

    下面這篇文章主要介紹了Class-based View,為什么要有這個Class-based View呢?view不都是一個方法嗎?跟類有啥關(guān)系?其實(shí)答案很明顯,用類其實(shí)是為了抽象,抽象出通用的,將可變的暴露出來,這樣我們就可以用最少的代碼實(shí)現(xiàn)復(fù)雜的功能了。下面來看看詳細(xì)的介紹吧。
    2017-02-02
  • Pandas的Apply函數(shù)具體使用

    Pandas的Apply函數(shù)具體使用

    這篇文章主要介紹了Pandas的Apply函數(shù)具體使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-07-07
  • 解密Python中的描述符(descriptor)

    解密Python中的描述符(descriptor)

    這篇文章主要介紹了解密Python中的描述符(descriptor),本文詳細(xì)講解了描述符(descriptor)的作用、訪問描述符、對描述符賦值、刪除描述符等內(nèi)容,需要的朋友可以參考下
    2015-06-06
  • Numpy 數(shù)組操作之元素添加、刪除和修改的實(shí)現(xiàn)

    Numpy 數(shù)組操作之元素添加、刪除和修改的實(shí)現(xiàn)

    本文主要介紹了Numpy 數(shù)組操作之元素添加、刪除和修改的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • Python?pandas替換指定數(shù)據(jù)的方法實(shí)例

    Python?pandas替換指定數(shù)據(jù)的方法實(shí)例

    在處理數(shù)據(jù)的時候,很多時候會遇到替換的情況,下面這篇文章主要給大家介紹了關(guān)于Python?pandas替換指定數(shù)據(jù)的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-07-07
  • 使用PM2+nginx部署python項(xiàng)目的方法示例

    使用PM2+nginx部署python項(xiàng)目的方法示例

    這篇文章主要介紹了使用PM2+nginx部署python項(xiàng)目的方法示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-11-11
  • python numpy創(chuàng)造圖像矩陣示例詳解

    python numpy創(chuàng)造圖像矩陣示例詳解

    這篇文章主要為大家介紹了python numpy創(chuàng)造圖像矩陣示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-08-08
  • PyAutoGUI圖形用戶界面自動化的超詳細(xì)教程

    PyAutoGUI圖形用戶界面自動化的超詳細(xì)教程

    PyautoGUI是一個純Python的自動化工具,能實(shí)現(xiàn)用程序自動控制鼠標(biāo)和鍵盤操作,下面這篇文章主要給大家介紹了關(guān)于PyAutoGUI圖形用戶界面自動化的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-04-04
  • Python切片操作深入詳解

    Python切片操作深入詳解

    這篇文章主要介紹了Python切片操作,結(jié)合實(shí)例形式詳細(xì)深入的分析了Python切片操作的原理、參數(shù)屬性、相關(guān)函數(shù)及操作注意事項(xiàng),需要的朋友可以參考下
    2018-07-07
  • Python實(shí)戰(zhàn)之markdown轉(zhuǎn)pdf(包含公式轉(zhuǎn)換)

    Python實(shí)戰(zhàn)之markdown轉(zhuǎn)pdf(包含公式轉(zhuǎn)換)

    由于我們markdown編輯器比較特殊,不是很方便瀏覽,如果轉(zhuǎn)換成pdf的話,就不需要可以的去安裝各種編輯器才可以看了。所以本文將介紹如何通過Python實(shí)現(xiàn)md轉(zhuǎn)pdf或者是docx,需要的朋友可以參考一下
    2021-12-12

最新評論