1. 程式人生 > >七 遞歸與二分法、匿名函數、內置函數

七 遞歸與二分法、匿名函數、內置函數

orm left mat 調用 嵌套 得到 結合 type 引用

一 遞歸與二分法

  一、遞歸調用的定義

  二、遞歸分為兩個階段:遞推,回溯

  三、python中的遞歸效率低且沒有尾遞歸優化

  四、可以修改遞歸最大深度

  五、 二分法

二 匿名函數

  一、 什麽是匿名函數?

  二、有名字的函數與匿名函數的對比

三 內置函數

一 遞歸與二分法

一、遞歸調用的定義

#遞歸調用是函數嵌套調用的一種特殊形式,函數在調用時,直接或間接調用了自身,就是遞歸調用

二、遞歸分為兩個階段:遞推,回溯

 1 #圖解。。。
 2 # salary(5)=salary(4)+300
 3 # salary(4)=salary(3)+300
4 # salary(3)=salary(2)+300 5 # salary(2)=salary(1)+300 6 # salary(1)=100 7 # 8 # salary(n)=salary(n-1)+300 n>1 9 # salary(1) =100 n=1 10 11 def salary(n): 12 if n == 1: 13 return 100 14 return salary(n-1)+300 15 16 print(salary(5))

三、python中的遞歸效率低且沒有尾遞歸優化

 1 #python中的遞歸
 2 python中的遞歸效率低,需要在進入下一次遞歸時保留當前的狀態,在其他語言中可以有解決方法:尾遞歸優化,即在函數的最後一步(而非最後一行)調用自己,尾遞歸優化
 3 但是python又沒有尾遞歸,且對遞歸層級做了限制
 4 
 5 #總結遞歸的使用:
 6 1. 必須有一個明確的結束條件
 7 
 8 2. 每次進入更深一層遞歸時,問題規模相比上次遞歸都應有所減少
 9 
10 3. 遞歸效率不高,遞歸層次過多會導致棧溢出(在計算機中,函數調用是通過棧(stack)這種數據結構實現的,每當進入一個函數調用,棧就會加一層棧幀,每當函數返回,棧就會減一層棧幀。由於棧的大小不是無限的,所以,遞歸調用的次數過多,會導致棧溢出)

四、可以修改遞歸最大深度

 1 import sys
 2 sys.getrecursionlimit()
 3 sys.setrecursionlimit(2000)
 4 n=1
 5 def test():
 6     global n
 7     print(n)
 8     n+=1
 9     test()
10 
11 test()
12 
13 雖然可以設置,但是因為不是尾遞歸,仍然要保存棧,內存大小一定,不可能無限遞歸

五、 二分法

想從一個按照從小到大排列的數字列表中找到指定的數字,遍歷的效率太低,用二分法(算法的一種,算法是解決問題的方法)可以極大低縮小問題規模

技術分享圖片
 1 l=[1,2,10,30,33,99,101,200,301,402] #從小到大排列的數字列表
 2 
 3 def search(num,l):
 4     print(l)
 5     if len(l) > 0:
 6         mid=len(l)//2
 7         if num > l[mid]:
 8             #in the right
 9             l=l[mid+1:]
10         elif num < l[mid]:
11             #in the left
12             l=l[:mid]
13         else:
14             print(find it)
15             return
16         search(num,l)
17     else:
18         #如果值不存在,則列表切為空
19         print(not exists)
20         return
21 search(100,l)
22 
23 實現類似於in的效果
類似於實現in的效果 技術分享圖片
 1 l=[1,2,10,30,33,99,101,200,301,402]
 2 
 3 def search(num,l,start=0,stop=len(l)-1):
 4     if start <= stop:
 5         mid=start+(stop-start)//2
 6         print(start:[%s] stop:[%s] mid:[%s] mid_val:[%s] %(start,stop,mid,l[mid]))
 7         if num > l[mid]:
 8             start=mid+1
 9         elif num < l[mid]:
