5. 數(shù)據(jù)結(jié)構(gòu)?
本章深入講解之前學(xué)過的一些內(nèi)容,同時(shí),還增加了新的知識(shí)點(diǎn)。
5.1. 列表詳解?
列表數(shù)據(jù)類型支持很多方法,列表對(duì)象的所有方法所示如下:
- list.append(x)
在列表末尾添加一個(gè)元素,相當(dāng)于
a[len(a):] = [x]
。
- list.extend(iterable)
用可迭代對(duì)象的元素?cái)U(kuò)展列表。相當(dāng)于
a[len(a):] = iterable
。
- list.insert(i, x)
在指定位置插入元素。第一個(gè)參數(shù)是插入元素的索引,因此,
a.insert(0, x)
在列表開頭插入元素,a.insert(len(a), x)
等同于a.append(x)
。
- list.remove(x)
從列表中刪除第一個(gè)值為 x 的元素。未找到指定元素時(shí),觸發(fā)
ValueError
異常。
- list.pop([i])
刪除列表中指定位置的元素,并返回被刪除的元素。未指定位置時(shí),
a.pop()
刪除并返回列表的最后一個(gè)元素。(方法簽名中 i 兩邊的方括號(hào)表示該參數(shù)是可選的,不是要求輸入方括號(hào)。這種表示法常見于 Python 參考庫(kù))。
- list.clear()
刪除列表里的所有元素,相當(dāng)于
del a[:]
。
- list.index(x[, start[, end]])
返回列表中第一個(gè)值為 x 的元素的零基索引。未找到指定元素時(shí),觸發(fā)
ValueError
異常。可選參數(shù) start 和 end 是切片符號(hào),用于將搜索限制為列表的特定子序列。返回的索引是相對(duì)于整個(gè)序列的開始計(jì)算的,而不是 start 參數(shù)。
- list.count(x)
返回列表中元素 x 出現(xiàn)的次數(shù)。
- list.sort(*, key=None, reverse=False)
就地排序列表中的元素(要了解自定義排序參數(shù),詳見
sorted()
)。
- list.reverse()
翻轉(zhuǎn)列表中的元素。
- list.copy()
返回列表的淺拷貝。相當(dāng)于
a[:]
。
多數(shù)列表方法示例:
>>> fruits = ['orange', 'apple', 'pear', 'banana', 'kiwi', 'apple', 'banana']
>>> fruits.count('apple')
2
>>> fruits.count('tangerine')
0
>>> fruits.index('banana')
3
>>> fruits.index('banana', 4) # Find next banana starting a position 4
6
>>> fruits.reverse()
>>> fruits
['banana', 'apple', 'kiwi', 'banana', 'pear', 'apple', 'orange']
>>> fruits.append('grape')
>>> fruits
['banana', 'apple', 'kiwi', 'banana', 'pear', 'apple', 'orange', 'grape']
>>> fruits.sort()
>>> fruits
['apple', 'apple', 'banana', 'banana', 'grape', 'kiwi', 'orange', 'pear']
>>> fruits.pop()
'pear'
insert
、remove
、sort
等方法只修改列表,不輸出返回值——返回的默認(rèn)值為 None
。1 這是所有 Python 可變數(shù)據(jù)結(jié)構(gòu)的設(shè)計(jì)原則。
還有,不是所有數(shù)據(jù)都可以排序或比較。例如,[None, 'hello', 10]
就不可排序,因?yàn)檎麛?shù)不能與字符串對(duì)比,而 None 不能與其他類型對(duì)比。有些類型根本就沒有定義順序關(guān)系,例如,3+4j < 5+7j
這種對(duì)比操作就是無效的。
5.1.1. 用列表實(shí)現(xiàn)堆棧?
使用列表方法實(shí)現(xiàn)堆棧非常容易,最后插入的最先取出(“后進(jìn)先出”)。把元素添加到堆棧的頂端,使用 append()
。從堆棧頂部取出元素,使用 pop()
,不用指定索引。例如:
>>> stack = [3, 4, 5]
>>> stack.append(6)
>>> stack.append(7)
>>> stack
[3, 4, 5, 6, 7]
>>> stack.pop()
7
>>> stack
[3, 4, 5, 6]
>>> stack.pop()
6
>>> stack.pop()
5
>>> stack
[3, 4]
5.1.2. 用列表實(shí)現(xiàn)隊(duì)列?
列表也可以用作隊(duì)列,最先加入的元素,最先取出(“先進(jìn)先出”);然而,列表作為隊(duì)列的效率很低。因?yàn)?,在列表末尾添加和刪除元素非常快,但在列表開頭插入或移除元素卻很慢(因?yàn)樗衅渌囟急仨氁苿?dòng)一位)。
實(shí)現(xiàn)隊(duì)列最好用 collections.deque
,可以快速?gòu)膬啥颂砑踊騽h除元素。例如:
>>> from collections import deque
>>> queue = deque(["Eric", "John", "Michael"])
>>> queue.append("Terry") # Terry arrives
>>> queue.append("Graham") # Graham arrives
>>> queue.popleft() # The first to arrive now leaves
'Eric'
>>> queue.popleft() # The second to arrive now leaves
'John'
>>> queue # Remaining queue in order of arrival
deque(['Michael', 'Terry', 'Graham'])
5.1.3. 列表推導(dǎo)式?
列表推導(dǎo)式創(chuàng)建列表的方式更簡(jiǎn)潔。常見的用法為,對(duì)序列或可迭代對(duì)象中的每個(gè)元素應(yīng)用某種操作,用生成的結(jié)果創(chuàng)建新的列表;或用滿足特定條件的元素創(chuàng)建子序列。
例如,創(chuàng)建平方值的列表:
>>> squares = []
>>> for x in range(10):
... squares.append(x**2)
...
>>> squares
[0, 1, 4, 9, 16, 25, 36, 49, 64, 81]
注意,這段代碼創(chuàng)建(或覆蓋)變量 x
,該變量在循環(huán)結(jié)束后仍然存在。下述方法可以無副作用地計(jì)算平方列表:
squares = list(map(lambda x: x**2, range(10)))
或等價(jià)于:
squares = [x**2 for x in range(10)]
上面這種寫法更簡(jiǎn)潔、易讀。
列表推導(dǎo)式的方括號(hào)內(nèi)包含以下內(nèi)容:一個(gè)表達(dá)式,后面為一個(gè) for
子句,然后,是零個(gè)或多個(gè) for
或 if
子句。結(jié)果是由表達(dá)式依據(jù) for
和 if
子句求值計(jì)算而得出一個(gè)新列表。 舉例來說,以下列表推導(dǎo)式將兩個(gè)列表中不相等的元素組合起來:
>>> [(x, y) for in [1,2,3] for y in [3,1,4] if x != y]
[(1, 3), (1, 4), (2, 3), (2, 1), (2, 4), (3, 1), (3, 4)]
等價(jià)于:
>>> combs = []
>>> for x in [1,2,3]:
... for y in [3,1,4]:
... if x != y:
... combs.append((x, y))
...
>>> combs
[(1, 3), (1, 4), (2, 3), (2, 1), (2, 4), (3, 1), (3, 4)]
表達(dá)式是元組(例如上例的 (x, y)
)時(shí),必須加上括號(hào):
>>> vec = [-4, -2, 0, 2, 4]
>>> # create a new list with the values doubled
>>> [x*2 for x in vec]
[-8, -4, 0, 4, 8]
>>> # filter the list to exclude negative numbers
>>> [x for x in vec if x >= 0]
[0, 2, 4]
>>> # apply a function to all the elements
>>> [abs(x) for x in vec]
[4, 2, 0, 2, 4]
>>> # call a method on each element
>>> freshfruit = [' banana', ' loganberry ', 'passion fruit ']
>>> [weapon.strip() for weapon in freshfruit]
['banana', 'loganberry', 'passion fruit']
>>> # create a list of 2-tuples like (number, square)
>>> [(x, x**2) for x in range(6)]
[(0, 0), (1, 1), (2, 4), (3, 9), (4, 16), (5, 25)]
>>> # the tuple must be parenthesized, otherwise an error is raised
>>> [x, x**2 for x in range(6)]
File "<stdin>", line 1, in <module>
[x, x**2 for x in range(6)]
^
SyntaxError: invalid syntax
>>> # flatten a list using a listcomp with two 'for'
>>> vec = [[1,2,3], [4,5,6], [7,8,9]]
>>> [num for elem in vec for num in elem]
[1, 2, 3, 4, 5, 6, 7, 8, 9]
列表推導(dǎo)式可以使用復(fù)雜的表達(dá)式和嵌套函數(shù):
>>> from math import pi
>>> [str(round(pi, i)) for i in range(1, 6)]
['3.1', '3.14', '3.142', '3.1416', '3.14159']
5.1.4. 嵌套的列表推導(dǎo)式?
列表推導(dǎo)式中的初始表達(dá)式可以是任何表達(dá)式,甚至可以是另一個(gè)列表推導(dǎo)式。
下面這個(gè) 3x4 矩陣,由 3 個(gè)長(zhǎng)度為 4 的列表組成:
>>> matrix = [
... [1, 2, 3, 4],
... [5, 6, 7, 8],
... [9, 10, 11, 12],
... ]
下面的列表推導(dǎo)式可以轉(zhuǎn)置行列:
>>> [[row[i] for row in matrix] for i in range(4)]
[[1, 5, 9], [2, 6, 10], [3, 7, 11], [4, 8, 12]]
As we saw in the previous section, the inner list comprehension is evaluated in
the context of the for
that follows it, so this example is
equivalent to:
>>> transposed = []
>>> for i in range(4):
... transposed.append([row[i] for row in matrix])
...
>>> transposed
[[1, 5, 9], [2, 6, 10], [3, 7, 11], [4, 8, 12]]
反過來說,也等價(jià)于:
>>> transposed = []
>>> for i in range(4):
... # the following 3 lines implement the nested listcomp
... transposed_row = []
... for row in matrix:
... transposed_row.append(row[i])
... transposed.append(transposed_row)
...
>>> transposed
[[1, 5, 9], [2, 6, 10], [3, 7, 11], [4, 8, 12]]
實(shí)際應(yīng)用中,最好用內(nèi)置函數(shù)替代復(fù)雜的流程語句。此時(shí),zip()
函數(shù)更好用:
>>> list(zip(*matrix))
[(1, 5, 9), (2, 6, 10), (3, 7, 11), (4, 8, 12)]
關(guān)于本行中星號(hào)的詳細(xì)說明,參見 解包實(shí)參列表。
5.2. del
語句?
del
語句按索引,而不是值從列表中移除元素。與返回值的 pop()
方法不同, del
語句也可以從列表中移除切片,或清空整個(gè)列表(之前是將空列表賦值給切片)。 例如:
>>> a = [-1, 1, 66.25, 333, 333, 1234.5]
>>> del a[0]
>>> a
[1, 66.25, 333, 333, 1234.5]
>>> del a[2:4]
>>> a
[1, 66.25, 1234.5]
>>> del a[:]
>>> a
[]
del
也可以用來刪除整個(gè)變量:
>>> del a
此后,再引用 a
就會(huì)報(bào)錯(cuò)(直到為它賦與另一個(gè)值)。后文會(huì)介紹 del
的其他用法。
5.3. 元組和序列?
列表和字符串有很多共性,例如,索引和切片操作。這兩種數(shù)據(jù)類型是 序列 (參見 序列類型 --- list, tuple, range)。隨著 Python 語言的發(fā)展,其他的序列類型也被加入其中。本節(jié)介紹另一種標(biāo)準(zhǔn)序列類型:元組。
元組由多個(gè)用逗號(hào)隔開的值組成,例如:
>>> t = 12345, 54321, 'hello!'
>>> t[0]
12345
>>> t
(12345, 54321, 'hello!')
>>> # Tuples may be nested:
... u = t, (1, 2, 3, 4, 5)
>>> u
((12345, 54321, 'hello!'), (1, 2, 3, 4, 5))
>>> # Tuples are immutable:
... t[0] = 88888
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: 'tuple' object does not support item assignment
>>> # but they can contain mutable objects:
... v = ([1, 2, 3], [3, 2, 1])
>>> v
([1, 2, 3], [3, 2, 1])
輸出時(shí),元組都要由圓括號(hào)標(biāo)注,這樣才能正確地解釋嵌套元組。輸入時(shí),圓括號(hào)可有可無,不過經(jīng)常是必須的(如果元組是更大的表達(dá)式的一部分)。不允許為元組中的單個(gè)元素賦值,當(dāng)然,可以創(chuàng)建含列表等可變對(duì)象的元組。
雖然,元組與列表很像,但使用場(chǎng)景不同,用途也不同。元組是 immutable (不可變的),一般可包含異質(zhì)元素序列,通過解包(見本節(jié)下文)或索引訪問(如果是 namedtuples
,可以屬性訪問)。列表是 mutable (可變的),列表元素一般為同質(zhì)類型,可迭代訪問。
構(gòu)造 0 個(gè)或 1 個(gè)元素的元組比較特殊:為了適應(yīng)這種情況,對(duì)句法有一些額外的改變。用一對(duì)空?qǐng)A括號(hào)就可以創(chuàng)建空元組;只有一個(gè)元素的元組可以通過在這個(gè)元素后添加逗號(hào)來構(gòu)建(圓括號(hào)里只有一個(gè)值的話不夠明確)。丑陋,但是有效。例如:
>>> empty = ()
>>> singleton = 'hello', # <-- note trailing comma
>>> len(empty)
0
>>> len(singleton)
1
>>> singleton
('hello',)
語句 t = 12345, 54321, 'hello!'
是 元組打包 的例子:值 12345
, 54321
和 'hello!'
一起被打包進(jìn)元組。逆操作也可以:
>>> x, y, z = t
稱之為 序列解包 也是妥妥的,適用于右側(cè)的任何序列。序列解包時(shí),左側(cè)變量與右側(cè)序列元素的數(shù)量應(yīng)相等。注意,多重賦值其實(shí)只是元組打包和序列解包的組合。
5.4. 集合?
Python 還支持 集合 這種數(shù)據(jù)類型。集合是由不重復(fù)元素組成的無序容器。基本用法包括成員檢測(cè)、消除重復(fù)元素。集合對(duì)象支持合集、交集、差集、對(duì)稱差分等數(shù)學(xué)運(yùn)算。
創(chuàng)建集合用花括號(hào)或 set()
函數(shù)。注意,創(chuàng)建空集合只能用 set()
,不能用 {}
,{}
創(chuàng)建的是空字典,下一小節(jié)介紹數(shù)據(jù)結(jié)構(gòu):字典。
以下是一些簡(jiǎn)單的示例
>>> basket = {'apple', 'orange', 'apple', 'pear', 'orange', 'banana'}
>>> print(basket) # show that duplicates have been removed
{'orange', 'banana', 'pear', 'apple'}
>>> 'orange' in basket # fast membership testing
True
>>> 'crabgrass' in basket
False
>>> # Demonstrate set operations on unique letters from two words
...
>>> a = set('abracadabra')
>>> b = set('alacazam')
>>> a # unique letters in a
{'a', 'r', 'b', 'c', 'd'}
>>> a - b # letters in a but not in b
{'r', 'd', 'b'}
>>> a | b # letters in a or b or both
{'a', 'c', 'r', 'd', 'b', 'm', 'z', 'l'}
>>> a & b # letters in both a and b
{'a', 'c'}
>>> a ^ b # letters in a or b but not both
{'r', 'd', 'b', 'm', 'z', 'l'}
與 列表推導(dǎo)式 類似,集合也支持推導(dǎo)式:
>>> a = {x for x in 'abracadabra' if x not in 'abc'}
>>> a
{'r', 'd'}
5.5. 字典?
字典 (參見 映射類型 --- dict) 也是一種常用的 Python 內(nèi)置數(shù)據(jù)類型。其他語言可能把字典稱為 聯(lián)合內(nèi)存 或 聯(lián)合數(shù)組。與以連續(xù)整數(shù)為索引的序列不同,字典以 關(guān)鍵字 為索引,關(guān)鍵字通常是字符串或數(shù)字,也可以是其他任意不可變類型。只包含字符串、數(shù)字、元組的元組,也可以用作關(guān)鍵字。但如果元組直接或間接地包含了可變對(duì)象,就不能用作關(guān)鍵字。列表不能當(dāng)關(guān)鍵字,因?yàn)榱斜砜梢杂盟饕⑶衅?code class="xref py py-meth docutils literal notranslate">append() 、extend()
等方法修改。
可以把字典理解為 鍵值對(duì) 的集合,但字典的鍵必須是唯一的?;ɡㄌ?hào) {}
用于創(chuàng)建空字典。另一種初始化字典的方式是,在花括號(hào)里輸入逗號(hào)分隔的鍵值對(duì),這也是字典的輸出方式。
字典的主要用途是通過關(guān)鍵字存儲(chǔ)、提取值。用 del
可以刪除鍵值對(duì)。用已存在的關(guān)鍵字存儲(chǔ)值,與該關(guān)鍵字關(guān)聯(lián)的舊值會(huì)被取代。通過不存在的鍵提取值,則會(huì)報(bào)錯(cuò)。
對(duì)字典執(zhí)行 list(d)
操作,返回該字典中所有鍵的列表,按插入次序排列(如需排序,請(qǐng)使用 sorted(d)
)。檢查字典里是否存在某個(gè)鍵,使用關(guān)鍵字 in
。
以下是一些字典的簡(jiǎn)單示例:
>>> tel = {'jack': 4098, 'sape': 4139}
>>> tel['guido'] = 4127
>>> tel
{'jack': 4098, 'sape': 4139, 'guido': 4127}
>>> tel['jack']
4098
>>> del tel['sape']
>>> tel['irv'] = 4127
>>> tel
{'jack': 4098, 'guido': 4127, 'irv': 4127}
>>> list(tel)
['jack', 'guido', 'irv']
>>> sorted(tel)
['guido', 'irv', 'jack']
>>> 'guido' in tel
True
>>> 'jack' not in tel
False
dict()
構(gòu)造函數(shù)可以直接用鍵值對(duì)序列創(chuàng)建字典:
>>> dict([('sape', 4139), ('guido', 4127), ('jack', 4098)])
{'sape': 4139, 'guido': 4127, 'jack': 4098}
字典推導(dǎo)式可以用任意鍵值表達(dá)式創(chuàng)建字典:
>>> {x: x**2 for x in (2, 4, 6)}
{2: 4, 4: 16, 6: 36}
關(guān)鍵字是比較簡(jiǎn)單的字符串時(shí),直接用關(guān)鍵字參數(shù)指定鍵值對(duì)更便捷:
>>> dict(sape=4139, guido=4127, jack=4098)
{'sape': 4139, 'guido': 4127, 'jack': 4098}
5.6. 循環(huán)的技巧?
在字典中循環(huán)時(shí),用 items()
方法可同時(shí)取出鍵和對(duì)應(yīng)的值:
>>> knights = {'gallahad': 'the pure', 'robin': 'the brave'}
>>> for k, v in knights.items():
... print(k, v)
...
gallahad the pure
robin the brave
在序列中循環(huán)時(shí),用 enumerate()
函數(shù)可以同時(shí)取出位置索引和對(duì)應(yīng)的值:
>>> for i, v in enumerate(['tic', 'tac', 'toe']):
... print(i, v)
...
0 tic
1 tac
2 toe
同時(shí)循環(huán)兩個(gè)或多個(gè)序列時(shí),用 zip()
函數(shù)可以將其內(nèi)的元素一一匹配:
>>> questions = ['name', 'quest', 'favorite color']
>>> answers = ['lancelot', 'the holy grail', 'blue']
>>> for q, a in zip(questions, answers):
... print('What is your {0}? It is {1}.'.format(q, a))
...
What is your name? It is lancelot.
What is your quest? It is the holy grail.
What is your favorite color? It is blue.
逆向循環(huán)序列時(shí),先正向定位序列,然后調(diào)用 reversed()
函數(shù):
>>> for i in reversed(range(1, 10, 2)):
... print(i)
...
9
7
5
3
1
按指定順序循環(huán)序列,可以用 sorted()
函數(shù),在不改動(dòng)原序列的基礎(chǔ)上,返回一個(gè)重新的序列:
>>> basket = ['apple', 'orange', 'apple', 'pear', 'orange', 'banana']
>>> for i in sorted(basket):
... print(i)
...
apple
apple
banana
orange
orange
pear
使用 set()
去除序列中的重復(fù)元素。使用 sorted()
加 set()
則按排序后的順序,循環(huán)遍歷序列中的唯一元素:
>>> basket = ['apple', 'orange', 'apple', 'pear', 'orange', 'banana']
>>> for f in sorted(set(basket)):
... print(f)
...
apple
banana
orange
pear
一般來說,在循環(huán)中修改列表的內(nèi)容時(shí),創(chuàng)建新列表比較簡(jiǎn)單,且安全:
>>> import math
>>> raw_data = [56.2, float('NaN'), 51.7, 55.3, 52.5, float('NaN'), 47.8]
>>> filtered_data = []
>>> for value in raw_data:
... if not math.isnan(value):
... filtered_data.append(value)
...
>>> filtered_data
[56.2, 51.7, 55.3, 52.5, 47.8]
5.7. 深入條件控制?
while
和 if
條件句不只可以進(jìn)行比較,還可以使用任意運(yùn)算符。
The comparison operators in
and not in
are membership tests that
determine whether a value is in (or not in) a container. The operators is
and is not
compare whether two objects are really the same object. All
comparison operators have the same priority, which is lower than that of all
numerical operators.
比較操作支持鏈?zhǔn)讲僮鳌@纾?code class="docutils literal notranslate">a < b == c 校驗(yàn) a
是否小于 b
,且 b
是否等于 c
。
比較操作可以用布爾運(yùn)算符 and
和 or
組合,并且,比較操作(或其他布爾運(yùn)算)的結(jié)果都可以用 not
取反。這些操作符的優(yōu)先級(jí)低于比較操作符;not
的優(yōu)先級(jí)最高, or
的優(yōu)先級(jí)最低,因此,A and not B or C
等價(jià)于 (A and (not B)) or C
。與其他運(yùn)算符操作一樣,此處也可以用圓括號(hào)表示想要的組合。
布爾運(yùn)算符 and
和 or
也稱為 短路 運(yùn)算符:其參數(shù)從左至右解析,一旦可以確定結(jié)果,解析就會(huì)停止。例如,如果 A
和 C
為真,B
為假,那么 A and B and C
不會(huì)解析 C
。用作普通值而不是布爾值時(shí),短路操作符返回的值通常是最后一個(gè)變量。
還可以把比較操作或邏輯表達(dá)式的結(jié)果賦值給變量,例如:
>>> string1, string2, string3 = '', 'Trondheim', 'Hammer Dance'
>>> non_null = string1 or string2 or string3
>>> non_null
'Trondheim'
注意,Python 與 C 不同,在表達(dá)式內(nèi)部賦值必須顯式使用 海象運(yùn)算符 :=
。 這避免了 C 程序中常見的問題:要在表達(dá)式中寫 ==
時(shí),卻寫成了 =
。
5.8. 序列和其他類型的比較?
序列對(duì)象可以與相同序列類型的其他對(duì)象比較。這種比較使用 字典式 順序:首先,比較前兩個(gè)對(duì)應(yīng)元素,如果不相等,則可確定比較結(jié)果;如果相等,則比較之后的兩個(gè)元素,以此類推,直到其中一個(gè)序列結(jié)束。如果要比較的兩個(gè)元素本身是相同類型的序列,則遞歸地執(zhí)行字典式順序比較。如果兩個(gè)序列中所有的對(duì)應(yīng)元素都相等,則兩個(gè)序列相等。如果一個(gè)序列是另一個(gè)的初始子序列,則較短的序列可被視為較?。ㄝ^少)的序列。 對(duì)于字符串來說,字典式順序使用 Unicode 碼位序號(hào)排序單個(gè)字符。下面列出了一些比較相同類型序列的例子:
(1, 2, 3) < (1, 2, 4)
[1, 2, 3] < [1, 2, 4]
'ABC' < 'C' < 'Pascal' < 'Python'
(1, 2, 3, 4) < (1, 2, 4)
(1, 2) < (1, 2, -1)
(1, 2, 3) == (1.0, 2.0, 3.0)
(1, 2, ('aa', 'ab')) < (1, 2, ('abc', 'a'), 4)
注意,對(duì)不同類型的對(duì)象來說,只要待比較的對(duì)象提供了合適的比較方法,就可以使用 <
和 >
進(jìn)行比較。例如,混合數(shù)值類型通過數(shù)值進(jìn)行比較,所以,0 等于 0.0,等等。否則,解釋器不會(huì)隨便給出一個(gè)對(duì)比結(jié)果,而是觸發(fā) TypeError
異常。
備注
- 1
別的語言可能會(huì)返回可變對(duì)象,允許方法連續(xù)執(zhí)行,例如,
d->insert("a")->remove("b")->sort();
。