八. Python基础(8)--函数
注意, 自定义的函数也可以是用来作逻辑判断的, 例如内置的startswith()等函数. |
def check_len(x): '''
:param x: :return: ''' if len(x) > 5: return True # 如果这里不写else, 并return一个False, 那么返回一个None, 相当于False. |
print(check_len.__doc__) ''' 结果: :param x: ******* :return: ******* ''' |
程序从上到下执行到一个函数定义的语句时, 内存中只是保存了函数名, 并没有执行函数体内的语句 |
|
① 减少代码的冗余 ② 提高代码的可读性 ③ 提高代码的重用性 ④ 提高代码的的可扩展性 ⑤ 解耦 |
面向过程的编程--功能与功能之间耦合很紧密 解耦--把一个功能尽量细化成多个小功能, 并且功能与功能之间的影响尽量减到最小--不宜把一个很复杂的功能放在一个函数里 |
程序要做到: "高内聚, 低耦合"high cohesion and low coupling |
def 函数名(参数1, 参数2): # 形参 ''' 这是一个解决什么问题的函数 :param 参数1: :param 参数2: :return: ''' print('函数体') 返回值 = [参数1, 参数2] return 返回值
接受返回值 = 函数名('刘桂香', 88) # 实参 print(接受返回值) |
函数体 ['刘桂香', 88] |
① 可以不给它传值的形参 ② 如果不给它传值, 那么实参接收默认值,如果给它传了值, 就接收传入的值 ③ 默认的值是在定义阶段就已经确定了 |
a = 18 def age(a1, a2 = a): print(a1, a2) a = 20
age(10) # 10 18 |
def demo3(a = []): a.append(1) print(a) # # demo3() # demo3() # 仍用默认的列表 # demo3() # 仍用默认的列表 # ''' # [1] # [1, 1] # [1, 1, 1]
# ''' # demo3([]) # 用新列表 # demo3([]) # 用新列表 # demo3([]) # 用新列表 ''' [1] [1] [1] '''
# demo3() # demo3([]) # 用新列表, 而不是默认的列表 # demo3() # 用默认的列表 ''' [1] [1] [1, 1] ''' |
# 还有一种解决方法是在进入函数体的时候置空默认的列表 def demo3(a = []): a = [] a.append(1) print(a) # # demo3() # demo3() # 仍用默认的列表 # demo3() # 仍用默认的列表 ''' [1] [1] [1] ''' |
8 ● 动态参数(dynamic argument)/变长参数(variable length argument)
We can collect the arguments using the * and ** specifiers in the function's parameter list; this gives you the (unnamed) positional arguments as a tuple and the (named) keyword arguments as a dictionary. 在形式参数前面添加标识符"*", 表示我们可以将任意数量的、无名的(unnamed)位置实参以元组的形式传递给形参. 在形式参数前面添加标识符"**", 表示我么可以将任意数量的、有名的(named)关键字实参以字典的形式传递给形参. |
#① 站在函数定义的角度: *做组合(gather)用, 将多个参数组合成一个元组 #② 站在函数调用的角度: *做打撒(unpack)用(打散列表, 元组, 字符串), 将一个列表或者元组打散成多个参数 #③ * 只针对按位置传参 |
def my_sum(*args): # 实参会组合成一个元组 count_sum = 0 for i in args: count_sum += i return count_sum
print(my_sum(1,2,3)) # 将1,2,3 组合成一个元组, 结果: 6 li = [1,2,3] print(my_sum(*li)) # 先将列表li打散成1,2,3, 再将1,2,3组合成后元组, 结果: 6 |
#① 站在函数定义的角度: *做组合(gather)用 #② 站在函数调用的角度: *做打撒(unpack)用(打散列表或者元组) #③ ** 只针对按关键字传参 |
def demo4(**kwargs): print(kwargs)
demo4(a = 1, b = 2, c = 3) # 关键字a,b,c不用加引号, 结果的键有引号: {'a': 1, 'b': 2, 'c': 3} dic = {'a': 1, 'b': 2, 'c': 3} demo4(**dic) # 先将词典dic打散, 然后组合为一个词典,结果和dic一模一样: {'a': 1, 'b': 2, 'c': 3} |
def demo5(*args, **kwargs): print(args) prind = {'a':11, 'b':22, 'c':33}
t = (1,2,3) d = {'a':11, 'b':22, 'c':33}
print(*t) # 1 2 3 demo5(*t) ''' (1, 2, 3) {} ''' |
# 站在传参(实参给形参传递参数)的角度or站在函数调用的角度: 所有的参数都是实际参数 ①按位置传值 ②按关键字传值
# 站在形参接受实参传递的值的角度or站在函数定义的角度:所有的参数都是形式参数 ①位置参数 ②默认参数 ③动态参数: *args, ** kwargs ※定义顺序:位置参数, 动态参数 * args, 默认参数, 动态参数 ** kwargs |
13 ● 作为形参的位置参数, 动态参数*args, 默认参数, 动态参数**kwargs的定义顺序
def func(位置参数1, 位置参数2, *args, 默认参数 = 10, **kwargs): print(位置参数1, 位置参数2) print(默认参数) print(args) print(kwargs)
# func(1,2,3,4,5,默认参数= 'hahaha', a =10, b =20) ''' 1 2 hahaha (3, 4, 5) {'a': 10, 'b': 20} ''' # func(1,2,3,4,5,a=10, b=20) ''' 1 2 10 (3, 4, 5) {'a': 10, 'b': 20} ''' |
① 关键字参数是针对实参而言的, 准确来说, 它是按照关键字传值的实参 ② 默认函数是针对形参而言的 |
# 有关键字参数 def foo(bar, baz): pass
foo(1, 2) # 按照位置(position)传值的实参 foo(baz=2, bar=1) # 按照关键字(keyword)传值的实参 |
# 有关默认参数 def foo(baz, bar =5): # 默认形参前可以有位置形参 print(baz, bar)
foo(1, 2) # 1 2 foo(1) # 1 5
# 下面的代码报错(non-default argument follows default argument) def foo(bar =5, baz): # 默认参数后不允许有位置参数 print(baz, bar)
foo(1, 2) |
# 详见下面代码, 注意区分关键字参数和默认参数 def fun1(x=5, **kwargs): print(x) print(kwargs)
#① fun1(a=1, b=2) ''' 5 {'a': 1, 'b': 2}
fun1(a=1, b=2) '''
#② fun1(x=7, a=1, b=2) ''' 7 {'a': 1, 'b': 2} ''' #③ fun1(7, a=1, b=2) ''' 7 {'a': 1, 'b': 2} '''
# ④ fun1(7, x=1, b=2) ''' fun1() got multiple values for argument 'x' '''
# ⑤ ''' def fun1(x=5, *args): print(x) print(args)
fun1(1,3,6) '''
# 注意: 这里虽然没有报错, 但是作为默认参数的x就没有意义了, 因为此时没有办法在不给x传值的情况下还能打印出默认的5
# ⑥ ''' def fun1(*args, x=5): print(x) print(args)
fun1(1,3,6) ''' # 注意: 这里的x不是关键字参数, 而是默认参数 ''' 5 (1, 3, 6) ''' |
def func(): print(123) def func2(): print(345)
if __name__ == '__main__': func2() # 如果注释掉此行, 那么func2不会运行
func() |
# 例如: 生成验证码 # 画线 # 打点 # 生成图片
def 生成验证码(): def 画线(): pass def 打点(): pass def 生成图片(): pass 画线() 打点() 生成图片() |
函数嵌套使用的目的: 为了保证某些功能特有的函数不被其他人随意地调用 |
# 方法1: def func(x, y, z): print(x, y, z) def func_inner(a,b,c): print('func_inner', a, b, c) func_inner(x, y, z)
func(4,5,6) ''' 4 5 6 func_inner 4 5 6 ''' |
# 方法2(升级版, 使用动态参数) def func(*args, **kwargs): print(args) print(kwargs) def func_inner(a,b,c): print('func_inner', a,b,c) func_inner(*args, **kwargs)
func(1,2, 3) ''' (1, 2, 3) {} func_inner 1 2 3 ''' func(1, b = 2, c = 3) ''' (1,) {'b': 2, 'c': 3} func_inner 1 2 3 ''' |
def fun(): return 123 # Python中, 一遇到return就会停止操作, 不会再执行 return 456 # 不会被执行
def fun(): return (123, 456) # 返回一个元组, 可以返回其它任何数据类型 |
# python返回值的种类: # ① 返回None--a, 如果函数里什么都不写; b, return; c,return None # ② 返回一个值, # ③ 多返回值, 以元组的形式 return 111, {123, 456} # 返回一个元组(111, {123, 456}) # 相当于: return(111, {123, 456}) |
# return的作用; # ① 返回值 # ② 结束函数 |
# ① 一个对象接受一个值 # ② 多个对象接受多个值, 个数要相等(拆包/解包(pack)的问题) |
# 站在函数定义的角度上, 所有的参数都叫形式参数, 简称形参 # 站在函数调用的角度上, 所有的参数都叫实际参数, 简称实参 |
The names given in the function definition are called parameters whereas the values you supply in the function call are called arguments. 在函数定义时赋予的名称(names)叫作形参, 在函数调用时提供的值(values)叫作实参 |
# Parameter: Functions are declared with 0 or more "formal parameters", which are unbound local variables. When the function is called, the argument # Argument: An expression passed to a function when it is called. |
函数的本质: an object resulting from evaluation of a def block or a lambda expression. |