第十課
函數(shù)
一 高階函數(shù)
? 接收一個(gè)或者多個(gè)函數(shù)作為參數(shù),或者將函數(shù)作為返回值返回的函數(shù)就是高階函數(shù)
list1 = [1, 2, 3, 4, 5, 6, 7, 8, 9]
def fn(list1):
def fn1(i):
if i % 2 == 0:
return True
new_list = []
for i in list1:
if fn1(i):
new_list.append(i)
return new_list
list2 = fn(list1)
print(list2)
>>>
[2, 4, 6, 8]
高階函數(shù) 可以將定義函數(shù)寫在前面。
list1 = [1, 2, 3, 4, 5, 6, 7, 8, 9]
def fn1(i):
if i % 2 == 0:
return True
def fn2(i):
if i % 2 != 0:
return True
def fn3(i):
if i < 5:
return True
def fn(fun, list):
new_list = []
for i in list1:
if fun(i):
new_list.append(i)
return new_list
list2 = fn(fn3, list)
print(list2)
>>>
[1, 2, 3, 4]
二. 閉包
? 將函數(shù)作為返回值也是高階函數(shù)我們也稱為閉包
? 閉包的好處
? 通過閉包可以創(chuàng)建一些只有當(dāng)前函數(shù)能訪問的變量
? 可以將一些私有數(shù)據(jù)藏到閉包中
-
閉包的特性:使變量不被銷毀
? 行成閉包的條件
? 函數(shù)嵌套
? 將內(nèi)部函數(shù)作為返回值返回
? 內(nèi)部函數(shù)必須要使用到外部函數(shù)的變量或者參數(shù)
def make_fn(fn):
nums = []
def fn1():
nums.append()
return sum(nums)/len(nums)
return fn1()
r = make_fn()
# 修改外部函數(shù)的變量
def func_out(num1):
def func_inner(num2):
ninlocal num1
num1 = 10
result = num1 + num2
print('結(jié)果:', result)
print(num1)
func_inner(1)
print(num1)
return func_inner()
f = func_out(1)
>>>
1
結(jié)果: 11
1
為幫助出坑特推薦
Pythontutor是一個(gè)在線可視化代碼執(zhí)行過程的的網(wǎng)站,支持的語言有Python、C++、Java等。
地址:http://www.pythontutor.com/
三. 裝飾器的引入
? 我們可以直接通過修改函數(shù)中的代碼來完成需求,但是會(huì)產(chǎn)生以下一些問題
? 如果修改的函數(shù)多,修改起來會(huì)比較麻煩
? 不方便后期的維護(hù)
? 這樣做會(huì)違反開閉原則(ocp)
? 程序的設(shè)計(jì),要求開發(fā)對(duì)程序的擴(kuò)展,要關(guān)閉對(duì)程序的修改
四. 裝飾器的使用
? 通過裝飾器,可以在不修改原來函數(shù)的情況下來對(duì)函數(shù)進(jìn)行擴(kuò)展
? 在開發(fā)中,我們都是通過裝飾器來擴(kuò)展函數(shù)的功能的
def add(a, b):
return a + b
def fn():
print('我是fn函數(shù)')
def start_end(old):
def new_fn(*args, **kwargs):
print('函數(shù)開始執(zhí)行。。。')
r = old(*args, **kwargs)
print('函數(shù)執(zhí)行結(jié)束。。。')
return r
return new_fn
f = start_end(add)
print(f(1,2))
>>>
函數(shù)開始執(zhí)行。。。
函數(shù)執(zhí)行結(jié)束。。。
3
@start_end
def speak():
print('hello world')
speak()
>>>
函數(shù)開始執(zhí)行。。。
hello world
函數(shù)執(zhí)行結(jié)束。。。