成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Python編程中3個常用的數據結構和算法

開發 后端 算法
本篇文章將介紹3種常見的數據結構和同數據有關的算法。此外,在collections模塊中也包含了針對各種數據結構的解決方案。

Python內置了許多非常有用的數據結構,比如列表(list)、集合(set)以及字典(dictionary)。就絕大部分情況而言,我們可以直接使用這些數據結構。但是,通常我們還需要考慮比如搜索、排序、排列以及篩選等這一類常見的問題。

本篇文章將介紹3種常見的數據結構和同數據有關的算法。此外,在collections模塊中也包含了針對各種數據結構的解決方案。

[[262323]]

1. 將序列分解為單獨的變量

(1) 問題

我們有一個包含 N 個元素的元組或序列,現在想將它分解為N個單獨的變量。

(2) 解決方案

任何序列(或可迭代的對象)都可以通過一個簡單的賦值操作來分解為單獨的變量。一個要求是變量的總數和結構要與序列相吻合。例如:

  1. >>> p = (4, 5) 
  2. >>> x, y = p 
  3. >>> x 
  4. >>> y 
  5. >>> 
  6. >>> data = [ 'ACME', 50, 91.1, (2012, 12, 21) ] 
  7. >>> name, shares, price, date = data 
  8. >>> name 
  9. 'ACME' 
  10. >>> date 
  11. (2012, 12, 21) 
  12. >>> name, shares, price, (year, mon, day) = data 
  13. >>> name 
  14. 'ACME' 
  15. >>> year 
  16. 2012 
  17. >>> mon 
  18. 12 
  19. >>> day 
  20. 21 
  21. >>> 

如果元素的數量不匹配,將得到一個錯誤提示。例如:

  1. >>> p = (4, 5) 
  2. >>> x, y, z = p 
  3. Traceback (most recent call last): 
  4.  File "<stdin>", line 1, in <module> 
  5. ValueError: need more than 2 values to unpack 
  6. >>> 

(3) 討論

實際上不僅僅只是元組或列表,只要對象恰好是可迭代的,那么就可以執行分解操作。這包括字符串、文件、迭代器以及生成器。比如:

  1. >>> s = 'Hello' 
  2. >>> a, b, c, d, e = s 
  3. >>> a 
  4. 'H' 
  5. >>> b 
  6. 'e' 
  7. >>> e 
  8. 'o' 
  9. >>> 

當做分解操作時,有時候可能想丟棄某些特定的值。Python并沒有提供特殊的語法來實現這一點,但是通常可以選一個用不到的變量名,以此來作為要丟棄的值的名稱。例如:

  1. >>> data = [ 'ACME', 50, 91.1, (2012, 12, 21) ] 
  2. >>> _, shares, price, _ = data 
  3. >>> shares 
  4. 50 
  5. >>> price 
  6. 91.1 
  7. >>> 

但是請確保選擇的變量名沒有在其他地方用到過。

2. 從任意長度的可迭代對象中分解元素

(1) 問題

需要從某個可迭代對象中分解出N個元素,但是這個可迭代對象的長度可能超過N,這會導致出現“分解的值過多(too many values to unpack)”的異常。

(2) 解決方案

Python的“*表達式”可以用來解決這個問題。例如,假設開設了一門課程,并決定在期末的作業成績中去掉第一個和最后一個,只對中間剩下的成績做平均分統計。如果只有4個成績,也許可以簡單地將4個都分解出來,但是如果有24個呢?*表達式使這一切都變得簡單:

  1. def drop_first_last(grades): 
  2.  first, *middle, last = grades 
  3.  return avg(middle) 

另一個用例是假設有一些用戶記錄,記錄由姓名和電子郵件地址組成,后面跟著任意數量的電話號碼。則可以像這樣分解記錄:

  1. >>> record = ('Dave', 'dave@example.com', '773-555-1212', '847-555-1212') 
  2. >>> name, email, *phone_numbers = user_record 
  3. >>> name 
  4. 'Dave' 
  5. >>> email 
  6. 'dave@example.com' 
  7. >>> phone_numbers 
  8. ['773-555-1212', '847-555-1212'] 
  9. >>> 

不管需要分解出多少個電話號碼(甚至沒有電話號碼),變量phone_numbers都一直是列表,而這是毫無意義的。如此一來,對于任何用到了變量phone_numbers的代碼都不必對它可能不是一個列表的情況負責,或者額外做任何形式的類型檢查。

