一、語(yǔ)法格式:
def test(x):
"解釋函數(shù)的作用"
x += 1
return x
def:定義函數(shù)的關(guān)鍵字
test:函數(shù)名
():定義函數(shù)的形參
"":文檔描述膜蠢,定義函數(shù)的作用
x+=1:函數(shù)的代碼塊
return:定義返回值
當(dāng)函數(shù)遇到一個(gè)return時(shí)整個(gè)函數(shù)結(jié)束
當(dāng)指定了默認(rèn)參數(shù)的時(shí)候堪藐,在調(diào)用函數(shù)的時(shí)候可以不指定參數(shù)
二莉兰、函數(shù)的參數(shù):
1、形參:
在函數(shù)中定義的變量礁竞,其只有在被調(diào)用的時(shí)候才會(huì)被分配內(nèi)存單元糖荒,在調(diào)用結(jié)束時(shí)釋放分配的內(nèi)存單元
2、實(shí)參:
在調(diào)研用函數(shù)的時(shí)候定義的常量模捂、變量捶朵、表達(dá)式、函數(shù)等
3狂男、例子:
def add(x,y):
res = x * y
return res
add = add(2,4)
在上面的例子中形參為add后面的x和y综看,實(shí)參為調(diào)用函數(shù)時(shí)的2和4;形參和實(shí)參的位置必須要一一對(duì)應(yīng)
三岖食、可變長(zhǎng)參數(shù):
可變長(zhǎng)參數(shù)是指可以向函數(shù)的傳遞不定個(gè)數(shù)的參數(shù)红碑;可變長(zhǎng)參數(shù)的原則是位置參數(shù)必須在關(guān)鍵字參數(shù)之前
1、位置參數(shù):
位置參數(shù)是和函數(shù)中的形參一一對(duì)應(yīng)的泡垃,如:
def add(x,y):
res = x * y
return res
add(2,4)
在上面的代碼中2和4對(duì)應(yīng)函數(shù)中x和y析珊,也就是x和y是位置參數(shù)
2、關(guān)鍵字參數(shù):
關(guān)鍵字參數(shù)是指可以接受0個(gè)或多個(gè)實(shí)參的參數(shù)蔑穴,如:
def test01(x, *args):
print(x)
print(args)
test01(1,2,3,4,5)
在上面的代碼中x是位置參數(shù)忠寻,對(duì)應(yīng)實(shí)參中的1;
而*args是一個(gè)關(guān)鍵字參數(shù),對(duì)應(yīng)實(shí)參中的2澎剥,3锡溯,4,5
3哑姚、*:
在函數(shù)的形參中*可接受字符串祭饭、數(shù)字、列表叙量、元組倡蝙、集合的數(shù)據(jù)類型;Python最后會(huì)將這些數(shù)據(jù)類型轉(zhuǎn)化為元組绞佩。
例子:
def test01(x, *args):
print(x)
print(args)
test01(1,2,3,4,5)
運(yùn)行結(jié)果:
1
(2, 3, 4, 5)
若傳遞的實(shí)參是一個(gè)可迭代的參數(shù)時(shí)需要加上一個(gè)*號(hào)寺鸥,如:
def test01(x, *args):
print(x)
print(args)
test01(1, *[2,3,4])
運(yùn)行結(jié)果:
1
(2, 3, 4)
4、**:
和*類似品山,不同的是**可接收的是鍵值對(duì)胆建,最后都會(huì)轉(zhuǎn)換為字典形式
例子:
def test02(x, **kwargs):
print(x)
print(kwargs)
test02(1, y=2, z=3)
運(yùn)行結(jié)果:
1
{'y': 2, 'z': 3}
和*類似**也可以接收字典類型,如:
例子:
def test02(x, **kwargs):
print(x)
print(kwargs)
test02(1, **{"y":2, "z":3})
運(yùn)行結(jié)果:
1
{'y': 2, 'z': 3}
5肘交、當(dāng) * 和 ** 共用時(shí):
因?yàn)榭勺冮L(zhǎng)參數(shù)的原則是位置參數(shù)必須在關(guān)鍵字參數(shù)之前笆载,所以當(dāng)*和**共用時(shí)為了避免產(chǎn)生歧義,需要將*放在**之前;如:
def test03(x, *args, **kwargs):
print(x)
print(args)
print(kwargs)
test03(1,2,3,4,y=1,z=2)
運(yùn)行結(jié)果:
1
(2, 3, 4)
{'y': 1, 'z': 2}
四凉驻、全局變量和局部變量
1腻要、全局變量的定義:
在Python代碼中的任意位置都能調(diào)用的變量;如:
name = "xiaoming"
def get_name():
print(name)
get_name()
運(yùn)行結(jié)果:
xiaoming
在上面的例子中在函數(shù)外定義了一個(gè)變量 “name”涝登,然后在函數(shù)內(nèi)部調(diào)用這個(gè)變量雄家;最后發(fā)現(xiàn)我們可以調(diào)用“name”變量;所以變量“name”被稱為全局變量
2胀滚、局部變量的定義:
在Python代碼中只在特定過(guò)程或函數(shù)中可以訪問(wèn)的變量,當(dāng)沒(méi)有在子程序中定義局部變量時(shí)會(huì)去上一層進(jìn)行查找是否存在相同的變量名趟济;如:
name = "xiaoming"
def get_name1():
name = "xiaohong"
print(name)
def get_name2():
print(name)
get_name1()
get_name2()
運(yùn)行結(jié)果:
xiaohong
xiaoming
根據(jù)上面的例子可以發(fā)現(xiàn)先定義的全局變量在函數(shù)“get_name1”中并沒(méi)有被調(diào)用,而是調(diào)用了函數(shù)內(nèi)部的定義的“name”變量蛛淋;而函數(shù)“get_name2”調(diào)用了全局變量“name”咙好,無(wú)法調(diào)用函數(shù)“get_name1”中定義的“name”變量篡腌;所以在函數(shù)“get_name1”中的變量“name”稱之為局部變量褐荷。
3、global關(guān)鍵字:
在函數(shù)中的局部變量之前使用 global 關(guān)鍵字可以將局部變量定義為全局變量嘹悼;如:
name = "xiaoming"
def get_name1():
global name
name = "xiaohong"
print(name)
def get_name2():
print(name)
get_name1()
get_name2()
運(yùn)行結(jié)果:
xiaohong
xiaohong
在上面的例子中叛甫,在函數(shù)“get_name1”中使用了關(guān)鍵字 global 將函數(shù)“get_name1”中的局部變量“name”轉(zhuǎn)化成了全局變量“name”;因?yàn)樵诖a的開(kāi)頭已經(jīng)定義了全局變量‘name=“xiaoming”’杨伙,所以在函數(shù)“get_name1”中的變量“name”將代碼開(kāi)頭的全局變量“name”的值改成了“xiaohong”其监,所以在函數(shù)“get_name2”中調(diào)用“name”的值成了“xiaohong”。
4限匣、函數(shù)的嵌套
函數(shù)支持多個(gè)函數(shù)進(jìn)行嵌套抖苦,如:
name = "xiaoming"
def get_name1():
name = "xiaobai"
print(name)
def get_name2():
global name
name = "xiaohong"
get_name2()
get_name1()
print(name)
運(yùn)行結(jié)果:
xiaobai
xiaohong
注:函數(shù)在沒(méi)有被調(diào)用的情況下只進(jìn)行編譯,不執(zhí)行米死。
當(dāng)在調(diào)用函數(shù)“get_name1()”的時(shí)候局部變量“name”的值是“xiaobai”锌历,所以在函數(shù)“get_name1()”中print的時(shí)候輸出“xiaobai”;在函數(shù)“get_name1()”中調(diào)用了函數(shù)“get_name2()”峦筒,在函數(shù)“get_name2()”中使用了global關(guān)鍵字將局部變量name變?yōu)榱巳肿兞縩ame并進(jìn)行了修改究西,所以在代碼最后print的時(shí)候輸出“xiaohong”
5、nonlocal關(guān)鍵字
nonlocal關(guān)鍵字和global關(guān)鍵字類似物喷,不同的是global關(guān)鍵字是將局部變量轉(zhuǎn)換為全局變量卤材;而nonlocal關(guān)鍵字是將局部變量轉(zhuǎn)為上一級(jí)函數(shù)的局部變量,當(dāng)函數(shù)只有一級(jí)的時(shí)候使用nonlocal程序會(huì)報(bào)錯(cuò)峦失,如:
name = "xiaoming"
def get_name1():
name = "xiaobai"
print(name)
def get_name2():
nonlocal name
name = "xiaohong"
get_name2()
print(name)
get_name1()
print(name)
運(yùn)行結(jié)果:
xiaobai
xiaohong
xiaoming
6扇丛、作用域
函數(shù)的作用域和函數(shù)的定義有關(guān),和函數(shù)的調(diào)用位置無(wú)關(guān)尉辑;如:
name = "xiaohong"
def name1():
name = "xiaoming"
def name2():
print(name)
return name2
name1()()
運(yùn)行結(jié)果:
xiaoming
解釋:
當(dāng)調(diào)用函數(shù)name1的時(shí)候程序會(huì)返回函數(shù)name2的內(nèi)存地址帆精,所以name1()()即時(shí)調(diào)用函數(shù)name2,因?yàn)閚ame2是在name1下嵌套的一個(gè)函數(shù),而name2本身沒(méi)有局部變量name实幕,所以程序會(huì)到name2的上一層函數(shù)中尋找是否有變量name吝镣,有即輸出,若沒(méi)有找到則去全局變量中找昆庇。
五末贾、向前引用(風(fēng)濕理論)
下面通過(guò)四種情況來(lái)理解向前引用
在Python中函數(shù)即變量!
情況1:
name = "xiaoming"
def get_name1():
print(name)
get_name2()
get_name1()
運(yùn)行結(jié)果:
報(bào)錯(cuò)
解釋:
當(dāng)創(chuàng)建一個(gè)函數(shù)的時(shí)候Python會(huì)在內(nèi)存中獲取一個(gè)空間整吆,用該空間來(lái)存放函數(shù)內(nèi)的代碼塊拱撵,并將這個(gè)內(nèi)存空間指向函數(shù)名,就和變量一樣表蝙;在情況1中指向的是get_name1拴测。因?yàn)樵趃et_name1中的代碼塊中又調(diào)用了一個(gè)函數(shù)get_name2,但在內(nèi)存中沒(méi)有存在指向get_name2的內(nèi)存空間府蛇,所以Python無(wú)法執(zhí)行這行代碼集索,固報(bào)錯(cuò)。
情況2:
name1 = "xiaoming"
name2 = "xiaohong"
def get_name1():
print(name1)
get_name2()
def get_name2():
print(name2)
get_name1()
運(yùn)行結(jié)果:
xiaoming
xiaohong
解釋:
和情況1類似汇跨,不同的是創(chuàng)建了一個(gè)函數(shù)“get_name2”务荆,即內(nèi)存中有內(nèi)存空間指向了get_name2,所以在get_name1中調(diào)用get_name2的時(shí)候Python能夠執(zhí)行相應(yīng)的內(nèi)容
情況3:
name1 = "xiaoming"
name2 = "xiaohong"
def get_name2():
print(name2)
def get_name1():
print(name1)
get_name2()
get_name1()
運(yùn)行結(jié)果:
xiaoming
xiaohong
解釋:
和情況2一樣穷遂,內(nèi)存中存在執(zhí)行g(shù)et_name1和get_name2的內(nèi)存空間函匕,所以get_name1和get_name2都能被調(diào)用并執(zhí)行。
情況4:
name1 = "xiaoming"
name2 = "xiaohong"
def get_name1():
print(name1)
get_name2()
get_name1()
def get_name2():
print(name2)
運(yùn)行結(jié)果:
報(bào)錯(cuò)
解釋:
因?yàn)镻ython代碼是從上往下執(zhí)行的蚪黑,所以當(dāng)我們?cè)诖a中間調(diào)用get_name1的時(shí)候Python會(huì)去內(nèi)存中找是否存在指向get_name1的內(nèi)存空間盅惜;可以發(fā)現(xiàn)能夠在內(nèi)存中找到指向get_name1的內(nèi)存空間,然后執(zhí)行其中的get_name1的代碼塊忌穿,發(fā)現(xiàn)其中調(diào)用了函數(shù)get_name2抒寂,但是get_name2函數(shù)在這時(shí)候還沒(méi)有被創(chuàng)建,內(nèi)存中不存在指向get_name2的內(nèi)存空間伴网,所以無(wú)法執(zhí)行蓬推。
六、遞歸函數(shù)
1澡腾、遞歸函數(shù)的定義
遞歸函數(shù)是指在函數(shù)的內(nèi)部調(diào)用自己
2沸伏、特性
- 必須有一個(gè)明確的結(jié)束條件
- 每次執(zhí)行一次遞歸后問(wèn)題的規(guī)模都應(yīng)比上一次減少
- 遞歸的效率不高,當(dāng)遞歸的次數(shù)較多的時(shí)候會(huì)占用內(nèi)存
3动分、案例:
1毅糟、使用遞歸函數(shù)計(jì)算1到100之間的和
def sum(n):
if int(n) == 1:
return n
else:
return sum(n - 1) + n
print(sum(100))
運(yùn)行結(jié)果:
5050
2、使用遞歸函數(shù)計(jì)算10的階乘
def product(n):
if n == 1:
return 1
else:
return product(n - 1) * n
print(product(10))
運(yùn)行結(jié)果:
3628800
4澜公、尾遞歸調(diào)用的優(yōu)化
定義:在函數(shù)執(zhí)行完最后一步的時(shí)候在去調(diào)用其它函數(shù)姆另。(注意:是最后一步而不是最后一行)
在遞歸函數(shù)的特性中已經(jīng)知道了遞歸函數(shù)會(huì)占用內(nèi)存喇肋,原因是在函數(shù)中調(diào)用函數(shù)時(shí)若當(dāng)前函數(shù)并沒(méi)有完全執(zhí)行完就調(diào)用了函數(shù)就會(huì)造成內(nèi)存空間的占用,如:
def func1(x):
return x
def func2(x):
return func1(x) + 1
print(func2(4))
解釋:
函數(shù)func2的return值就相當(dāng)于兩步:1迹辐、res = func1(x) 2蝶防、return res + 1;在執(zhí)行第一步的時(shí)候函數(shù)func2會(huì)等待func1執(zhí)行完后返回的值明吩,所以遞歸函數(shù)若用非尾調(diào)用的方法寫會(huì)作用內(nèi)存空間
將上面的代碼改為尾調(diào)用
def func1(x):
return x + 1
def func2(x):
return func1(x)
print(func2(4))
七间学、匿名函數(shù)
和def定義的函數(shù)不同的是匿名函數(shù)是一個(gè)表達(dá)式,定義方法如下:
lambda x:x+1
這相當(dāng)于def函數(shù)中的
def test(x):
return x+1
例子:
1印荔、生存一個(gè)列表低葫,要求元素必須是2的1-10次方,如:[2,4,8,16,...]
list01 = []
func = lambda x:2**x
for i in range(1,11):
list01.append(func(i))
print(list01)
運(yùn)行結(jié)果:
[2, 4, 8, 16, 32, 64, 128, 256, 512, 1024]
2仍律、修改上面的代碼將列表內(nèi)的所有元素都乘以100后輸出
list01 = []
func = lambda x:2**x*100
for i in range(1,11):
list01.append(func(i))
print(list01)
運(yùn)行結(jié)果:
[200, 400, 800, 1600, 3200, 6400, 12800, 25600, 51200, 102400]