Python代码重用之函数入门
函数是组织好的,可重复使用的,用来实现单一,或相关联功能的代码段。
在开发程序时,使用函数可以提高编写的效率以及代码的重用。
一、函数的基本使用
函数的使用就是定义和调用两个步骤。
函数的定义
定义函数的格式如下:
def 函数名():
    函数封装的代码
    ……
说明:
1、def 是英文 define 的缩写。
2、函数名称应该能够表达函数封装代码的功能,方便后续的调用。
3、函数名称的命名应该符合标识符的命名规则。
- 可以由字母、下划线和数字组成
- 不能以数字开头
- 不能与关键字重名
函数的调用
调用函数很简单的,通过 函数名() 即可完成对函数的调用。
实例:
# 定义函数
def say_hello():
    print("hello 1")
    print("hello 2")
    print("hello 3")
    
# 只有在调用函数时,之前定义的函数才会被执行
# 调用函数
say_hello()
定义好函数之后,只表示这个函数封装了一段代码而已,如果不主动调用函数,函数是不会主动执行的。
那么能否将函数调用放在函数定义的上方呢?
答案是不能的!因为在使用函数名调用函数之前,必须要保证 Python 已经知道函数的存在,否则控制台会提示报错信息:
NameError: name 'xxx' is not defined
名称错误:xxx 这个名字没有被定义
函数的文档注释
在开发中,如果希望给函数添加注释,应该在定义函数的下方,使用连续的三对引号。
在连续的三对引号之间编写对函数的说明文字。
因为函数体相对比较独立,函数定义的上方,应该和其他代码(包括注释)保留两个空行。
二、函数的参数
在函数名的后面的小括号内部填写参数,多个参数之间使用 ',' (逗号)分隔。
实例
def sum_2_num(num1, num2):
    result = num1 + num2
    print("%d + %d = %d" % (num1, num2, result))
sum_2_num(50, 20)
形参和实参
形参:定义函数时,小括号中的参数,是用来接收参数用的,在函数内部作为变量使用。
实参:调用函数时,小括号中的参数,是用来把数据传递到函数内部用的。
1、不可变和可变的参数
无论传递的参数是可变还是不可变的,只要针对参数使用赋值语句,只会在函数内部修改局部变量的引用,不会影响到外部变量的引用。
def demo(num, num_list):
    # 赋值语句
    num = 200
    num_list = [1, 2, 3]
    print(num)  # 200
    print(num_list)  # [1,2,3]
gl_num = 99
gl_list = [4, 5, 6]
demo(gl_num, gl_list)
print(gl_num)  # 99
print(gl_list)  # [4,5,6]
但是如果传递的参数是可变类型,并且在函数内部,使用方法修改了数据的内容,同样会影响到外部的数据。
def mutable(num_list):
    num_list.extend([1, 2, 3])
    print(num_list)  # [6, 7, 8, 1, 2, 3]
gl_list = [6, 7, 8]
mutable(gl_list)
print(gl_list)  # [6, 7, 8, 1, 2, 3]
在 python 中,列表变量调用 += 本质上是在执行列表变量的 extend 方法,不会修改变量的引用。
def demo(num, num_list):
    num += num
    # 函数执行结束后,外部数据同样会发生变化
    num_list += num_list
    print(num)  # 18
    print(num_list)  # [1, 2, 3, 1, 2, 3]
gl_num = 9
gl_list = [1, 2, 3]
demo(gl_num, gl_list)
print(gl_num)  # 9
print(gl_list)  # [1, 2, 3, 1, 2, 3]
2、缺省参数
定义函数时,可以给某个参数指定一个默认值,具有默认值的参数就叫做缺省参数。
调用函数时,如果没有传入缺省参数的值,则在函数内部使用定义函数时指定的参数默认值。
函数的缺省参数,将常见的值设置为参数的缺省值,从而简化函数的调用。
例如:对列表排序的方法
gl_num_list = [6, 3, 9]
# 默认就是升序排序,因为这种应用需求更多
gl_num_list.sort()
print(gl_num_list)
# 当需要降序排序时,才需要传递 reverse 参数
gl_num_list.sort(reverse=True)
print(gl_num_list)
指定函数的缺省参数
在参数后使用赋值语句,可以指定参数的缺省值。
def print_info(name, gender=True):
    gender_text = "男生"
    if not gender:
        gender_text = "女生"
    print("%s 是 %s" % (name, gender_text))
注意事项:
1、缺省参数,需要使用最常见的值作为默认值!
2、必须保证带有默认值的缺省参数在参数列表末尾。
3、在调用函数时,如果有多个缺省参数,需要指定参数名,这样解释器才能够知道参数的对应关系!
def print_info(name, title="", gender=True):
    """
    :param title: 职位
    :param name: 班上同学的姓名
    :param gender: True 男生 False 女生
    """
    
    gender_text = "男生"
    if not gender:
        gender_text = "女生"
    print("%s%s 是 %s" % (title, name, gender_text))
