[轉]Python十個高大上的語法

来源:https://www.cnblogs.com/nihao2020/archive/2019/12/07/12000241.html
-Advertisement-
Play Games

Python 是一種代表簡單思想的語言,其語法相對簡單,很容易上手。不過,如果就此小視 Python 語法的精妙和深邃,那就大錯特錯了。本文精心篩選了最能展現 Python 語法之精妙的十個知識點,並附上詳細的實例代碼。如能在實戰中融會貫通、靈活使用,必將使代碼更為精煉、高效,同時也會極大提升代碼B ...


Python 是一種代表簡單思想的語言,其語法相對簡單,很容易上手。不過,如果就此小視 Python 語法的精妙和深邃,那就大錯特錯了。本文精心篩選了最能展現 Python 語法之精妙的十個知識點,並附上詳細的實例代碼。如能在實戰中融會貫通、靈活使用,必將使代碼更為精煉、高效,同時也會極大提升代碼B格,使之看上去更老練,讀起來更優雅。

1. for - else

什麼?不是 if 和 else 才是原配嗎?No,你可能不知道,else 是個腳踩兩隻船的家伙,for 和 else 也是一對,而且是合法的。十大裝B語法,for-else 絕對算得上南無灣!不信,請看:

>>> for i in [1,2,3,4]:
print(i)
else:
print(i, '我是else')

1
2
3
4
4 我是else
1
2
3
4
5
6
7
8
9
10
如果在 for 和 else 之間(迴圈體內)有第三者 if 插足,也不會影響 for 和 else 的關係。因為 for 的級別比 if 高,else 又是一個攀附權貴的家伙,根本不在乎是否有 if,以及是否執行了滿足 if 條件的語句。else 的眼裡只有 for,只要 for 順利執行完畢,else 就會屁顛兒屁顛兒地跑一遍:

>>> for i in [1,2,3,4]:
if i > 2:
print(i)
else:
print(i, '我是else')

3
4
4 我是else
1
2
3
4
5
6
7
8
9
那麼,如何拆散 for 和 else 這對冤家呢?只有當 for 迴圈被 break 語句中斷之後,才會跳過 else 語句:

>>> for i in [1,2,3,4]:
if i>2:
print(i)
break
else:
print(i, '我是else')

3
1
2
3
4
5
6
7
8
2. 一顆星(*)和兩顆星(**)

有沒有發現,星(*)真是一個神奇的符號!想一想,沒有它,C語言還有啥好玩的?同樣,因為有它,Python 才會如此的儀態萬方、風姿綽約、楚楚動人!Python 函數支持預設參數和可變參數,一顆星表示不限數量的單值參數,兩顆星表示不限數量的鍵值對參數。

我們還是舉例說明吧:設計一個函數,返回多個輸入數值的和。我們固然可以把這些輸入數值做成一個list傳給函數,但這個方法,遠沒有使用一顆星的可變參數來得優雅:

>>> def multi_sum(*args):
s = 0
for item in args:
s += item
return s

>>> multi_sum(3,4,5)
12
1
2
3
4
5
6
7
8
Python 函數允許同時全部或部分使用固定參數、預設參數、單值(一顆星)可變參數、鍵值對(兩顆星)可變參數,使用時必須按照前述順序書寫。

>>> def do_something(name, age, gender='男', *args, **kwds):
print('姓名:%s,年齡:%d,性別:%s'%(name, age, gender))
print(args)
print(kwds)

>>> do_something('xufive', 50, '男', 175, 75, math=99, english=90)
姓名:xufive,年齡:50,性別:男
(175, 75)
{'math': 99, 'english': 90}
1
2
3
4
5
6
7
8
9
此外,一顆星和兩顆星還可用於列表、元組、字典的解包,看起來更像C語言:

>>> a = (1,2,3)
>>> print(a)
(1, 2, 3)
>>> print(*a)
1 2 3
>>> b = [1,2,3]
>>> print(b)
[1, 2, 3]
>>> print(*b)
1 2 3
>>> c = {'name':'xufive', 'age':51}
>>> print(c)
{'name': 'xufive', 'age': 51}
>>> print(*c)
name age
>>> print('name:{name}, age:{age}'.format(**c))
name:xufive, age:51
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
3. 三元表達式

熟悉 C/C++ 的程式員,初上手 python 時,一定會懷念經典的三元操作符,因為想表達同樣的思想,用python 寫起來似乎更麻煩。比如:

>>> y = 5
>>> if y < 0:
print('y是一個負數')
else:
print('y是一個非負數')

y是一個非負數
1
2
3
4
5
6
7
其實,python 是支持三元表達式的,只是稍微怪異了一點,類似於我們山東人講話。比如,山東人最喜歡用倒裝句:打球去吧,要是不下雨的話;下雨,咱就去自習室。翻譯成三元表達式就是:

打球去吧 if 不下雨 else 去自習室
來看看三元表達式具體的使用:

>>> y = 5
>>> print('y是一個負數' if y < 0 else 'y是一個非負數')
y是一個非負數
1
2
3
python 的三元表達式也可以用來賦值:

>>> y = 5
>>> x = -1 if y < 0 else 1
>>> x
1
1
2
3
4
4. with - as

with 這個詞兒,英文裡面不難翻譯,但在 Python 語法中怎麼翻譯,我還真想不出來,大致上是一種上下文管理協議。作為初學者,不用關註 with 的各種方法以及機制如何,只需要瞭解它的應用場景就可以了。with 語句適合一些事先需要準備,事後需要處理的任務,比如,文件操作,需要先打開文件,操作完成後需要關閉文件。如果不使用with,文件操作通常得這樣:

fp = open(r"D:\CSDN\Column\temp\mpmap.py", 'r')
try:
contents = fp.readlines()
finally:
fp.close()
1
2
3
4
5
如果使用 with - as,那就優雅多了:

>>> with open(r"D:\CSDN\Column\temp\mpmap.py", 'r') as fp:
contents = fp.readlines()
1
2
5. 列表推導式

在各種稀奇古怪的語法中,列表推導式的使用頻率應該時最高的,對於代碼的簡化效果也非常明顯。比如,求列表各元素的平方,通常應該這樣寫(當然也有其他寫法,比如使用map函數):

>>> a = [1, 2, 3, 4, 5]
>>> result = list()
>>> for i in a:
result.append(i*i)

>>> result
[1, 4, 9, 16, 25]
1
2
3
4
5
6
7
如果使用列表推導式,看起來就舒服多了:

>>> a = [1, 2, 3, 4, 5]
>>> result = [i*i for i in a]
>>> result
[1, 4, 9, 16, 25]
1
2
3
4
事實上,推導式不僅支持列表,也支持字典、集合、元組等對象。有興趣的話,可以自行研究。我有一篇博文《一行 Python 代碼能實現什麼喪心病狂的功能?》,裡面的例子,都是列表推導式實現的。

6. 列表索引的各種騷操作

Python 引入負整數作為數組的索引,這絕對是喜大普奔之舉。想想看,在C/C++中,想要數組最後一個元素,得先取得數組長度,減一之後做索引,嚴重影響了思維的連貫性。Python語言之所以獲得成功,我個人覺得,在諸多因素裡面,列表操作的便捷性是不容忽視的一點。請看:

>>> a = [0, 1, 2, 3, 4, 5]
>>> a[2:4]
[2, 3]
>>> a[3:]
[3, 4, 5]
>>> a[1:]
[1, 2, 3, 4, 5]
>>> a[:]
[0, 1, 2, 3, 4, 5]
>>> a[::2]
[0, 2, 4]
>>> a[1::2]
[1, 3, 5]
>>> a[-1]
5
>>> a[-2]
4
>>> a[1:-1]
[1, 2, 3, 4]
>>> a[::-1]
[5, 4, 3, 2, 1, 0]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
如果說,這些你都很熟悉,也經常用,那麼接下來這個用法,你一定會感覺很神奇:

>>> a = [0, 1, 2, 3, 4, 5]
>>> b = ['a', 'b']
>>> a[2:2] = b
>>> a
[0, 1, 'a', 'b', 2, 3, 4, 5]
>>> a[3:6] = b
>>> a
[0, 1, 'a', 'a', 'b', 4, 5]
1
2
3
4
5
6
7
8
7. lambda函數

lambda 聽起來很高大上,其實就是匿名函數(瞭解js的同學一定很熟悉匿名函數)。匿名函數的應用場景是什麼呢?就是僅在定義匿名函數的地方使用這個函數,其他地方用不到,所以就不需要給它取個阿貓阿狗之類的名字了。下麵是一個求和的匿名函數,輸入參數有兩個,x和y,函數體就是x+y,省略了return關鍵字。

