1、 什麼是函數遞歸調用
函數的遞歸調用是函數嵌套調用的一種特殊形式,在調用一個函數的過程當中又直接或者間接地調用該函數
自己,稱之爲函數的遞歸調用
遞歸調用必須有兩個明確的階段:
1. 回溯: 一次次遞歸調用下去,說白了就一個重複的過程,但須要注意的是每一次重複問題的規模都應該有所減小,直到逼近一個最終的結果,即回溯階段必定要有一個明確的結束條件
2. 遞推: 往回一層一層推算出結果python
import sys print(sys.getrecursionlimit()) sys.setrecursionlimit(2048)
def foo(n): print('from foo',n) foo(n+1) foo(0)
def bar(): print('from bar') foo() def foo(): print('from foo') bar() foo()
age(5)=age(4)+2 age(4)=age(3)+2 age(3)=age(2)+2 age(2)=age(1)+2 age(1)=18 age(n)=age(n-1)+2 #n>1 age(n)=18 #n=1
# 遞歸調用就是一個重複的過程,可是每一次重複問題的規模都應該有所減小,而且應該在知足某種條件的狀況下結束重複,開始進入遞推階段算法
def age(n): if n == 1: return 10 return age(n-1) + 2 print(age(5)) ------------------------------------------- 18
l=[1,[2,[3,[4,[5,[6,[7,[8,[9,[10,[11,]]]]]]]]]]] def search(l): for item in l: if type(item) is not list: # 不是列表直接打印 print(item) else: # 判斷是列表則繼續循環,判斷... search(item) search(l) ---------------------------------------------------------- 1 2 3 4 5 6 7 8 9 10 11
算法:是如何高效率地解決某一個問題的方法/套路
# 二分法app
nums=[13,15,17,23,31,53,74,81,93,102,103,201,303,403,503,777] find_num=503 def binary_search(nums,find_num): print(nums) if len(nums) == 0: print('not exists') return mid_index=len(nums) // 2 if find_num > nums[mid_index]: # in the right nums=nums[mid_index+1:] # 切一半 # 從新執行二分的邏輯 binary_search(nums,find_num) elif find_num < nums[mid_index]: #in the left nums=nums[0:mid_index] # 從新執行二分的邏輯 binary_search(nums,find_num) else: print('find it') # binary_search(nums,find_num) binary_search(nums,94) -------------------------------------------------------------------- [13, 15, 17, 23, 31, 53, 74, 81, 93, 102, 103, 201, 303, 403, 503, 777] [102, 103, 201, 303, 403, 503, 777] [102, 103, 201] [102] [] not exists
#####################################################################ide
l=[0,1,2,3,4,5,6,7,8,9] def search(l,num): print(l) if len(l)==0: print("not") return my_index=len(l) // 2 # 10/2 =5 ##//地板除,取整數 if num > l[my_index]: l=l [my_index+1:] #5+1 search(l,num) elif num < l[my_index]: l=l[0:my_index] search(l,num) else: print("find it") search(l,3) --------------------------------------------------------------------------------------------------------------- [0, 1, 2, 3, 4, 5, 6, 7, 8, 9] [0, 1, 2, 3, 4] [3, 4] [3] find it
2、三元表達式函數
def max2(x, y): # if x > y: # return x # else: # return y return x if x > y else y # 三元表達式實現的效果就是:條件成立的狀況下返回一個值,不成立的狀況下返回另一種值 # res=條件成立狀況下返回的值 if 條件 else 條件不成立狀況下返回的值 name=input('your name: ').strip() res="SB" if name == 'lqz' else "NB" print(res)
3、列表生成式、字典生成式
spa
names=['alex','lqz','yyh','fm'] l=[] for name in names: res=name + '_DSB' l.append(res) print(l) l=[name + '_DSB' for name in names] print(l) --------------------------------------------------- ['alex_DSB', 'lqz_DSB', 'yyh_DSB', 'fm_DSB'] ['alex_DSB', 'lqz_DSB', 'yyh_DSB', 'fm_DSB']
names=['alex_sb','lqz_sb','yyh_sb','fm_sb','egon'] l=[] for name in names: if name.endswith('sb'): l.append(name) print(l) l=[name for name in names if name.endswith('sb')] print(l) -------------------------------------------------------- ['alex_sb', 'lqz_sb', 'yyh_sb', 'fm_sb'] ['alex_sb', 'lqz_sb', 'yyh_sb', 'fm_sb']
items=[ ('name','egon'), ('age',18), ('sex','male'), ] dic=dict(items) print(dic) ------------------------------------ {'name': 'egon', 'age': 18, 'sex': 'male'}
# 補充排序
l=['a','b','c','d'] for i,v in enumerate(l): print(i,v) ------------------------------------------- 0 a 1 b 2 c 3 d
keys=['name','age','sex'] vals=['egon',18,'male'] dic={} for i,k in enumerate(keys): # print(i,k) dic[k]=vals[i] print(dic) dic={k:vals[i] for i,k in enumerate(keys)} print(dic) dic={k:vals[i] for i,k in enumerate(keys) if i > 0} print(dic) ---------------------------------------------------------- {'name': 'egon', 'age': 18, 'sex': 'male'} {'name': 'egon', 'age': 18, 'sex': 'male'} {'age': 18, 'sex': 'male'}
print({i:i for i in range(10)}) print({i for i in range(10)}) print({i for i in 'hello'}) ------------------------------------------- {0: 0, 1: 1, 2: 2, 3: 3, 4: 4, 5: 5, 6: 6, 7: 7, 8: 8, 9: 9} {0, 1, 2, 3, 4, 5, 6, 7, 8, 9} {'l', 'h', 'o', 'e'}
4、匿名函數
遞歸
1 匿名函數:就是沒有名字的函數
2 爲什麼要用:
用於僅僅臨時使用一次的場景,沒有重複使用的需求ip
def sum2(x,y): return x+y print(lambda x,y:x+y) print((lambda x,y:x+y)(1,2)) --------------------------------------- <function <lambda> at 0x0000022BFE155510> 3
# 匿名函數的精髓就是沒有名字,爲其綁定名字是沒有意義的get
f=lambda x,y:x+y print(f) print(f(1,2)) -------------------------------------- <function <lambda> at 0x000002E84DFB2EA0> 3
# 匿名函數與內置函數結合使用
# max,min,sorted,map,filter,reduce
salaries={ 'egon':300000, 'alex':100000000, 'wupeiqi':10000, 'yuanhao':2000 } # 求薪資最高的那我的名:即比較的是value,但取結果是key res=max(salaries) print(res) ----------------------------------------------------------- yuanhao
# 能夠經過max函數的key參數來改變max函數的比較依據,運行原理:
# max函數會「for循環」出一個值,而後將該值傳給key指定的函數
# 調用key指定的函數,將拿到的返回值看成比較依據
salaries={ 'egon':300000, 'alex':100000000, 'wupeiqi':10000, 'yuanhao':2000 } def func(name): # 返回一我的的薪資 return salaries[name] res=max(salaries,key=func) #'egon' print(res) ------------------------------------------------------ alex
d={"a":1,"b":2,'c':3} print(max(d)) def func(k): return d[k] print(func("a")) # 函數根據key取對應的value res=max(d,key=func) print(res) # 匿名函數寫法 print(max(d,key=lambda s:d[s])) -------------------------------------------- c 1 c c
# 求最大值
salaries={ 'egon':300000, 'alex':100000000, 'wupeiqi':10000, 'yuanhao':2000 } res=max(salaries,key=lambda name:salaries[name]) #'egon' print(res) ----------------------------------------------------- alex
# 求最小值
salaries={ 'egon':300000, 'alex':100000000, 'wupeiqi':10000, 'yuanhao':2000 } res=min(salaries,key=lambda name:salaries[name]) #'egon' print(res) ------------------------------------------------------- yuanhao
# sorted排序
nums=[11,33,22,9,31] res=sorted(nums,reverse=True) print(nums) print(res) salaries={ 'egon':300000, 'alex':100000000, 'wupeiqi':10000, 'yuanhao':2000 } for v in salaries.values(): print(v) res=sorted(salaries.values()) print(res) res=sorted(salaries,key=lambda name:salaries[name],reverse=True) print(res) ------------------------------------------------------------------------ [11, 33, 22, 9, 31] [33, 31, 22, 11, 9] 300000 100000000 10000 2000 [2000, 10000, 300000, 100000000] ['alex', 'egon', 'wupeiqi', 'yuanhao']
map:把一個列表按照咱們自定義的映射規則映射成一個新的列表
names=['alex','lxx','wxx','yxx'] res=map(lambda name: name + "dSB", names) print(list(res)) ---------------------------------------------- ['alexdSB', 'lxxdSB', 'wxxdSB', 'yxxdSB']
filter: 從一個列表中過濾出符合咱們過濾規則的值
運行原理:至關於for循環取出每個人名,而後傳給匿名函數,將調用匿名函數返回值爲True的那我的名給留下來
names=['alex_sb','lxx_sb','wxx_sb','egon','yxx'] res=filter(lambda name:name.endswith('sb'),names) print(list(res)) print([name for name in names if name.endswith('sb')]) ------------------------------------------------------------ ['alex_sb', 'lxx_sb', 'wxx_sb'] ['alex_sb', 'lxx_sb', 'wxx_sb']
# reduce: 把多個值合併成一個結果
from functools import reduce l=['a','b','c','d'] res=reduce(lambda x,y:x+y,l,'A') # 'A','a' => 'Aa' # 'Aa','b'=>'Aab' # 'Aab','c'=>'Aabc' # 'Aabc','d'=>'Aabcd' print(res) --------------------------------------- Aabcd
------------------------------------------------------------------------------------------------------------------------------
from functools import reduce l=['a','b','c','d'] res=reduce(lambda x,y:x+y,l) # 'a','b'=>'ab' print(res) res=reduce(lambda x,y:x+y,range(1,101)) # 1,2=>3 # 3,3=>6 print(res) ---------------------------------------------------------------------------- abcd 5050
做業:
做業: 1 使用遞歸打印斐波那契數列(前兩個數的和獲得第三個數,如:0 1 1 2 3 4 7...) 2 一個嵌套不少層的列表,如l=[1,2,[3,[4,5,6,[7,8,[9,10,[11,12,13,[14,15]]]]]]],用遞歸取出全部的值 3 編寫用戶登陸裝飾器,在登陸成功後無需從新登陸,同一帳號重複輸錯三次密碼則鎖定5分鐘 四、求文件a.txt中總共包含的字符個數?思考爲什麼在第一次以後的n次sum求和獲得的結果爲0?(須要使用sum函數) 五、文件shopping.txt內容以下 mac,20000,3 lenovo,3000,10 tesla,1000000,10 chicken,200,1 求總共花了多少錢? 打印出全部商品的信息,格式爲[{'name':'xxx','price':333,'count':3},...] 求單價大於10000的商品信息,格式同上 明日默寫: 二分查找