print_info("小明")
print_info("老王", title="班长")
print_info("小美", gender=False)
# 小明 是 男生
# 班长老王 是 男生
# 小美 是 女生
3、多值参数
有时可能需要一个函数能够处理的参数个数是不确定的,这个时候,就可以使用多值参数。
python 中有两种多值参数:
参数名前增加一个 * 可以接收元组,参数名前增加两个 * 可以接收字典。
一般在给多值参数命名时,习惯使用以下两个名字:
*args,存放元组参数,**kwargs 存放字典参数。
args 是 arguments 的缩写,有变量的含义。
kw 是 keyword 的缩写,kwargs 可以记忆成键值对参数。
def demo(num, *args, **kwargs):
    print(num)
    print(args)
    print(kwargs)
demo(1, 2, 3, 4, 5, name="小明", age=18, gender=True)
# 1
# (2, 3, 4, 5)
# {'name': '小明', 'age': 18, 'gender': True}
多值参数实例:计算任意多个数字的和。
def sum_numbers(*args):
    num = 0
    for n in args:
        num += n
    return num
print(sum_numbers(1, 2, 3))  # 6
三、函数的返回值
在程序开发中,有时候会希望 一个函数执行结束后,告诉调用者一个结果,以便调用者针对具体的结果做后续的处理。
返回值是函数完成工作后,最后给调用者的一个结果。
在函数中使用 return 关键字可以返回结果。
调用函数一方,可以使用变量来接收函数的返回结果。
注意:return表示返回,函数内部在 return 后面的代码都不会被执行。
实例:
def sum_2_num(num1, num2):
    """对两个数字的求和"""
    return num1 + num2
# 调用函数,并使用 result 变量接收计算结果
result = sum_2_num(10, 20)
print("计算结果是 %d" % result)
那么问题来了,一个函数执行后能否返回多个结果么?
在 Python 中,可以将一个元组使用赋值语句同时赋值给多个变量。
注意:变量的数量需要和元组中的元素数量保持一致。
实例:
def measure():
    """测量温度和湿度"""
    print("测量开始...")
    temp = 39
    wetness = 50
    print("测量结束...")
    # 如果函数返回的类型是元组,小括号可以省略
    # return (temp, wetness)
    return temp, wetness
# 如果函数返回的类型是元组,同时希望单独的处理元组中的元素
# 可以使用多个变量,一次接收函数的返回结果
gl_temp, gl_wetness = measure()
print(gl_temp)
print(gl_wetness)
四、局部变量和全局变量
局部变量是在函数内部定义的变量,只能在函数内部使用。
全局变量是在函数外部定义的变量(没有定义在某一个函数内),所有函数内部都可以使用这个变量。
在其他的开发语言中,大多不推荐使用全局变量——可变范围太大,导致程序不好维护!
但是 Python 使用全局变量的地方比较多,源码作了特定的处理,可以更方便地使用全局变量。
1、局部变量
函数执行结束后,函数内部的局部变量,会被系统回收。
不同的函数,可以定义相同的名字的局部变量,但是彼此之间不会产生影响。
在函数内部使用,临时保存函数内部需要使用的数据。
局部变量的生命周期
所谓生命周期就是变量从被创建到被系统回收的过程。
局部变量在函数执行时才会被创建,函数执行结束后局部变量被系统回收。
2、全局变量
函数执行时,需要处理变量时会:
首先查找函数内部是否存在指定名称的局部变量,如果有,直接使用,如果没有,查找函数外部是否存在指定名的全局变量,如果有,直接使用,如果还没有,程序报错!
# 定义一个全局变量
num = 10
def demo1():
    print(num)
demo1()
# 10
在函数内部,可以通过全局变量的引用获取对应的数据,但是,不允许直接修改全局变量的引用,即通过普通的赋值语句不能修改全局变量的值。
# 定义一个全局变量
num = 10
def demo1():
    # 在python中,是不允许直接修改全部变量的值
    # 如果使用赋值语句,会在函数内部,定义一个局部变量
    num = 99
    print(num)
def demo2():
    print(num)
demo1()  # 99
demo2()  # 10
那如何才能在函数内部修改全局变量的值呢?
如果在函数中需要修改全局变量,需要使用 global 进行声明。
# 定义一个全局变量
num = 10
def demo1():
    # 使用 global 声明变量可以修改全局变量的值
    global num
    num = 99
    print(num)
def demo2():
    print(num)
demo1()  # 99
demo2()  # 99
为了保证所有的函数都能够正确使用到全局变量,应该将全局变量定义在其他函数的上方。

 
                    
                     
                    
                 
                    
                
 
                
            
         
         浙公网安备 33010602011771号
浙公网安备 33010602011771号