Python中的縮進(Indentation)決定了代碼的做用域範圍。這一點和傳統的c/c++有很大的不一樣(傳統的c/c++使用花括號花括號{}符決定做用域的範圍;python使用縮進空格來表示做用域的範圍,相同縮進行的代碼是處於同一範圍)。
每行代碼中開頭的空格數(whitespace)用於計算該行代碼的縮進級別(Indentation level),注意一個Tab會被替換爲1~8個Space(具體的空格數量,不一樣的編譯器有不一樣的數量),縮進級別爲0表示無縮進空格。html
在一個源文件不建議同時使用空格和製表縮進符,當使用別人代碼的時候幾乎是不可能知作別人使用的是空格仍是製表符,這時最好統一縮進,在IDEL編輯器中可使用 Edit > Untabify Region 將任何製表符轉化爲空格。python
Python中的每一條語句都有一個縮進級別,而且縮進級別會使用棧的數據結構進行存儲。在開始讀取文件以前,0(表示縮進級別爲0,無縮進)會被首先壓入棧中。而後從文件開頭到末尾,依次讀取每行邏輯代碼,每行邏輯代碼的縮進級別都會和棧頂值進行比較,若是相等,那麼什麼都不會發生;若是比棧頂值大的話,那麼該行邏輯代碼的縮進級別就會被壓入棧中,同時會生成一個縮進標記(INDENT TOKEN);若是比棧頂值小的話,那麼棧中全部比該行邏輯代碼縮進級別大的值都會從棧中移除,而且還會生成一個擴展標記(DEDENT TOKEN)。
下面是一個正確的縮進案例:c++
def perm(l):#0 # Compute the list of all permutations of l if len(l) <= 1:# 1 return [l]# 2 r = []# 3 for i in range(len(l)):# 4 s = l[:i] + l[i+1:]# 5 p = perm(s)# 6 for x in p:# 7 r.append(l[i:i+1] + x)# 8 return r# 9
上面的#0行,縮進0個字符,因爲文件讀取以前0已經被壓入棧中了,因此棧中的數據不會發生改變。#1縮進4個字符,4被壓入棧中。#2縮進18個字符,18被壓入棧中。#3縮進4個字符,18被彈出棧,棧頂值又爲4了。#4和#3縮進同樣,因此不更新棧數據。#5縮進13個字符,因此13被壓入棧中。#五、#6和#7的縮進同樣,不更新棧數據。#8縮進14個字符,14被壓入棧中。#9的縮進4個字符,因此棧中的13和14都會彈出,棧頂值又恢復爲4。
下面是一個錯誤的案例數據結構
def perm(l): #1 error: first line indented for i in range(len(l)): #2 error: not indented s = l[:i] + l[i+1:] p = perm(l[:i] + l[i+1:]) #3 error: unexpected indent for x in p: r.append(l[i:i+1] + x)#4 return r #5 error: inconsistent dedent
#一、#2和#3處的錯誤解釋的很清楚了。#5的縮進級別在棧中找不到因此出錯,#5處的縮進級別是14,比它的上面一行縮進級別18,因此在#5處應該進行出棧處理,可是在棧中找不到14這個級別,因此出錯。
又好比:app
if True: print("hello girl") else: print("hello boy") print("end")
最後一行代碼的縮進級別是1,並且比上一行代碼的縮進級別小,因此應該出棧處理,可是在出站的時候找不到棧中之前有1的級別,因此報錯。修改這個錯誤,只須要將最後一行的空格去掉就能夠。編輯器
參考文章:spa