Python-函数基础

什么是函数

具备某一功能的工具

对于工具的使用,我们应该先准备好工具,然后再使用。即我们对函数应该先定义后引用。和变量一样。

为什么用函数

1、程序的组织结构不清晰,可读性差
2、如果要用到重复的功能,只能拷贝功能的实现代码=》代码冗余
3、可扩展性差

如何使用函数

函数使用原则:先定义后引用

# 定义函数:
def 函数名(参数1,参数2,参数3,...):
            """文档注释"""
            代码1
            代码2
            代码3
            ...
            return 返回值

# 调用函数:
函数名(值1,值2,值3,...)

函数的基本使用

# 1、定义函数:申请内存空间把函数体代码保存下来,然后把内存地址绑定给函数名-》函数名=函数的内存地址
def sayhi():
    print('*'*10)
    print('hello')
    print('*'*10)

print(sayhi)  # 函数名本质是一串内存地址,注意下方开头是function,表明是一个函数。
---------------------
<function hello at 0x7fdc6a5ba1f0>


# 2、调用函数: 函数名()=> 函数的内存地址(),会触发函数体代码的运行
sayhi()

定义函数的三种格式

# 2.1: 无参函数,即函数本身不需要参数传入
def login():
    inp_name=input("your name: ").strip()
    inp_pwd=input("your pwd: ").strip()
    if inp_name == "yang" and inp_pwd == "123":
        print('login successful')
    else:
        print('login error')

login()

# 2.2: 有参函数,函数内需要外部参数传入
def max2(x,y):
    if x > y:
        print(x)
    else:
        print(y)
max2(10,20)
max2(11,22)


'''2.3: 空函数,空函数在项目初期,有些功能还没做完时,可以先用空函数顶替,防止程序无法运行'''
def func():
    pass

函数的返回值

利用return 可以将函数内处理之后的结果返回

# 函数内可以有多个return,但只要执行一次函数就立刻结束,并会把return后的值当作本次调用的结果返回
'''
函数可以有三种形式的返回值
1、return 值:返回的就是该值本身
2、return 值1,值2,值3:返回一个元组
3、没有return:默认返回None
'''
def max2(x, y):
    if x > y:
        return x
    else:
        return y


res = max2(1, 2)
print(res)
--------------------
2

函数调用的三种格式

# 语句形式:单纯地调用一下函数就完了
def hello(s,n):
    print(s*n)
    print('hello')
    print(s*n)

hello('*',30)

# 表达式形式:
def max2(x,y):
    if x > y:
        return x
    else:
        return y

res=max2(11,22) * 12
print(res)

#可以把函数的调用当作值传给另外一个函数
print(max(33,max2(11,22)))

总结

