python第十课函数(二)

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/qq_19175749/article/details/80054460

局部变量

<1>什么是局部变量

如下图所示:

 

<2>小总结

局部变量,就是在函数内部定义的变量

不同的函数,可以定义相同的名字的局部变量,但是各用个的不会产生影响

局部变量的作用,为了临时保存数据需要在函数中定义变量来进行存储,这就是它的作用

全局变量

<1>什么是全局变量

如果一个变量,既能在一个函数中使用,也能在其他的函数中使用,这样的变量就是全局变量

demo如下:

 

    # 定义全局变量

    a = 100

 

    def test1():

        print(a)

 

    def test2():

        print(a)

 

    # 调用函数

    test1()

    test2()

运行结果:

 

<2>全局变量和局部变量名字相同问题

看如下代码:

 

<3>修改全局变量

既然全局变量,就是能够在所以的函数中进行使用,那么可否进行修改呢?

代码如下:

 

<4>总结1:

在函数外边定义的变量叫做全局变量

全局变量能够在所有的函数中进行访问

如果在函数中修改全局变量,那么就需要使用global进行声明,否则出错

如果全局变量的名字和局部变量的名字相同,那么使用的是局部变量的,小技巧强龙不压地头蛇

<5>可变类型的全局变量

>>> a = 1

>>> def f():

...     a += 1

...     print a

...

>>> f()

Traceback (most recent call last):

  File "<stdin>", line 1, in <module>

  File "<stdin>", line 2, in f

UnboundLocalError: local variable 'a' referenced before assignment

>>>

>>>

>>> li = [1,]

>>> def f2():

...     li.append(1)

...     print li

...

>>> f2()

[1, 1]

>>> li

[1, 1]

<5>总结2:

在函数中不使用global声明全局变量时不能修改全局变量的本质是不能修改全局变量的指向,即不能将全局变量指向新的数据。

对于不可变类型的全局变量来说,因其指向的数据不能修改,所以不使用global时无法修改全局变量。

对于可变类型的全局变量来说,因其指向的数据可以修改,所以不使用global时也可修改全局变量。

 

函数返回值(二)

python中我们可不可以返回多个值?

>>> def divid(a, b):

...     shang = a//b

...     yushu = a%b

...     return shang, yushu

...

>>> sh, yu = divid(5, 2)

>>> sh5

>>> yu1

本质是利用了元组

 

 


函数参数(二)

1. 缺省参数

调用函数时,缺省参数的值如果没有传入,则被认为是默认值。下例会打印默认的age,如果age没有被传入:

def printinfo( name, age = 35 ):

   # 打印任何传入的字符串

   print "Name: ", name

   print "Age ", age

# 调用printinfo函数

printinfo(name="miki" )

printinfo( age=9,name="miki" )

以上实例输出结果:

Name:  miki

Age  35

Name:  miki

Age  9

注意:带有默认值的参数一定要位于参数列表的最后面。

>>> def printinfo(name, age=35, sex):

...     print name

...

  File "<stdin>", line 1

SyntaxError: non-default argument follows default argument

2.不定长参数

有时可能需要一个函数能处理比当初声明时更多的参数。这些参数叫做不定长参数,声明时不会命名。

基本语法如下: 

 def functionname([formal_args,] *args, **kwargs):

       "函数_文档字符串"

       function_suite

       return [expression]

加了星号(*)的变量args会存放所有未命名的变量参数,args为元组;而加**的变量kwargs会存放命名参数,即形如key=value的参数, kwargs为字典。

>>> def fun(a, b, *args, **kwargs):

...     """可变参数演示示例"""

...     print "a =", a

...     print "b =", b

...     print "args =", args

...     print "kwargs: "

...     for key, value in kwargs.items():

...         print key, "=", value

...

>>> fun(1, 2, 3, 4, 5, m=6, n=7, p=8)  # 注意传递的参数对应

a = 1

b = 2

args = (3, 4, 5)

kwargs:

p = 8

m = 6

n = 7

>>>

>>>

>>>

>>> c = (3, 4, 5)

>>> d = {"m":6, "n":7, "p":8}

>>> fun(1, 2, *c, **d)    # 注意元组与字典的传参方式

a = 1

b = 2

args = (3, 4, 5)

kwargs:

p = 8

m = 6

n = 7

>>>

>>>

>>>

>>> fun(1, 2, c, d) # 注意不加星号与上面的区别

a = 1

b = 2

args = ((3, 4, 5), {'p': 8, 'm': 6, 'n': 7})

kwargs:

>>>

>>>

3. 引用传参

可变类型与不可变类型的变量分别作为函数参数时,会有什么不同吗?

Python有没有类似C语言中的指针传参呢?