由*修飾的變量也可以位于列表的第一個位置。例如,比方說用一系列的值來代表公司過去8個季度的銷售額。如果想對最近一個季度的銷售額同前7個季度的平均值做比較,可以這么做:

  1. *trailing_qtrs, current_qtr = sales_record 
  2. trailing_avg = sum(trailing_qtrs) / len(trailing_qtrs) 
  3. return avg_comparison(trailing_avg, current_qtr) 

從Python解釋器的角度來看,這個操作是這樣的:

  1. >>> *trailing, current = [10, 8, 7, 1, 9, 5, 10, 3] 
  2. >>> trailing 
  3. [10, 8, 7, 1, 9, 5, 10] 
  4. >>> current 

(3) 討論

對于分解未知或任意長度的可迭代對象,這種擴展的分解操作可謂是量身定做的工具。通常,這類可迭代對象中會有一些已知的組件或模式(例如,元素1之后的所有內容都是電話號碼),利用*表達式分解可迭代對象使得開發者能夠輕松利用這些模式,而不必在可迭代對象中做復雜花哨的操作才能得到相關的元素。

*式的語法在迭代一個變長的元組序列時尤其有用。例如,假設有一個帶標記的元組序列:

  1. records = [ 
  2.  ('foo', 1, 2), 
  3.  ('bar', 'hello'), 
  4.  ('foo', 3, 4), 
  5. def do_foo(x, y): 
  6.  print('foo', x, y) 
  7. def do_bar(s): 
  8.  print('bar', s) 
  9. for tag, *args in records: 
  10.  if tag == 'foo': 
  11.  do_foo(*args) 
  12. elif tag == 'bar': 
  13.  do_bar(*args) 

當和某些特定的字符串處理操作相結合,比如做拆分(splitting)操作時,這種*式的語法所支持的分解操作也非常有用。例如:

  1. >>> line = 'nobody:*:-2:-2:Unprivileged User:/var/empty:/usr/bin/false' 
  2. >>> uname, *fields, homedir, sh = line.split(':') 
  3. >>> uname 
  4. 'nobody' 
  5. >>> homedir 
  6. '/var/empty' 
  7. >>> sh 
  8. '/usr/bin/false' 
  9. >>> 

有時候可能想分解出某些值然后丟棄它們。在分解的時候,不能只是指定一個單獨的*,但是可以使用幾個常用來表示待丟棄值的變量名,比如_或者ign(ignored)。例如:

  1. >>> record = ('ACME', 50, 123.45, (12, 18, 2012)) 
  2. >>> name, *_, (*_, year) = record 
  3. >>> name 
  4. 'ACME' 
  5. >>> year 
  6. 2012 
  7. >>> 

*分解操作和各種函數式語言中的列表處理功能有著一定的相似性。例如,如果有一個列表,可以像下面這樣輕松將其分解為頭部和尾部:

  1. >>> items = [1, 10, 7, 4, 5, 9] 
  2. >>> head, *tail = items 
  3. >>> head 
  4. >>> tail 
  5. [10, 7, 4, 5, 9] 
  6. >>> 

在編寫執行這類拆分功能的函數時,人們可以假設這是為了實現某種精巧的遞歸算法。例如:

  1. >>> def sum(items): 
  2. ... head, *tail = items 
  3. ... return head + sum(tail) if tail else head 
  4. ... 
  5. >>> sum(items) 
  6. 36 
  7. >>> 

但是請注意,遞歸真的不算是Python的強項,這是因為其內在的遞歸限制所致。因此,最后一個例子在實踐中沒太大的意義,只不過是一點學術上的好奇罷了。

3. 保存最后N個元素

(1) 問題

我們希望在迭代或是其他形式的處理過程中對最后幾項記錄做一個有限的歷史記錄統計。

(2) 解決方案

保存有限的歷史記錄可算是collections.deque的應用場景了。例如,下面的代碼對一系列文本行做簡單的文本匹配操作,當發現有匹配時就輸出當前的匹配行以及最后檢查過的N行文本。

  1. from collections import deque 
  2. def search(lines, pattern, history=5): 
  3.  previous_lines = deque(maxlen=history
  4.  for line in lines: 
  5.  if pattern in line: 
  6.  yield line, previous_lines 
  7.  previous_lines.append(line) 
  8. # Example use on a file 
  9. if __name__ == '__main__': 
  10.  with open('somefile.txt') as f: 
  11.  for line, prevlines in search(f, 'python', 5): 
  12.  for pline in prevlines: 
  13.  print(pline, end=''
  14.  print(line, end=''
  15.  print('-'*20) 

(3) 討論

如同上面的代碼片段中所做的一樣,當編寫搜索某項記錄的代碼時,通常會用到含有yield關鍵字的生成器函數。這將處理搜索過程的代碼和使用搜索結果的代碼成功解耦開來。如果對生成器還不熟悉,請參見4.3節。

deque(maxlen=N)創建了一個固定長度的隊列。當有新記錄加入而隊列已滿時會自動移除最老的那條記錄。例如:

  1. >>> q = deque(maxlen=3
  2. >>> q.append(1) 
  3. >>> q.append(2) 
  4. >>> q.append(3) 
  5. >>> q 
  6. deque([1, 2, 3], maxlen=3
  7. >>> q.append(4) 
  8. >>> q 
  9. deque([2, 3, 4], maxlen=3
  10. >>> q.append(5) 
  11. >>> q 
  12. deque([3, 4, 5], maxlen=3

盡管可以在列表上手動完成這樣的操作(append、del),但隊列這種解決方案要優雅得多,運行速度也快得多。

更普遍的是,當需要一個簡單的隊列結構時,deque可祝你一臂之力。如果不指定隊列的大小,也就得到了一個無界限的隊列,可以在兩端執行添加和彈出操作,例如:

  1. >>> q = deque() 
  2. >>> q.append(1) 
  3. >>> q.append(2) 
  4. >>> q.append(3) 
  5. >>> q 
  6. deque([1, 2, 3]) 
  7. >>> q.appendleft(4) 
  8. >>> q 
  9. deque([4, 1, 2, 3]) 
  10. >>> q.pop() 
  11. >>> q 
  12. deque([4, 1, 2]) 
  13. >>> q.popleft() 

從隊列兩端添加或彈出元素的復雜度都是O(1)。這和列表不同,當從列表的頭部插入或移除元素時,列表的復雜度為O(N)。

 

責任編輯:趙寧寧 來源: 今日頭條
相關推薦

2023-04-27 09:13:20

排序算法數據結構

2022-02-22 15:27:46

數據結構容器算法

2021-10-07 09:04:49

Collections數據結構

2021-05-12 09:07:09

Java數據結構算法

2012-04-28 14:21:47

Java數據結構線性結構

2020-08-12 08:30:20

數據結構算法

2021-03-09 06:30:32

JAVA數據結構算法

2021-03-18 08:44:20

Java數據結構算法

2021-04-13 09:37:41

Java數據結構算法

2023-09-21 16:13:20

Python數據結構

2009-08-13 18:34:49

C#數據結構和算法

2021-01-28 07:33:34

JavaScript鏈表數據

2011-07-20 17:10:54

C++

2021-06-08 10:41:00

Go語言算法

2019-05-21 14:28:35

代碼算法編程

2023-10-26 09:17:48

算法編程

2023-09-25 12:23:18

Python

2019-06-10 14:45:26

面試數據結構算法

2021-03-08 06:28:57

JAVA數據結構與算法稀疏數組

2021-03-10 08:42:19

Java數據結構算法
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲精品一区二区在线观看 | 久久99精品国产99久久6男男 | 成人欧美一区二区三区黑人孕妇 | gogo肉体亚洲高清在线视 | 国产一区二区三区免费 | 婷婷色成人 | 中文字幕日韩一区二区 | 激情av网站 | 成人在线网 | 亚洲免费精品 | 久久久成人精品 | 国产精品久久久久久中文字 | 久久精品亚洲欧美日韩久久 | 日韩国产在线 | 国产一级视频在线播放 | 欧美夜夜 | 欧美激情亚洲天堂 | 日韩成人在线免费观看 | 热re99久久精品国99热观看 | 成人免费一区二区 | 手机av在线 | 欧美在线视频一区 | 91伦理片 | 熟女毛片| 日本一道本视频 | 亚洲成色777777在线观看影院 | 狠狠躁天天躁夜夜躁婷婷老牛影视 | 成人av播放| 91日b| 欧美福利精品 | 人人看人人干 | 国产伦精品一区二区三区照片91 | 午夜精品久久久久久 | 日本特黄a级高清免费大片 国产精品久久性 | 一级片免费在线观看 | 手机看片在线播放 | 亚洲一区二区网站 | 一区二区三区中文 | 久久久久国产精品一区 | 精品视频在线一区 | 全部免费毛片在线播放网站 |