>>> lambda x,y: x+y
<function <lambda> at 0x000001B2DE5BD598>
>>> (lambda x,y: x+y)(3,4) # 因為匿名函數沒有名字,使用的時候要用括弧把它包起來
1
2
3
匿名函數一般不會單獨使用,而是配合其他方法,為其他方法提供內置的演算法或判斷條件。比如,使用排序函數sorted對多維數組或者字典排序時,就可以指定排序規則。

>>> a = [{'name':'B', 'age':50}, {'name':'A', 'age':30}, {'name':'C', 'age':40}]
>>> sorted(a, key=lambda x:x['name']) # 按姓名排序
[{'name': 'A', 'age': 30}, {'name': 'B', 'age': 50}, {'name': 'C', 'age': 40}]
>>> sorted(a, key=lambda x:x['age']) # 按年齡排序
[{'name': 'A', 'age': 30}, {'name': 'C', 'age': 40}, {'name': 'B', 'age': 50}]
1
2
3
4
5
再舉一個數組元素求平方的例子,這次用map函數:

>>> a = [1,2,3]
>>> for item in map(lambda x:x*x, a):
print(item, end=', ')

1, 4, 9,
1
2
3
4
5
8. yield 以及生成器和迭代器

yield 這詞兒,真不好翻譯,翻詞典也沒用。我乾脆就讀作“一愛得”,算是外來辭彙吧。要理解 yield,得先瞭解 generator(生成器)。要瞭解generator,得先知道 iterator(迭代器)。哈哈哈,繞暈了吧?算了,我還是說白話吧。

話說py2時代,range()返回的是list,但如果range(10000000)的話,會消耗大量記憶體資源,所以,py2又搞了一個xrange()來解決這個問題。py3則只保留了xrange(),但寫作range()。xrange()返回的就是一個迭代器,它可以像list那樣被遍歷,但又不占用多少記憶體。generator(生成器)是一種特殊的迭代器,只能被遍歷一次,遍歷結束,就自動消失了。總之,不管是迭代器還是生成器,都是為了避免使用list,從而節省記憶體。那麼,如何得到迭代器和生成器呢?

python內置了迭代函數 iter,用於生成迭代器,用法如下:

>>> a = [1,2,3]
>>> a_iter = iter(a)
>>> a_iter
<list_iterator object at 0x000001B2DE434BA8>
>>> for i in a_iter:
print(i, end=', ')

1, 2, 3,
1
2
3
4
5
6
7
8
yield 則是用於構造生成器的。比如,我們要寫一個函數,返回從0到某正整數的所有整數的平方,傳統的代碼寫法是這樣的:

>>> def get_square(n):
result = list()
for i in range(n):
result.append(pow(i,2))
return result

>>> print(get_square(5))
[0, 1, 4, 9, 16]
1
2
3
4
5
6
7
8
但是如果計算1億以內的所有整數的平方,這個函數的記憶體開銷會非常大,這是 yield 就可以大顯身手了:

>>> def get_square(n):
for i in range(n):
yield(pow(i,2))

>>> a = get_square(5)
>>> a
<generator object get_square at 0x000001B2DE5CACF0>
>>> for i in a:
print(i, end=', ')

0, 1, 4, 9, 16,
1
2
3
4
5
6
7
8
9
10
11
如果再次遍歷,則不會有輸出了。

9. 裝飾器

剛弄明白迭代器和生成器,這又來個裝飾器,Python 咋這麼多器呢?的確,Python 為我們提供了很多的武器,裝飾器就是最有力的武器之一。裝飾器很強大,我在這裡嘗試從需求的角度,用一個簡單的例子,說明裝飾器的使用方法和製造工藝。

假如我們需要定義很多個函數,在每個函數運行的時候要顯示這個函數的運行時長,解決方案有很多。比如,可以在調用每個函數之前讀一下時間戳,每個函數運行結束後再讀一下時間戳,求差即可;也可以在每個函數體內的開始和結束位置上讀時間戳,最後求差。不過,這兩個方法,都沒有使用裝飾器那麼簡單、優雅。下麵的例子,很好地展示了這一點。

>>> import time
>>> def timer(func):
def wrapper(*args,**kwds):
t0 = time.time()
func(*args,**kwds)
t1 = time.time()
print('耗時%0.3f'%(t1-t0,))
return wrapper

