# 函數
[TOC]
## 基礎
```python
# 也支持可變函數
a = abs # 變量a指向abs函數
a(-1) # 所以也可以通過a調用abs函數
## 空函數
def nop():
pass # pass可以用來作為占位符,比如現在還沒想好怎么寫函數的代碼,就可以先放一個pass
age = 20
if age >= 18:
pass # pass可以寫在任何地方;
def my_abs(x):
if not isinstance(x, (int, float)): # x必須是int或者float
raise TypeError('bad operand type')
if x >= 0:
return x
else:
return -x
## 返回多個值
import math
def move(x, y, step, angle=0):
nx = x + step * math.cos(angle)
ny = y - step * math.sin(angle)
return nx, ny
# 返回的其實是tuple(151.96152422706632, 70.0) ,但是,在語法上,返回一個tuple可以省略括號
# 而多個變量可以同時接收一個tuple,按位置賦給對應的值
x, y = move(100, 100, 60, math.pi / 6)
```
## 參數
```python
# 位置參數
def power(x, n):
s = 1
while n > 0:
n = n - 1
s = s * x
return s
# 默認參數
def enroll(name, gender, age=6, city='Beijing'):
print('name:', name)
print('gender:', gender)
print('age:', age)
print('city:', city)
enroll('Adam', 'M', city='Tianjin') # 位置參數可以不按位置調用
# 默認參數盡量不要傳可變的值,比如list,set,dict;否則會被污染
def add_end(L=[]):
L.append('END')
return L
add_end() # ['END']
add_end() # ['END', 'END']
add_end() # ['END', 'END', 'END']
# 通過None來解決可變參數被污染,函數的默認參數其實是全局變量,函數內部才是局部變量;
# 把可變的全局變量約束在局部變量里面,就可以防止被污染
def add_end(L=None):
if L is None:
L = []
L.append('END')
s = '局部變量'
return L
```
### 可變參數
1. 在參數前面加了一個*號
2. 在函數內部,參數numbers接收到的是一個tuple
```python
# 給定一組數字a,b,c……,請計算a*a + b*b + c*c + ……
def calc(*numbers):
sum = 0
for n in numbers:
sum = sum + n * n
return sum
print(calc(1,2,3,4))
```
### 關鍵字參數
1. 關鍵字參數允許你傳入0個或任意個含參數名的參數,這些關鍵字參數在函數內部自動組裝為一個dict
```python
def person(name, age, **kw):
print('name:', name, 'age:', age, 'data:', kw)
person('Michael', 30) # name: Michael age: 30 other: {}
person('Bob', 35, city='Beijing') # name: Bob age: 35 other: {'city': 'Beijing'}
person('Adam', 45, gender='M', job='Engineer') # name: Adam age: 45 other: {'gender': 'M', 'job': 'Engineer'}
extra = {'city': 'Beijing', 'job': 'Engineer'};
person('Jack', 24, **extra) # 也可以直接傳dict進來
```
### 命名關鍵字參數
1. 限制關鍵字參數的名字
2. 命名關鍵字參數需要一個特殊分隔符*,*后面的參數被視為命名關鍵字參數。
```python
def person(name, age, *, city, job):
print(name, age, city, job)
person('Jack', 24, city='Beijing', job='Engineer')
person('Jack', 24, job='Engineer') # 報錯,必須寫全參數
person('Jack', 24, name='zxg') # 報錯
# 已經有了一個可變參數,后面跟著的命名關鍵字參數就不再需要一個特殊分隔符*了
def person(name, age, *args, city, job):
print(name, age, args, city, job)
# Jack 24 ('2018-2-9', 'zhouzhou') Beijing Engineer
person('Jack', 24, '2018-2-9', 'zhouzhou' , city='Beijing', job='Engineer')
# 前面的可變參數不能是關鍵字可變參數,否則會報錯
def person(name, age, city , job, **kw ):
print(name, age, 'data:' , kw, city, job)
person('Jack', 24, city='Beijing', job='Engineer' , date='2018-2-9', sex='man')
def f1(a, b, c=0, *args, **kw):
print('a =', a, 'b =', b, 'c =', c, 'args =', args, 'kw =', kw)
f1(1, 2)
f1(1, 2, 3)
f1(1, 2, 3, 'a', 'b')
f1(1, 2, 3, 'a', 'b', x=99)
# 對于任意函數,都可以通過類似func(*args, **kw)的形式調用它,無論它的參數是如何定義的
args = (1, 2, 3, 4)
kw = {'d': 99, 'x': '#'}
f1(*args, **kw)
```