>>> def selfAdd(a):

...     """自增"""

...     a += a

...

>>> a_int = 1

>>> a_int

1

>>> selfAdd(a_int)

>>> a_int

1

>>> a_list = [1, 2]

>>> a_list

[1, 2]

>>> selfAdd(a_list)

>>> a_list

[1, 2, 1, 2]

Python中函数参数是引用传递(注意不是值传递)。对于不可变类型,因变量不能修改,所以运算不会影响到变量自身;而对于可变类型来说,函数体中的运算有可能会更改传入的参数变量。

想一想为什么

>>> def selfAdd(a):

...     """自增"""

...     a = a + a   # 我们更改了函数体的这句话

...

>>> a_int = 1

>>> a_int

1

>>> selfAdd(a_int)

>>> a_int

1

>>> a_list = [1, 2]

>>> a_list

[1, 2]

>>> selfAdd(a_list)

>>> a_list

[1, 2]      # 想一想为什么没有变呢?

递归函数

<1>什么是递归函数

通过前面的学习知道一个函数可以调用其他函数。

如果一个函数在内部不调用其它的函数,而是自己本身的话,这个函数就是递归函数。

<2>递归函数的作用

举个例子,我们来计算阶乘 n! = 1 * 2 * 3 * ... * n

解决办法1:

 

看阶乘的规律

1! = 1

2! = 2 × 1 = 2 × 1!

3! = 3 × 2 × 1 = 3 × 2!

4! = 4 × 3 × 2 × 1 = 4 × 3!

...

n! = n × (n-1)!

解决办法2:

 

原理


匿名函数

lambda关键词能创建小型匿名函数。这种函数得名于省略了用def声明函数的标准步骤。

lambda函数的语法只包含一个语句,如下:

    lambda [arg1 [,arg2,.....argn]]:expression

如下实例:

  

sum = lambda arg1, arg2: arg1 + arg2

 

    #调用sum函数

    print "Value of total : ", sum( 10, 20 )

    print "Value of total : ", sum( 20, 20 )

以上实例输出结果:

    Value of total :  30

    Value of total :  40

    Value of total :  40

Lambda函数能接收任何数量的参数但只能返回一个表达式的值

匿名函数不能直接调用print,因为lambda需要一个表达式

应用场合

函数作为参数传递

自己定义函数

>>> def fun(a, b, opt):...     print "a =", a...     print "b =", b...     print "result =", opt(a, b)

...>>> fun(1, 2, lambda x,y:x+y)

a = 1

b = 2

result = 3

作为内置函数的参数

想一想,下面的数据如何指定按age或name排序?

stus = [

    {"name":"zhangsan", "age":18},

    {"name":"lisi", "age":19},

    {"name":"wangwu", "age":17}

]

name排序:

>>> stus.sort(key = lambda x:x['name'])

>>> stus

[{'age': 19, 'name': 'lisi'}, {'age': 17, 'name': 'wangwu'}, {'age': 18, 'name': 'zhangsan'}]

age排序:

>>> stus.sort(key = lambda x:x['age'])

>>> stus

[{'age': 17, 'name': 'wangwu'}, {'age': 18, 'name': 'zhangsan'}, {'age': 19, 'name': 'lisi'}]

 

函数使用注意事项

1. 自定义函数

<1>无参数、无返回值

 def 函数名():

        语句

<2>无参数、有返回值

  def 函数名():

        语句

        return 需要返回的数值

注意:

一个函数到底有没有返回值,就看有没有return,因为只有return才可以返回数据

在开发中往往根据需求来设计函数需不需要返回值

函数中,可以有多个return语句,但是只要执行到一个return语句,那么就意味着这个函数的调用完成

<3>有参数、无返回值

def 函数名(形参列表):

        语句

注意:

在调用函数时,如果需要把一些数据一起传递过去,被调用函数就需要用参数来接收

参数列表中变量的个数根据实际传递的数据的多少来确定

<4>有参数、有返回值

    def 函数名(形参列表):

        语句

        return 需要返回的数值

<5>函数名不能重复

2. 调用函数

<1>调用的方式为:

    函数名([实参列表])

<2>调用时,到底写不写 实参

如果调用的函数 在定义时有形参,那么在调用的时候就应该传递参数

<3>调用时,实参的个数和先后顺序应该和定义函数中要求的一致

<4>如果调用的函数有返回值,那么就可以用一个变量来进行保存这个值

3. 作用域

<1>在一个函数中定义的变量,只能在本函数中用(局部变量)

<2>在函数外定义的变量,可以在所有的函数中使用(全局变量)

猜你喜欢

转载自blog.csdn.net/qq_19175749/article/details/80054460