10             stop=mid-1
11         else:
12             print(find it,mid)
13             return
14         search(num,l,start,stop)
15     else: #如果stop > start則意味著列表實際上已經全部切完,即切為空
16         print(not exists)
17         return
18 
19 search(301,l)
20 
21 實現類似於l.index(30)的效果
類似於實現l.index效果

二 匿名函數

一、 什麽是匿名函數?

技術分享圖片
1 匿名就是沒有名字
2 def func(x,y,z=1):
3     return x+y+z
4 
5 匿名
6 lambda x,y,z=1:x+y+z #與函數有相同的作用域,但是匿名意味著引用計數為0,使用一次就釋放,除非讓其有名字
7 func=lambda x,y,z=1:x+y+z 
8 func(1,2,3)
9 #讓其有名字就沒有意義
View Code

二、有名字的函數與匿名函數的對比

1 #有名函數與匿名函數的對比
2 有名函數:循環使用,保存了名字,通過名字就可以重復引用函數功能
3 
4 匿名函數:一次性使用,隨時隨時定義
5 
6 應用:max,min,sorted,map,reduce,filter

三 內置函數

1 #註意:內置函數id()可以返回一個對象的身份,返回值為整數。這個整數通常對應與該對象在內存中的位置,但這與python的具體實現有關,不應該作為對身份的定義,即不夠精準,最精準的還是以內存地址為準。is運算符用於比較兩個對象的身份,等號比較兩個對象的值,內置函數type()則返回一個對象的類型
2 
3 #更多內置函數:https://docs.python.org/3/library/functions.html?highlight=built#ascii 

技術分享圖片

技術分享圖片
 1 #字符串可以提供的參數 ‘s‘ None
 2 >>> format(some string,s)
 3 some string
 4 >>> format(some string)
 5 some string
 6 
 7 #整形數值可以提供的參數有 ‘b‘ ‘c‘ ‘d‘ ‘o‘ ‘x‘ ‘X‘ ‘n‘ None
 8 >>> format(3,b) #轉換成二進制
 9 11
10 >>> format(97,c) #轉換unicode成字符
11 a
12 >>> format(11,d) #轉換成10進制
13 11
14 >>> format(11,o) #轉換成8進制
15 13
16 >>> format(11,x) #轉換成16進制 小寫字母表示
17 b
18 >>> format(11,X) #轉換成16進制 大寫字母表示
19 B
20 >>> format(11,n) #和d一樣
21 11
22 >>> format(11) #默認和d一樣
23 11
24 
25 #浮點數可以提供的參數有 ‘e‘ ‘E‘ ‘f‘ ‘F‘ ‘g‘ ‘G‘ ‘n‘ ‘%‘ None
26 >>> format(314159267,e) #科學計數法,默認保留6位小數
27 3.141593e+08
28 >>> format(314159267,0.2e) #科學計數法,指定保留2位小數
29 3.14e+08
30 >>> format(314159267,0.2E) #科學計數法,指定保留2位小數,采用大寫E表示
31 3.14E+08
32 >>> format(314159267,f) #小數點計數法,默認保留6位小數
33 314159267.000000
34 >>> format(3.14159267000,f) #小數點計數法,默認保留6位小數
35 3.141593
36 >>> format(3.14159267000,0.8f) #小數點計數法,指定保留8位小數
37 3.14159267
38 >>> format(3.14159267000,0.10f) #小數點計數法,指定保留10位小數
39 3.1415926700
40 >>> format(3.14e+1000000,F)  #小數點計數法,無窮大轉換成大小字母
41 INF
42 
43 #g的格式化比較特殊,假設p為格式中指定的保留小數位數,先嘗試采用科學計數法格式化,得到冪指數exp,如果-4<=exp<p,則采用小數計數法,並保留p-1-exp位小數,否則按小數計數法計數,並按p-1保留小數位數
44 >>> format(0.00003141566,.1g) #p=1,exp=-5 ==》 -4<=exp<p不成立,按科學計數法計數,保留0位小數點
45 3e-05
46 >>> format(0.00003141566,.2g) #p=1,exp=-5 ==》 -4<=exp<p不成立,按科學計數法計數,保留1位小數點
47 3.1e-05
48 >>> format(0.00003141566,.3g) #p=1,exp=-5 ==》 -4<=exp<p不成立,按科學計數法計數,保留2位小數點
49 3.14e-05
50 >>> format(0.00003141566,.3G) #p=1,exp=-5 ==》 -4<=exp<p不成立,按科學計數法計數,保留0位小數點,E使用大寫
51 3.14E-05
52 >>> format(3.1415926777,.1g) #p=1,exp=0 ==》 -4<=exp<p成立,按小數計數法計數,保留0位小數點
53 3
54 >>> format(3.1415926777,.2g) #p=1,exp=0 ==》 -4<=exp<p成立,按小數計數法計數,保留1位小數點
55 3.1
56 >>> format(3.1415926777,.3g) #p=1,exp=0 ==》 -4<=exp<p成立,按小數計數法計數,保留2位小數點
57 3.14
58 >>> format(0.00003141566,.1n) #和g相同
59 3e-05
60 >>> format(0.00003141566,.3n) #和g相同
61 3.14e-05
62 >>> format(0.00003141566) #和g相同
63 3.141566e-05
64 
65 format(了解即可)
format 技術分享圖片
 1 字典的運算:最小值,最大值,排序
 2 salaries={
 3     egon:3000,
 4     alex:100000000,
 5     wupeiqi:10000,
 6     yuanhao:2000
 7 }
 8 
 9 叠代字典,取得是key,因而比較的是key的最大和最小值