>>> @timer
def do_something(delay):
print('函數do_something開始')
time.sleep(delay)
print('函數do_something結束')


>>> do_something(3)
函數do_something開始
函數do_something結束
耗時3.077
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
timer() 是我們定義的裝飾器函數,使用@把它附加在任何一個函數(比如do_something)定義之前,就等於把新定義的函數,當成了裝飾器函數的輸入參數。運行 do_something() 函數,可以理解為執行了timer(do_something) 。細節雖然複雜,不過這麼理解不會偏差太大,且更易於把握裝飾器的製造和使用。

10. 巧用斷言assert

所謂斷言,就是聲明表達式的布爾值必須為真的判定,否則將觸發 AssertionError 異常。嚴格來講,assert是調試手段,不宜使用在生產環境中,但這不影響我們用斷言來實現一些特定功能,比如,輸入參數的格式、類型驗證等。

>>> def i_want_to_sleep(delay):
assert(isinstance(delay, (int,float))), '函數參數必須為整數或浮點數'
print('開始睡覺')
time.sleep(delay)
print('睡醒了')


>>> i_want_to_sleep(1.1)
開始睡覺
睡醒了
>>> i_want_to_sleep(2)
開始睡覺
睡醒了
>>> i_want_to_sleep('2')
Traceback (most recent call last):
File "<pyshell#247>", line 1, in <module>
i_want_to_sleep('2')
File "<pyshell#244>", line 2, in i_want_to_sleep
assert(isinstance(delay, (int,float))), '函數參數必須為整數或浮點數'
AssertionError: 函數參數必須為整數或浮點數
————————轉自csdn天元浪子---


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 根據 CSS Scroll Snap Module Level 1 規範,CSS 新增了一批能夠控制滾動的屬性,讓滾動能夠在僅僅通過 CSS 的控制下,得到許多原本需要 JS 腳本介入才能實現的美好交互。 Tips:本文截的一些 Gif 圖涉及容器滾動,效果不是很好,可以點進 Demo 里實際感受下 ...
  • 點擊操作 節點的方式 案例 案例1:點擊按鈕,設置p變色 節點的方式做 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>title</title> <style> div { width: 200px; he ...
  • 介紹 "Ayer" 是一個乾凈且優雅的Hexo主題,自帶響應式,載入速度很快,該有的功能都有,可配置項也很多,非常適合作為你的博客主題,主題內還附送了6張精美的高清壁紙。歡迎使用和Star支持,如果你在使用過程中有任何疑問或者建議,歡迎聯繫我! "項目地址" "效果預覽" "中文說明" 特性 乾凈且 ...
  • 移動京東項目練習記錄 1. 流式佈局與移動端適配的基本要求 2. 綁定事件的幾個方式 on 只能綁定一次事件,但是可以相容各種瀏覽器 addeventlistener 適合各種谷歌相關的瀏覽器,可以綁定多次事件 attachEvent 適合IE瀏覽器,可以多次綁定 但是裡面事件名有on eg:tou ...
  • 回顧概念 文檔: document 元素: 頁面中所有的標簽, 元素 element, 標簽 元素 對象 節點: 頁面中所有的內容(標簽,屬性,文本(文字,換行,空格,回車)) Node 根元素:html標簽 需求 用之前學習的知識點能否解決?獲取div中所有的標簽,設置每個標簽的背景顏色 節點 任 ...
  • part3課程介紹 節點 為什麼要學節點 節點的操作的相關屬性 >作用 12行代碼 有用的 熟練的問題, 節點的案例 元素的創建三種方式 重點的內容 為什麼要有元素的創建 添加元素 移除元素 事件的 part2複習 阻止超鏈接預設的跳轉事件: return false; 獲取元素的方式 根據id獲取 ...
  • 簡潔清爽的代碼風格應該是大多數工程師所期待的。在工作中筆者常常因為起名字而糾結,誇張點可以說是編程5分鐘,命名兩小時!究竟為什麼命名成為了工作中的攔路虎。 ...
  • 練習1 編寫computer類,包含CPU、記憶體、硬碟等屬性,getDetails方法用於返回computer的詳細信息。 編寫PC子類,繼承computer類,添加特有屬性【品牌brand】 編寫notepad類,繼承computer類,添加特有屬性【顏色color】 編寫test object, ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...