Note One:about sequences and lists
序列是不可變列表。一旦創建了一個序列就不能以任何方式改變它。
例 1.21. 定義序列
>>>
t = ("a", "b", "mpilgrim", "z", "example")
>>>
t
('a', 'b', 'mpilgrim', 'z', 'example')
>>>
t[0]
'a'
>>>
t[-1]?????????????????????????????????????
'example'
>>>
t[1:3]????????????????????????????????????
('b', 'mpilgrim')
|
序列的定義同列表的定義方式相同,除了整個元素集是用小括號包圍的而不是方括號。
|
|
序列的元素象列表一樣按定義的次序進行排序。序列的索引象列表一樣從0開始,所以一個非空序列的第一個元素總是 t[0]。
|
|
負數索引象列表一樣從序列尾部開始計數。
|
|
分片也可以使用,就象列表一樣。注意當分割一個列表時,會得到一個新的列表;當分割一個序列時,會得到一個新的序列。
|
例 1.22. 序列沒有方法?
>>>?
t
('a', 'b', 'mpilgrim', 'z', 'example')
>>>?
t.append("new")??
Traceback (innermost last):
File "<interactive input>", line 1, in ?
AttributeError: 'tuple' object has no attribute 'append'
>>>?
t.remove("z")??
?????????????
Traceback (innermost last):
File "<interactive input>", line 1, in ?
AttributeError: 'tuple' object has no attribute 'remove'
>>>?t.index("example")??
Traceback (innermost last):
? File "<interactive input>", line 1, in ?
AttributeError: 'tuple' object has no attribute 'index'
>>>"z"?in t??
1???
|
你不能向序列增加元素。序列沒有 append 或 extend 方法。
|
|
你不能從序列中除掉元素。序列沒有 remove 或 pop 方法。
|
|
你不能在序列中查找元素。序列沒有 index 方法。
|
|
然而,你可以使用 in 來看一看是否一個元素存在于序列中。
|
那么序列有什么好處呢?
-
序列比列表操作速度快。如果你定義了一個值集合常量,并且唯一要用它做的是不斷地遍歷它,使用序列代替列表。
-
記得我說過
字典關鍵字
可以是整數,字符串和“幾種其它的類型”嗎?序列就是那些類型之一。序列可以在字典中被用作關鍵字,但是列表不行。[2]
-
序列用在字符串格式化,這一點我們會很快看到。
Note Two : Mapping in the Lists
列表映射介紹?
??????
>>>??li = [1, 9, 8, 4]
>>>??[elem*2 for elem in li]????? 
[2, 18, 16, 8]
>>>??li?????????????????????
[1, 9, 8, 4]
|
為了對這一點有一個感性認識,從右向左看它。 li 是一個將要映射的列表。Python循環遍歷 li 一次一個元素,臨時將每個元素的值賦給變量 elem。然后Python使用函數 elem*2 ,接著將結果追加到返回列表中。
|
|
注意列表映射不改變被映射的列表。
|
Note Three :好東西?lambda?函數
Python支持一種有趣的語法,它允許你快速定義單行的最小函數。這些叫做 lambda 的函數是從Lisp中借用來的,可以被用在任何需要函數的地方。
出于歷史的原因,lambda 函數的語法與通常的函數有些細微的不同。
例 2.20. lambda 函數介紹
>>>
def
f(x):?????????
...????
return x*2
...????
>>>
f(3)
6
>>>
g = lambda x: x*2?
>>>
g(3)
6
>>>
(lambda x: x*2)(3)
6
|
這是一個通常的函數聲明,盡管以前你可能沒有看到過定義在交互式窗口中的函數。這個 ... 說明它是一個多行的交互語句。只要在第一行的后面敲入回車,Python IDE會讓你接著輸入命令。
|
|
這是一個 lambda 函數,它完成同上面普通函數相同的事情。注意這里的簡短的語法;沒有小括號, return 是默認的,并且函數沒有名字,只有將它賦值給變量的變量名。
|
|
你甚至可以不將 lambda 函數賦值給一個變量而使用它。這不是舉世無雙的東西,它只是展示了 lambda 函數只是一個內聯函數。
|
總之, lambda 函數是一個可以接收任意多個參數(包括
可選參數
)并且返回單個表達式值的函數。 lambda 函數不能包含命令,它們所包含的表達式不能超過一個。不要試圖向 lambda 函數中塞入太多的東西;如果你需要更復雜的東西,應該定義一個普通函數,然后想讓它多長就多長。
|
lambda 函數是風格問題。不一定非要使用它們,任何能夠使用它們的地方,都可以定義一個分離的普通的函數,用它來替換。我將它們用在需要封裝特殊的,非重用的代碼上,用許多小的一行函數不會弄亂我的代碼。
|
例 2.21. 在 in apihelper.py 中的 lambda 函數?
processFunc = collapse and (lambda s: " ".join(s.split())) or (lambda s: s)
順便這里有幾件事情需要注意。首先,我們使用了
and-or
技巧的簡單形式,沒問題,因為一個 lambda 函數
在一個布爾環境下
總為真。(這并不意味著 lambda 函數不能返回假值。函數本身總是為真,它的返回值可以為任何值。)
第二,我們使用了 split 函數沒帶參數。你已經看到過它帶
1個或2個參數
的使用,但是不帶參數它按空白進行分割。
例 2.22. split 不帶參數????
>>>s = "this?? is\na\ttest"???????? 
>>>print s??
?this?? is
a test
>>>print s.split()????? 
['this', 'is', 'a', 'test']
>>>print???" ".join(s.split())??????? 
'this is a test'????
|
這是一個多行字符串,通過轉義字符的定義代替了
三重引號
。 \n 是一個回車; \t 是一個制表符。
|
|
split 不帶參數按空白進行分割。所以三個空格,一個回車,和一個制表符都是一樣的。
|
|
你可以將空白統一化,通過分割一個字符串,然后用單個空格作為分隔符將其重新接起來。這就是 help 函數所做的,將多行文檔字符串合并成單行。
|
那么 help 函數到底用這些 lambda 函數, split 函數,和 and-or 技巧做了什么呢?
例 2.23. 將函數賦給一個變量
processFunc = collapse and (lambda s: " ".join(s.split())) or (lambda s: s)
processFunc 現在是一個函數,但它為哪一個函數要看 collapse 變量的值。如果 collapse 為真, processFunc(string) 將壓縮空白;否則,processFunc(string) 將返回未改變的參數。
在一個不很建壯的語言實現它,象VB,你將可能創建一個函數,它接收一個字符串和一個 collapse 參數,使用一個 if 語句來判斷是否要壓縮空白或不壓縮,然后返回相應的值。這樣效率低,因為函數將不得不處理每種可能性;每次你調用它,它將不得不在給出你所想要的東西之前,判斷是否要壓縮空白。在Python中,你可以將那種判斷邏輯拿到函數外面,而定義一個裁減過的 lambda 函數來給出確切的(并且唯一)你想要的。這樣做更有效率,更漂亮,并且更少導致那些令人討厭的(哦,想到那些參數就頭昏)的錯誤。