Python函數(shù)和模塊的使用詳情
一、定義函數(shù)
在Python中可以使用def關(guān)鍵字來定義函數(shù),命名規(guī)則跟變量的命名規(guī)則是一致的。在函數(shù)名后面的圓括號中可以放置傳遞給函數(shù)的參數(shù),而函數(shù)執(zhí)行完成后我們可以通過return關(guān)鍵字來返回一個值。
定義代碼如下:
def fac(num): ? ? """求階乘""" ? ? result = 1 ? ? for n in range(1, num + 1): ? ? ? ? result *= n ? ? #返回值 ? ? return result
二、函數(shù)的參數(shù)
在Python中,函數(shù)的參數(shù)可以有默認(rèn)值,也支持使用可變參數(shù),所以Python并不需要像其他語言一樣支持函數(shù)的重載,因為我們在定義一個函數(shù)的時候可以讓它有多種不同的使用方式,
下面是兩個小例子:
from random import randint ? ? def roll_dice(n=2): ? ? """搖色子""" ? ? total = 0 ? ? for _ in range(n): ? ? ? ? total += randint(1, 6) ? ? return total ? ? def add(a=0, b=0, c=0): ? ? """三個數(shù)相加""" ? ? return a + 2 * b + 3 * c ? ? # 如果沒有指定參數(shù)那么使用默認(rèn)值搖兩顆色子 print(roll_dice()) # 搖三顆色子 print(roll_dice(3)) #函數(shù)重載,沒有指定函數(shù)變量時,參數(shù)按照順序傳值 print(add()) print(add(1)) print(add(1, 2)) print(add(1, 2, 3)) # 傳遞參數(shù)時可以不按照設(shè)定的順序進行傳遞 print(add(c=50, a=100, b=200)) # 或者可以不傳 print(add(c=50, b=200)) ? if __name__ == '__main__': ? ? pass
沒有傳入對應(yīng)參數(shù)的值時將使用該參數(shù)的默認(rèn)值。
不確定參數(shù)個數(shù)的時候,我們可以使用可變參數(shù),代碼如下所示:
# 在參數(shù)名前面的*表示args是一個可變參數(shù) def add(*args): ? ? total = 0 ? ? for val in args: ? ? ? ? total += val ? ? return total ? ? # 在調(diào)用add函數(shù)時可以傳入0個或多個參數(shù) print(add()) print(add(1)) print(add(1, 2)) print(add(1, 2, 3)) print(add(1, 3, 5, 7, 9))
三、用模塊管理函數(shù)
由于Python沒有函數(shù)重載的概念,那么后面的定義函數(shù)會覆蓋之前的定義,也就意味著兩個函數(shù)同名函數(shù)實際上只有一個是存在的。
def foo(): ? ? print('hello, world!') ? ? def foo(): ? ? print('goodbye, world!') ? ? # 下面的代碼會輸出什么呢? foo() ? if __name__ == '__main__': ? ? pass ? #實際輸出 goodbye, world!
多人協(xié)作進行團隊開發(fā)的時候,團隊中可能有多個程序員都定義了名為foo的函數(shù),那么怎么解決這種命名沖突呢?
重點如何避免函數(shù)覆蓋!!!
在Python中每個.py文件就代表了一個模塊(module),我們在不同的模塊中可以有同名的函數(shù),在使用函數(shù)的時候我們通過import
關(guān)鍵字導(dǎo)入指定的模塊就可以區(qū)分到底要使用的是哪個模塊中的foo
函數(shù),
代碼如下所示:
module1.py ? def foo(): ? ? print('hello, world!') ? module2.py ? def foo(): ? ? print('goodbye, world!')
導(dǎo)入模塊:
test.py ? import module1 as m1 import module2 as m2 ? m1.foo() m2.foo()
注意:但是如果將代碼寫成了下面的樣子,那么程序中調(diào)用的是最后導(dǎo)入的那個foo,因為后導(dǎo)入的foo覆蓋了之前導(dǎo)入的foo。
from module1 import foo from module2 import foo ? # 輸出goodbye, world!,使用的是后導(dǎo)入的模塊 foo()
需要說明的是,如果我們導(dǎo)入的模塊除了定義函數(shù)之外還有可以執(zhí)行代碼,那么Python解釋器在導(dǎo)入這個模塊時就會執(zhí)行這些代碼,事實上我們不想執(zhí)行這些代碼,那我們將可執(zhí)行代碼放入如下所示的條件中,這樣的話除非直接運行該模塊,if條件下的這些代碼是不會執(zhí)行的,因為只有直接執(zhí)行的模塊的名字才是"__main__"
。
def foo(): ? ? pass ? def bar(): ? ? pass ? # __name__是Python中一個隱含的變量它代表了模塊的名字 # 只有被Python解釋器直接執(zhí)行的模塊的名字才是__main__ if __name__ == '__main__': ? ? print('call foo()') ? ? foo() ? ? print('call bar()') ? ? bar()
import module3 # 導(dǎo)入module3時 不會執(zhí)行模塊中if條件成立時的代碼 因為模塊的名字是module3而不是__main__
四、變量的作用域
def foo(): ? ? b = 'hello' ? ? ? # Python中可以在函數(shù)內(nèi)部再定義函數(shù) ? ? def bar(): ? ? ? ? c = True ? ? ? ? print(a) ? ? ? ? print(b) ? ? ? ? print(c) ? ? ? bar() ? ? # print(c) ?# NameError: name 'c' is not defined ? ? if __name__ == '__main__': ? ? a = 100 ? ? # print(b) ?# NameError: name 'b' is not defined ? ? foo()
打印結(jié)果:
100
hello
True
上面的代碼能夠順利的執(zhí)行并且打印出100、hello和True,但我們注意到了,在bar函數(shù)的內(nèi)部并沒有定義a和b兩個變量,那么a和b是從哪里來的。
我們在上面代碼的if分支中定義了一個變量a,這是一個全局變量(global variable),屬于全局作用域,因為它沒有定義在任何一個函數(shù)中。
在上面的foo函數(shù)中我們定義了變量b,這是一個定義在函數(shù)中的局部變量(local variable),屬于局部作用域,在foo函數(shù)的外部并不能訪問到它;但對于foo函數(shù)內(nèi)部的bar函數(shù)來說,變量b屬于嵌套作用域,在bar函數(shù)中我們是可以訪問到它的。
bar函數(shù)中的變量c屬于局部作用域,在bar函數(shù)之外是無法訪問的。
Python查找變量順序:
“局部作用域”-》“嵌套作用域”-》“全局作用域”和“內(nèi)置作用域”的順序進行搜索。
內(nèi)置作用域:Python內(nèi)置的那些標(biāo)識符,我們之前用過的input、print、int等都屬于內(nèi)置作用域。
我們希望通過函數(shù)調(diào)用修改全局變量a的值,但實際上下面的代碼是做不到的。
代碼如下:
def foo(): ? ? a = 200 ? ? print(a) ?# 200 ? ? if __name__ == '__main__': ? ? #這里變量值不能修改函數(shù)局部變量的值 ? ? a = 100 ? ? foo() ? ? print(a) ?# 100
在調(diào)用foo函數(shù)后,我們發(fā)現(xiàn)a的值仍然是100,這是因為當(dāng)我們在函數(shù)foo中寫a = 200的時候,是重新定義了一個名字為a的局部變量,它跟全局作用域的a并不是同一個變量。
因為局部作用域中有了自己的變量a,因此foo函數(shù)不再搜索全局作用域中的a。如果我們希望在foo函數(shù)中修改全局作用域中的a,
代碼如下所示:
def foo(): ? ? global a ? ? a = 200 ? ? print(a) ?# 200 ? ? if __name__ == '__main__': ? ? a = 100 ? ? foo() ? ? print(a) ?# 200
可以使用global關(guān)鍵字來指示foo函數(shù)中的變量a來自于全局作用域。
注:
在實際開發(fā)中,我們應(yīng)該盡量減少對全局變量的使用,因為全局變量的作用域和影響過于廣泛,可能會發(fā)生意料之外的修改和使用。
除此之外全局變量比局部變量擁有更長的生命周期,可能導(dǎo)致對象占用的內(nèi)存長時間無法被垃圾回收
到此這篇關(guān)于Python函數(shù)和模塊的使用詳情的文章就介紹到這了,更多相關(guān)Python函數(shù)模塊內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- 使用Python內(nèi)置模塊與函數(shù)進行不同進制的數(shù)的轉(zhuǎn)換
- 使用Python函數(shù)進行模塊化的實現(xiàn)
- Python3使用xml.dom.minidom和xml.etree模塊兒解析xml文件封裝函數(shù)的方法
- Python函數(shù)和模塊的使用總結(jié)
- python中datetime模塊中strftime/strptime函數(shù)的使用
- Python使用functools模塊中的partial函數(shù)生成偏函數(shù)
- 使用Python內(nèi)置的模塊與函數(shù)進行不同進制的數(shù)的轉(zhuǎn)換
- python使用multiprocessing模塊實現(xiàn)帶回調(diào)函數(shù)的異步調(diào)用方法
- 在Python中使用itertools模塊中的組合函數(shù)的教程
- python中使用sys模板和logging模塊獲取行號和函數(shù)名的方法
相關(guān)文章
Python讀寫及備份oracle數(shù)據(jù)庫操作示例
這篇文章主要介紹了Python讀寫及備份oracle數(shù)據(jù)庫操作,結(jié)合實例形式分析了Python針對Oracle數(shù)據(jù)庫操作的相關(guān)庫安裝,以及使用cx_Oracle與pandas庫進行Oracle數(shù)據(jù)庫的查詢、插入、備份等操作相關(guān)實現(xiàn)技巧,需要的朋友可以參考下2018-05-05Python數(shù)據(jù)分析從入門到進階之分類算法全面教程
數(shù)據(jù)分析是處理和解釋數(shù)據(jù)以發(fā)現(xiàn)有用信息和洞察的過程,其中,分類算法是數(shù)據(jù)分析領(lǐng)域的一個重要組成部分,它用于將數(shù)據(jù)分為不同的類別或組,本文將介紹分類算法的基本概念和進階技巧,以及如何在Python中應(yīng)用這些算法,包括示例代碼和實際案例2023-11-11Python中循環(huán)后使用list.append()數(shù)據(jù)被覆蓋問題的解決
這篇文章主要給大家介紹了關(guān)于Python中循環(huán)后使用list.append()數(shù)據(jù)被覆蓋問題的解決方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-07-07Python利用operator模塊實現(xiàn)對象的多級排序詳解
python中的operator模塊提供了一系列的函數(shù)操作。下面這篇文章主要給大家介紹了在Python中利用operator模塊實現(xiàn)對象的多級排序的相關(guān)資料,需要的朋友可以參考借鑒,下面來一起看看吧。2017-05-05python機器學(xué)習(xí)實戰(zhàn)之最近鄰kNN分類器
這篇文章主要介紹了python機器學(xué)習(xí)實戰(zhàn)之最近鄰kNN分類器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-12-12