1. 程式人生 > >python0.10-----變量作用域和異常

python0.10-----變量作用域和異常

fun AR 範圍 finally 作用 clas 變量 適合 內存

變量的作用域:也可以稱為命名空間,即變量可以使用的範圍。程序的變量並不是在所有的位置都能使用的,訪問的權限決定於變量在哪裏賦值(定義)的。只有當變量在模塊,類,函數中定義的時候,才會有作用域的概念。if分支結構,for-else,while,for遍歷循環結構,try-except(-else)\try-except-finally等關鍵字語句塊不會產生作用域。

局部變量與局部作用域:

#定義函數

def discount(price,rate)

  final_price=price*rate

  return final_price

#調用函數

new_price=discount(price=10,rate=0.5)

print(final_price)

解讀:當執行discount(price=10,rate=0.5)這個函數調用語句時,系統就會在棧空間裏開辟內存,定以變量price,rate,final_price並給它們賦值。調用結束後,也就是discount(price=10,rate=0.5)語句結束時,變量price,rate,final_price會被自動清除,因此上面print(final_price)這個訪問局部變量的語句會引發異常NameError:name ‘final_price’ is not defined。可以把price,rate,final_price變量作用的範圍成為局部作用域(local)。

全局變量與全局作用域:

def func():

  b=2 #當給b賦值時,由於屏蔽機制,在棧空間中創建局部變量b。

  print(b) #在函數中訪問局部變量,大小為2。

b=1 #這裏的b是全局變量 ,大小為1。

func() #調用函數

解讀:全局變量b的作用範圍是整個模塊的代碼塊(稱為全局作用域),在局部函數中可以隨意訪問全局變量b。在函數內試圖修改全局變量b時,python使用屏蔽機制,函數會屏蔽全局變量b,在棧區域定義一個名叫b的局部變量,兩個變量雖然名字相同,但是它們在內存中的位置不一樣,它們不一樣!!!

註意:可以使用global關鍵字在函數內部修改或者創建全局變量。但是建議不要這麽做,因為它會使程序可讀性變差,程序很難維護,做為函數盡自己的本分就好。例如:

def func():

  global b #屏蔽機制失效,可以在函數內操縱全局變量

  b=2

print(b) #在函數中訪問全局變量

b=1 #全局變量

print(b) #此時b的值還是1

func() #調用函數

print(b) #此時b的值被func函數改成2了

外部變量與嵌套作用域:

在解釋這個概念前先引入閉包的概念:

閉包(closure):python中閉包從表現形式上定義為:如果在一個內部函數裏,對外部作用域(但不是在全局作用域)的變量進行引用,那麽內部函數就會被認為是閉包(closure)。

外部變量和嵌套作用域:

def FunX(x):

  def FunY(y):

    return x*y

  return FunY

解讀:對於FunY來說,FunX就是外部作用域,x是外部作用域的變量,FunY是內部函數,並且FunY函數引用了這個變量x。符合這兩個條件,那麽可稱內部函數FunY就是一個閉包。事實上,這裏說的外部作用於就是嵌套作用域(enclosing)。全局範圍內,不能調用FunY。局部變量作用於外部變量,跟之前局部變量作用於全局變量一樣。局部變量可以訪問外部變量,但不能直接修改外部變量,若修改,則屏蔽機制生效,在閉包中創建局部變量,並且不影響外部變量。

def FunX(x): #x為外部變量

  def FunY(y): #y為局部變量,FunY為閉包

    x=1 #當給x賦值時,由於屏蔽機制,在棧空間中創建局部變量x,不影響外部變量x。

    return x*y

  return FunY

若執意修改,采用nonlocal的關鍵字進行修改。但是建議不要這麽做,因為它會使程序可讀性變差,程序很難維護,做為閉包盡自己的本分就好。

例子:

def FunX(x): #x為外部變量

  def FunY(y): #y為局部變量,FunY函數為閉包

    nonlocal x #聲明操作的x為外部變量

    x=1 #現在外部變量x的值為1。

    return x*y

  return FunY

異常處理:

一般來說編碼錯誤叫做bug。而不是由於編碼錯誤產生,而是由於用戶操作失誤,程序運行會報錯,這叫做異常。程序可預測的錯誤,例如存儲空間不足,網絡錯誤,也叫做異常。若程序出現問題,希望程序越過錯誤,不要停止執行。此時可以用異常處理技術。

異常處理的結構有2種:分為try-execpt(-else),try-except-finally

try-execpt(-else):

格式:

try:

  語句t

except 異常碼1 as e:

  語句1

except 異常碼2 as e:

  語句2

………………

except 異常碼n as e:

  語句n

else:   #else語句可有可無

  語句e

邏輯:

1:系統檢測try語句塊中的語句t是否有異常,而異常的類型是多種多樣的,不同的異常對應不同的異常碼,發生異常時,except捕獲對應的異常碼,執行相應的異常處理語句。