10 >>> max(salaries)
11 yuanhao
12 >>> min(salaries)
13 alex
14 
15 可以取values,來比較
16 >>> max(salaries.values())
17 >>> min(salaries.values())
18 但通常我們都是想取出,工資最高的那個人名,即比較的是salaries的值,得到的是鍵
19 >>> max(salaries,key=lambda k:salary[k])
20 alex
21 >>> min(salaries,key=lambda k:salary[k])
22 yuanhao
23 
24 
25 
26 也可以通過zip的方式實現
27 salaries_and_names=zip(salaries.values(),salaries.keys())
28 
29 先比較值,值相同則比較鍵
30 >>> max(salaries_and_names)
31 (100000000, alex)
32 
33 
34 salaries_and_names是叠代器,因而只能訪問一次
35 >>> min(salaries_and_names)
36 Traceback (most recent call last):
37   File "<stdin>", line 1, in <module>
38 ValueError: min() arg is an empty sequence
39 
40 
41 
42 sorted(iterable,key=None,reverse=False)
43 
44 !!!lambda與內置函數結合使用!!!
lambda與內置函數結合使用 技術分享圖片
 1 #1、語法
 2 # eval(str,[,globasl[,locals]])
 3 # exec(str,[,globasl[,locals]])
 4 
 5 #2、區別
 6 #示例一:
 7 s=1+2+3
 8 print(eval(s)) #eval用來執行表達式,並返回表達式執行的結果
 9 print(exec(s)) #exec用來執行語句,不會返回任何值
10 ‘‘‘
11 None
12 ‘‘‘
13 
14 #示例二:
15 print(eval(1+2+x,{x:3},{x:30})) #返回33
16 print(exec(1+2+x,{x:3},{x:30})) #返回None
17 
18 # print(eval(‘for i in range(10):print(i)‘)) #語法錯誤,eval不能執行表達式
19 print(exec(for i in range(10):print(i)))
20 
21 eval與exec
eval與exec 技術分享圖片
 1 compile(str,filename,kind)
 2 filename:用於追蹤str來自於哪個文件,如果不想追蹤就可以不定義
 3 kind可以是:single代表一條語句,exec代表一組語句,eval代表一個表達式
 4 s=for i in range(10):print(i)
 5 code=compile(s,‘‘,exec)
 6 exec(code)
 7 
 8 
 9 s=1+2+3
10 code=compile(s,‘‘,eval)
11 eval(code)
12 
13 complie(了解即可)
complie

七 遞歸與二分法、匿名函數、內置函數