'''
函数的使用一定要分两个阶段去看:
1、定义阶段:只检测语法,不执行代码
2、调用阶段:执行函数体代码
'''
# 如果发生的语法错误,定义阶段就会立马检测出来
def func():
    print("hello"  # 直接报错

# 如果发生的不是语法错误,而是逻辑错误,只能在调用阶段检测到
def func():
    xxx
          
func()  # 报错

三、函数参数

函数的参数分为两大类:形参与实参

# 1、形参:在定义函数时,括号内定义的变量名,称之为形式参数,简称形参=》变量名
def func(x,y):  # x,y就是形参
    x=1
    y=2
print(x)
print(y)


# 2、实参:在调用函数时,括号内传入的值,称之为实际参数,简称实参=》变量值
func(1,2)

在python中参数的种类

'''
1、位置参数:
(1)位置形参:在函数定义阶段按照从左到右的顺序依次定义形参(变量名),称之为位置形参
 特点:必须被传值
'''
def func(x,y):
    print(x,y)

func(1,2)
func(1) # 少一个不行
func(1,2,3) # 多一个也不行

'''
(2)位置实参:在函数调用阶段按照从左到右的顺序依次定义实参(传入的变量值),称之为位置实参
特点:按照位置传值,一一对应
'''
def func(x1,x2,x3,x4,x5,x6):
    print(x1,x2,x3,x4,x5,x6)

func(1,2,3,4,5,6)

'''
2、关键字实参:在函数调用阶段按照key=value的形式为指定的形参名传值,该形式称之为关键字实参
特点:在传值时可以完全打乱顺序,但是仍然能够指名道姓地为指定的形参传值
'''
def func(name, age):
    print(name, age)

func("yang",18)
func(18,"yang")
func(age=18,name="yang")
-----------------------
yang 18
18 yang
yang 18


'''
注意:可以混用位置实参与关键字实参,但是
1 位置实参必须放在关键字实参的前面
2 不能为同一个形参重复赋值
'''
def func(name, age, salary):
    print(name,age,salary)


func('yang',salary=3.1,age=18)
# func('yang',salary=3.1,18) # 错误
func('yang', 18, salary=3.1)
# func('yang',18,age=19,salary=3.3) # 错误
-------------------------------
yang 18 3.1
yang 18 3.1


'''
3、默认形参:在函数定义阶段就已经为某个形参赋值,该形参称之为有默认值的形参,简称默认形参
特点: 定义阶段就已经被赋值意味着在函数调用阶段可以不用为其赋值
'''
def func(x,y=2):
    print(x,y)

func(1)  # 不会报错,因为有默然形参,如果没有传值,那么y默认为2
func(1,33333)
'''
注意:
1 默认形参应该放在位置形参的后面
'''
def func(y=2,x): # 错误
    pass
'''


2 默认形参的值通常应该是不可变类型,因为若值是可变类型,我们获取到的是一个id地址,传入值的时候也是通过id,就会导致传的每个值都通过这个id存值,使得所有值都放在了一起
'''
def func(name,hobby,hobbies=[]):
    hobbies.append(hobby)
    print("%s 的爱好是:%s" %(name,hobbies))

func("yang",'play')
func('yang','music')
func("egon",'read')
---------------------------------------
yang 的爱好是:['play']
yang 的爱好是:['play', 'music']
egon 的爱好是:['play', 'music', 'read']
'''
要实现以上单独存的功能,应该把hobbies放到函数里面定义
'''
def func(name,hobby,hobbies=None):
    if hobbies is None:
        hobbies=[]
    hobbies.append(hobby)
    print("%s 的爱好是:%s" %(name,hobbies))

func("yang",'play')
func('yang','music')
func("egon",'read')
------------------------------------
yang 的爱好是:['play']
yang 的爱好是:['music']
egon 的爱好是:['read']
'''
3 默认形参的值只在函数定义阶段被赋值一次,函数定义之后的改变对默认形参没有影响
'''
m=333
def func(x,y=m): # y=333
    print(x,y)

m=444
func(1)
-------------------
1 333


def register(name,age,gender='male'):
    print(name)
    print(age)
    print(gender)

register('yang',18,)
register('jack',20,)
register('mary',18,'female')

四:可变长参数

可变长指的是参数的个数不固定
站在实参的角度,实参是用来为形参赋值的,如果实参的个数不固定,那么必须要有对应的形参能够接收溢出实参
*与**在形参与实参中的应用
在形参中用*与**

在形参名前加 *

* 会把溢出的位置实参存成元组,然后赋值其后的形参名

def func(x,*y):  # y=(2,3,4)
    print(x)
    print(y)


func(1,2,3,4)
func(1)
----------------
1
(2, 3, 4)
1
()
# func() # 位置形参x必须被传值

在形参名前加**

**会把溢出的关键字实参存成字典,然后赋值其后的形参名

def func(x, **y):  # y=(2,3,4)
    print(x)
    print(y)


func(1,a=111,b=222,c=333)
func(a=111, b=222, x=1, c=333)
----------------------------------
1
{'a': 111, 'b': 222, 'c': 333}
1
{'a': 111, 'b': 222, 'c': 333}

在实参前加*

*会把其后的值打散成位置实参

def func(x,y,z):
    print(x,y,z)

nums = [1, 2, 3]
func(*nums)  # func(1,2,3)
# 若只有一个形参,则多余出来的值没处放,会出错
-------------------
1 2 3

在实参前加 **

**会把其后的值打散关键字实参

def func(x,y,z):
    print(x, y, z)

dic = {'y': 111, 'z': 222, 'x': 333}
func(**dic)  # func(y=111,z=222,x=333)
------------------------
333 111 222

*args与**kwargs一起使用

'''
同时使用*与**的作用是可以把外部传的值原封不动的传给另一个函数,这在闭包函数,装饰器中有很大作用
'''
def index(x,y,z,a,b,c):
    print("index===>",x,y,z,a,b,c)

def wrapper(*args, **kwargs):  
  # args=(1, 2, 3,)   kwargs={"a":111,"b":222,"c":333}
    index(*args, **kwargs)  
    # index(*(1, 2, 3,),**{"a":111,"b":222,"c":333})
    # index(1,2,3,c=333,b=222,a=111)

wrapper(1, 2, 3, a=111, b=222, c=333)
---------------------------------
index===> 1 2 3 111 222 333

五、函数对象

函数对象指的是函数可以被当成变量去使用

def foo():  # foo = 函数的内存地址
    print('from foo')

可以被赋值

f = foo
print(f is foo)
f()

可以当作参数传给一个函数

def bar(func):
    print(func)
    func()

bar(foo)

可以当成一个函数的返回值

def bar(func):
    return func

res=bar(foo)
print(res)

可以当成容器类型的元素

在之前学习if判断时候,我们举过这个例子,当有大量if判断的分支且结果是执行函数时,可以用字典。

l = [foo]

print(l)
l[0]()

示例:当我们的主函数是由多个其他函数组成,即我们需要使用许多的if来判断用户输入且调用对应的函数时候,可以使用这种容器的方法,使得代码量减少,同时,在面试时,若被问到如何减少if操作时,可使用字典替换

def login():
    print('登录功能......')

def withdraw():
    print('提现功能......')

def transfer():
    print('转账功能......')

def recharge():
    print('充值功能')

func_dic={
    "1": [login,"登录"],
    "2": [withdraw,"提现"],
    "3": [transfer,"转账"],
    "4": [recharge,"充值"]
}

while True:
    print("0    退出")
    for k in func_dic:
        print("%s    %s" %(k,func_dic[k][1]))

    choice = input("请输入你的指令编号: ").strip()
    if choice == "0":
        break
    if choice in func_dic:
        func_dic[choice][0]()
    else:
        print('输入的指令不存在')

六、函数嵌套

函数的嵌套调用

def bar():
    print('from bar')

def foo():
    print('from foo')
    bar()

foo()
---------------------
from foo
from bar
def max2(x,y):
    if x > y:
        return x
    else:
        return y

def max4(a,b,c,d):
    res1 = max2(a,b)
    res2 = max2(res1,c)
    res3 = max2(res2,d)
    print(res3)

max4(1,2,3,4)
-----------------
4

函数的嵌套定义

def f1():
    print('from f1')
    def f2():
        print("from f2")
f1()  # 此时不会触发f2运行
-----------------
from f1

定义在函数内的函数特点是: 正常情况只能在函数体内调用

from math import pi

def circle(radius,mode=0):
    def perimiter(radius):
        return 2 * pi * radius

    def area(radius):
        return pi * (radius ** 2)

    if mode == 0:
        return perimiter(radius)
    elif mode == 1:
        return area(radius)

res1=circle(3,0)
res2=circle(3,1)
print(res1)
print(res2)
-------------------------
18.84955592153876
28.274333882308138


def func():
    x = 10
    print(x)
    def f2():
        print('from f2')
    f2()
func()
print(x)
---------------------------------
10
from f2
NameError: name 'x' is not defined

七、名称空间与作用域

名称空间

名称空间在python中有非常重要的地位,在python之禅中专门提到了名称空间,在python解释器中输入 impoet this 可以看到python之禅.

Namespaces are one honking great idea -- let's do more of those!

名称空间就是存放名字的地方
名(函数名、变量名等):存放在内存里
值(变量值等):存放在内存里
通过划分出各个名字空间,允许在不同的空间内,可以有相同的名字

内置名称空间

存放内置的名字(python解释器启动时便定义好的)如 print、input、len
生命周期:解释器启动则产生,解释器关闭则销毁 (python解释器在pyhton程序执行时运行,执行结束时结束)

全局名称空间

存放的是顶级的名字

生命周期:python程序运行时则产生,python程序结束则销毁

局部名称空间

即函数内的名字
只有调用函数时,局部名字才产生,定义函数时并不会产生,因为代码并未执行,python解释器只会扫描是否有语法错误。

生命周期:调用函数时产生,函数调用结束则销毁

名字的查找优先级

从当前位置往外查找,如果当前是在局部:局部名称空间->全局名称空间->内置名称空间
从当前位置往外查找,如果当前是在全局:全局名称空间->内置名称空间

内置名称空间

存放的是内置的名字,如print,input,len等等
生命周期: 解释器启动则产生,解释器关闭则销毁

全局名称空间

存放的是顶级的名字
生命周期: python程序运行时则产生,python程序结束则销毁

一般来说,没有缩进的名字都是顶级名字,但是如果这个名字在if分支里,他也属于顶级名字

x = 10

def func():
    x = 111
    print(x)


if 1:
    y = 6666  # y也是顶级名字

局部名称空间

函数内的名字

生命周期: 调用函数时则产生,函数调用结束则销毁

名字的查找优先级:
从当前位置往外查找,如果当前是在局部:局部名称空间->全局名称空间->内置名称空间
从当前位置往外查找,如果当前是在全局:全局名称空间->内置名称空间

def func():
    len = 222
    # print(len)

# len = 111

func()
print(len)

# 名称空间可以理解为一层套一层的关系,问题是嵌套关系是在函数定义阶段生成的,还是在函数调用阶段生成的?

x = 111

def foo():
    print(x)

def bar(f):
    x=222
    f()

bar(foo)
------------
111

一个非常重要的结论

名称空间的嵌套关系是函数定义阶段(即扫描语法时)就固定死的,与函数的调用位置无关

x = 111
def func():
    print(x)
    x=2222

func()
'''
以上这个例子会报错,因为python解释器在扫描语法时发现需要打印x,但是x定义在print后面,
本来如果func中不写x=2222,那么func会到外面去找111,但是在函数中写了,那么会优先用
自己的,但是又因为还未定义,所以会报错
'''
# 练习
x=0
def f1():
    x=1
    def f2():
        x=2
        print(x)

    f2()
f1()
---------
2

全局范围/全局作用域对应内置名称空间+全局名称空间
特点:全局存活,全局有效

局部范围/局部作用域对应局部名称空间
特点:临时存活,局部有效

案例1

x = 10

def func(x):
    x = 20

func(x)
print(x)
--------------
10

案例2

x = [11,22,33]

def func(x): 
    x[0] = 66

func(x)
print(x)
--------------
[66,22,33]

案例3

x = [11,22,33]
def func():
    x[0] = 66

func()
print(x)
-------------
[66,22,33]

案例4

x = 10
def func():
    global x
    x=22

func()
print(x)
----------
22

案例5:nonlocal生命名字是来自于外层函数的(了解)

x = 10


def f1():
    x = 111

    def f2():
        nonlocal x
        x = 222

    f2()
    print(x)


f1()
print(x)
-------------
222
10
posted @ 2020-12-01 09:21  王寄鱼  阅读(105)  评论(0编辑  收藏  举报