2:若系統檢測try語句塊中的語句t出現異常,但是沒有捕獲相應的異常碼,那麽異常會被提交到上一層的try語句,或者到程序的最上層。

3:若系統檢測try語句塊中的語句t沒有出現異常,則執行else分支下的語句e。

4:詳細的錯誤原因會放到e中,e為字符串類型。

善變的try-execpt(-else)之一:

#使用except而不使用任何的異常類型,這個很常用。因為查異常類型很麻煩,而且也很難處理。

try:

  語句t

except :

  語句e

善變的try-execpt(-else)之二:

#使用except帶著多種異常

try:

  語句t

except (NameError,ZeroDivisionError):

  語句e

解釋:若出現了NameError,ZeroDivisionError,都會在except裏面處理。

try-except-finally:

格式:

try:

  語句t

except 異常碼1 as e:

  語句1

except 異常碼2 as e:

  語句2

………………

except 異常碼n as e:

  語句n

finally:

  語句e

#不管語句t有沒有出錯,語句e必定執行。

用處:在文件處理中,應該使用finally,因為文件讀寫的過程中,不管讀寫過程是否出錯,文件的關閉必須要執行,因此這裏非常適合finally來使用。

異常機制還需註意:

#1 .異常其實是class類,所有的錯誤都繼承自BaseExeception。所以在捕獲的時候,它捕獲了該類型的異常,還把子類的異常一網打盡。

#2.若內存存滿了,需要在異常處理裏面釋放一部分內存。

#3.跨多層調用,main()調用func2,func2調用func1,只要調用main()時補獲了異常,就可以處理,程序可以不停止運行。

例如:
#定義

def func1(num):

  print(1/num)

def func2(num):

  func1(num)

def main():

  func2(0)

try:

  main()   #調用

except ZeroDivision as e:

  語句e

斷言(assert):

格式:

assert 表達式,’異常提示’

解釋:當表達式為真時,不會出現異常提示。當表達式為假時,會拋出assert異常,異常類型後面可以跟隨自己編寫的‘異常提示’,‘異常提示’可要可不要。當然assert拋出的異常也是可以捕獲的。但是assert一定要在try裏面使用,並且表達式的格式一定要正確,異常提示會賦值給e。

例如:

try:

  assert 2==1,‘我好快樂‘

except BaseException as e:

  print(‘發生錯誤‘,e)

else:

  print(‘沒有發生錯誤‘)

#輸出:

發生錯誤 我好快樂

標準異常名稱 描述

BaseException 所有異常的基類

SystemExit 解釋器請求退出

KeyboardInterrupt 用戶中斷執行(通常是輸入^C)

Exception 常規錯誤的基類

StopIteration 叠代器沒有更多的值

GeneratorExit 生成器(generator)發生異常來通知退出

SystemExit Python 解釋器請求退出

StandardError 所有的內建標準異常的基類

ArithmeticError 所有數值計算錯誤的基類

FloatingPointError 浮點計算錯誤

OverflowError 數值運算超出最大限制

ZeroDivisionError 除(或取模)零 (所有數據類型)

AssertionError 斷言語句失敗

AttributeError 對象沒有這個屬性

EOFError 沒有內建輸入,到達EOF 標記

EnvironmentError 操作系統錯誤的基類

IOError 輸入/輸出操作失敗

OSError 操作系統錯誤

WindowsError 系統調用失敗

ImportError 導入模塊/對象失敗

KeyboardInterrupt 用戶中斷執行(通常是輸入^C)

LookupError 無效數據查詢的基類

IndexError 序列中沒有沒有此索引(index)

KeyError 映射中沒有這個鍵

MemoryError 內存溢出錯誤(對於Python 解釋器不是致命的)

NameError 未聲明/初始化對象 (沒有屬性)

UnboundLocalError 訪問未初始化的本地變量

ReferenceError 弱引用(Weak reference)試圖訪問已經垃圾回收了的對象

RuntimeError 一般的運行時錯誤

NotImplementedError 尚未實現的方法

SyntaxError Python 語法錯誤

IndentationError 縮進錯誤

TabError Tab 和空格混用

SystemError 一般的解釋器系統錯誤

TypeError 對類型無效的操作

ValueError 傳入無效的參數

UnicodeError Unicode 相關的錯誤

UnicodeDecodeError Unicode 解碼時的錯誤

UnicodeEncodeError Unicode 編碼時錯誤

UnicodeTranslateError Unicode 轉換時錯誤

Warning 警告的基類

DeprecationWarning 關於被棄用的特征的警告

FutureWarning 關於構造將來語義會有改變的警告

OverflowWarning 舊的關於自動提升為長整型(long)的警告

PendingDeprecationWarning 關於特性將會被廢棄的警告

RuntimeWarning 可疑的運行時行為(runtime behavior)的警告

SyntaxWarning 可疑的語法的警告

UserWarning 用戶代碼生成的警告

python0.10-----變量作用域和異常