函数

函数

什么是函数

在程序中,函数就是具备某一功能的工具,事先将工具准备好就是函数的定义,遇到应用场景拿来就
用就是函数的调用。

为何用函数

如果不使用函数,写程序时将会遇到这三个问题:

  • 程序冗长
  • 程序的扩展性差
  • 程序的可读性差

如何用函数

先定义函数,后调用。

  • 定义函数:
def 函数名(param1、param2....):
    """
    函数功能的描述信息
    :param1:描述
    :param2:描述
    :param3:返回值
    """
    code 1
    code 2
    code 3

    ...

    return 返回值
  • 调用函数
函数名(param1, param2...)

注册功能函数

# 注册功能函数
def register():
    """注册功能"""
    username = input('username: ').strip()
    pwd = input('password: ').strip()

    with open(r'E:孔	est	est.txt', 'a', encoding='utf8') as fa:
        fa.write(f"{username}:{pwd}
")
        fa.flush()


register()
# 复用
register()
register()

登陆功能函数

# 登陆功能函数
def login():
    """登陆功能"""
    inp_username = input('username: ').strip()
    inp_pwd = input('password: ').strip()

    with open('E:孔	est	est.txt', 'rt', encoding='utf8') as fr:
        for user_info in fr:
            user_info = user_info.strip('
')
            user_info_list = user_info.split(':')
            if inp_username == user_info_list[0] and inp_pwd == user_info_list[1]:
                print('login successful')
                break
        else:
            print('failed')


login()

函数定义阶段

def func():
    bar()  # 不属于语法错误,不会报错
    print('*'*10)

1.只检测语法,不执行函数体代码

函数调用阶段

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


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


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


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


foo()
'''
from foo
from bar
'''

1.执行函数代码

定义函数的三种形式

  • 无参函数
  • 有参函数
  • 空函数

无参函数

定义函数时参数是函数体接收外部传值的一种媒介,其实就是一个变量名

在函数阶段括号内没有参数,称为无参函数。需要注意的是:定义时无参,意味着调用时也无需传入参数。

如果函数体代码逻辑不需要依赖外部传入的值,必须得定义成无参函数。

def func():
    print('hello wiliiam')


func()  # hello william

有参函数

在函数定义阶段括号内有参数,称为有参函数。需要注意的是:定义时有参,意味着调用时必须传入参数。

如果函数体代码逻辑需要依赖外部传入的值,必须得定义成有参函数。

def sum_self(x, y):
    '''求和'''
    res = x + y
    print(res)


sum_self(1, 2)  # 3

空函数

当你只知道你需要实现某个功能,但不知道该如何用代码实现时,你可以暂时写个空函数,然后先实现其他的功能。

def func():
    pass

函数的返回值

什么是返回值

函数内部代码经过一系列逻辑处理获得的结果。

def func():
    name = 'william'
    return name


name = func()
print(name)
william

为什么要有返回值

如果需要在程序中拿到函数的处理结果做进一步处理,则需要函数必须有返回值。

注意

  • return是一个函数结束的标志,函数内可以有多个return,只要执行到return,函数就会停止

  • return的返回值可以返回任意数据类型

  • return的返回值无个数限制,即可以使用逗号隔开返回多个值

    0个:返回None
    
    1个:返回值是该值本身
    
    多个:返回值是元组
    
# 为什么要有返回值
def max_self(salary_x, salary_y):
    if salary_x > salary_y:
        return salary_x
    else:
        return salary_y


max_salary = max_self(20000, 30000)
print(max_salary*12)
360000
# 函数返回多个值
def func():
    name = 'william'
    age = 18
    hobby_list = ['running', 'reading', 'music']
    return name, age, hobby_list


name, age, hobby_list = func()
print(f"name,age,hobby_list:{name,age,hobby_list}")
name,age,hobby_list:('william', 18, ['running', 'reading', 'music'])

函数的调用

什么是函数调用

函数名(...)即调用函数,会执行函数体代码,直到碰到return或者执行完函数体内所有代码结束。

函数运行完毕所有代码,如果函数体不写return,则会返回None.

def foo():
    pass


print(foo())
None

为何调用函数

使用函数的功能

函数调用的三种形式

def max_self(x, y):
    if x > y:
        return x
    else:
        return y


# 1.
max_self(1, 2)
# 2.
res = max_self(1, 2)*12
# 3.
max_self(max_self(20000, 30000), 40000)

函数参数的应用

形参和实参

形参

在函数定义阶段括号内定义的参数,称之为形式参数,简称形参,本质就是变量名。

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

实参

在函数调用阶段括号内传入的参数,称之为实际参数,简称实参,本质就是变量的值。

func(1, 2)

位置参数

位置形参

在函数定义阶段,按照从左到右的顺序依次定义的形参,称之为位置形参。

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

特点:按照位置定义的形参,都必须被传值,多一个不行,少一个也不行。

位置实参

在函数调用阶段,按照从左到右的顺序依次定义的实参,称之为位置实参。

func(1, 2)

特点:按照位置为对应的形参依次传值。

关键字实参

在调用函数时,按照key=value的形式为指定的参数传值,称为关键字实参。

func(y=2, x=1)

特点:可以打破位置的限制,但仍能为指定的形参赋值。

注意:

1.可以混用位置实参和关键字实参,但是位置实参必须在关键字实参的左边。

2.可以混用位置实参和关键字实参,但不能对一个形参重复赋值。

func(1, y=2)
func(y=2, x)  # SyntaxError: positional argument follows keyword argument
func(x, x=1)

默认形参

在定义阶段,就已经被赋值。

def func(x, y=10):
    print(x)
    print(y)


func(3)
func(2, 9)
3
10
2
9

特点:在定义阶段就已经被赋值,意味着在调用时可以不用为其赋值。

注意:

1.位置形参必须放在默认形参的左边。

2.默认形参的值只在定义阶段赋值一次,就是说默认参数的值在函数定义阶段就已经固定了。

m = 10


def foo(x=m):
    print(x)


m = 111
foo()  # 10
foo(15)
10
15

默认参数的值通常应该是不可变类型。

# 演示形参是可变类型
def register(name, hobby, hobby_list=[]):
    hobby_list.append(hobby)
    print(f'{name} prefer {hobby}')
    print(f'{name} prefer {hobby_list}')


register('william', 'reading')
register('tom', 'fish')
register('jerry', 'juice')
william prefer reading
william prefer ['reading']
tom prefer fish
tom prefer ['reading', 'fish']
jerry prefer juice
jerry prefer ['reading', 'fish', 'juice']
# 修改形参是可变类型代码
def register(name, hobby, hobby_list=None):
    if hobby_list is None:
        hobby_list = []
    hobby_list.append(hobby)
    print(f"{name} prefer {hobby}")
    print(f"{name} prefer {hobby_list}")


register('william', 'reading')
register('tom', 'fish')
register('jerry', 'juice')
william prefer reading
william prefer ['reading']
tom prefer fish
tom prefer ['fish']
jerry prefer juice
jerry prefer ['juice']

总结

实参的应用:取决于个人习惯

形参的应用:

1.大多数情况的调用值不一样,就应该将该参数定义成位置形参。

2.大多数情况的调用值一样,就应该将该参数定义成默认形参。

可变长参数

可变长参数:指的是在调用函数时,传入的参数个数可以不固定。

调用函数时,传值的方式无非两种,一种是位置实参,另一种是关键字实参,因此形参也必须得有两种解决方法,以此来分别接收溢出的位置实参(*)与关键字实参(**)。

可变长形参之*

形参中的*会将溢出的位置实参全部接收,然后存储元组形式,然后把元组赋值给*后的参数。需要注意的是:*后的参数名约定俗成为args。

def sum_self(*args):
    res = 0
    for num in args:
        res += num
    return res


res = sum_self(1, 2, 3, 4)
print(res)
10

可变长实参之*

实参中的*,*会将*后的参数的值循环取出,打散成位置实参。以后但凡碰到实参中带*的,它就是位置实参,应该马上打散成位置实参去看。

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


func(1, *(1, 2), 3, 4)
1 1 2 (3, 4)

可变长形参之**

形参中的**会将溢出的关键字实参全部接收,然后存储字典的形式,然后把字典赋值给**后的参数。需要注意的是:**后的参数名约定俗成为kwargs。

def func(**kwargs):
    print(kwargs)


func(a=5, c=6)
{'a': 5, 'c': 6}

可变长实参之**

实参中的**,**会将**后的参数的值循环取出,打散成关键字实参。以后但凡碰到实参中带**的,它就是关键字实参,应该马上打散成关键字实参去看。

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


func(1, 3, 4, c=2, **{'a': 1, 'b': 2})
1 3 4 {'c': 2, 'a': 1, 'b': 2}

可变长参数应用

def index(name, age, sex):
    print(f'name:{name},age:{age},sex:{sex}')


def wrapper(*args, **kwargs):
    print(f'args:{args}')
    print(f'kwargs:{kwargs}')
    index(*args, **kwargs)


wrapper('william', sex='male', age=18)
wrapper(name='william', sex='male', age=18)
args:('william',)
kwargs:{'sex': 'male', 'age': 18}
name:william,age:18,sex:male
args:()
kwargs:{'name': 'william', 'sex': 'male', 'age': 18}
name:william,age:18,sex:male

命名关键字形参

现在有一个需求:函数的使用者必须按照关键字实参传。

def register(x, y, **kwargs):
    if 'name' not in kwargs or 'age' not in kwargs:
        print('用户名和年龄必须使用关键字的形式传值')
        return
    print(kwargs['name'])
    print(kwargs['age'])


register(1, 2, name='william', age=18)
william
18

命名关键字形参:在函数定义阶段,**后面的参数都是命名关键字参数。

特点:在传值时,必须按照key=value的方式传值,并且key必须命名关键字参数的指定的参数名。

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


register(1, 2, x='william', age=19)
# TypeError: register() got multiple values for argument 'x'
原文地址:https://www.cnblogs.com/WilliamKong94/p/10945665.html