首页 > 代码库 > 第七篇 python基础之函数,递归,内置函数

第七篇 python基础之函数,递归,内置函数

阅读目录

一 数学定义的函数与python中的函数

二 为何使用函数

背景提要

三 函数和过程

四 函数参数

五 局部变量和全局变量

六 前向引用之‘函数即变量‘

七 嵌套函数和作用域

八 递归

九 匿名函数

十 函数式编程

十一 内置函数

十二 本节作业

 

一 数学定义的函数与python中的函数

 

初中数学函数定义:一般的,在一个变化过程中,如果有两个变量x和y,并且对于x的每一个确定的值,y都有唯一确定的值与其对应,那么我们就把x称为自变量,把y称为因变量,y是x的函数。自变量x的取值范围叫做这个函数的定义域

 

例如y=2*x

 

 

 

python中函数定义:函数是逻辑结构化和过程化的一种编程方法。

 

 

  python中函数定义方法:

   

  def test(x):

      "The function definitions"

      x+=1

      return x

      

 def:定义函数的关键字

 test:函数名

():内可定义形参

"":文档描述(非必要,但是强烈建议为你的函数添加描述信息)

x+=1:泛指代码块或程序处理逻辑

return:定义返回值

 

 

调用运行:可以带参数也可以不带

函数名()

 

补充:

 

1.编程语言中的函数与数学意义的函数是截然不同的俩个概念,编程语言中的函数是通过一个函数名封装好一串用来完成某一特定功能的逻辑,数学定义的函数就是一个等式,等式在传入因变量值x不同会得到一个结果y,这一点与编程语言中类似(也是传入一个参数,得到一个返回值),不同的是数学意义的函数,传入值相同,得到的结果必然相同且没有任何变量的修改(不修改状态),而编程语言中的函数传入的参数相同返回值可不一定相同且可以修改其他的全局变量值(因为一个函数a的执行可能依赖于另外一个函数b的结果,b可能得到不同结果,那即便是你给a传入相同的参数,那么a得到的结果也肯定不同)

 

2.函数式编程就是:先定义一个数学函数(数学建模),然后按照这个数学模型用编程语言去实现它。至于具体如何实现和这么做的好处,且看后续的函数式编程。

 

 

二 为何使用函数

 

 

背景提要

 

现在老板让你写一个监控程序,监控服务器的系统状况,当cpu\memory\disk等指标的使用量超过阀值时即发邮件报警,你掏空了所有的知识量,写出了以下代码

 

 

 while True:

     if cpu利用率 > 90%:

         #发送邮件提醒

         连接邮箱服务器

         发送邮件

         关闭连接

      

     if 硬盘使用空间 > 90%:

         #发送邮件提醒

        连接邮箱服务器

        发送邮件

        关闭连接

     

    if 内存占用 > 80%:

        #发送邮件提醒

        连接邮箱服务器

        发送邮件

         关闭连接

 

 

 

 

def 发送邮件(内容)

    #发送邮件提醒

    连接邮箱服务器

    发送邮件

    关闭连接

     

while True:

     

    if cpu利用率 > 90%:

        发送邮件(‘CPU报警‘)

     

    if 硬盘使用空间 > 90%:

        发送邮件(‘硬盘报警‘)

     

    if 内存占用 > 80%:

        发送邮件(‘内存报警‘)

 

 

总结使用函数的好处:

 

1.代码重用

 

2.保持一致性,易维护

 

3.可扩展性

 

 

三 函数和过程

 

过程定义:过程就是简单特殊没有返回值的函数

 

这么看来我们在讨论为何使用函数的的时候引入的函数,都没有返回值,没有返回值就是过程,没错,但是在python中有比较神奇的事情

 

 

  def test01():

      msg=‘hello The little green frog‘

     print msg

  

  def test02():

      msg=‘hello WuDaLang‘

      print msg

      return msg

   

 

t1=test01()

 

t2=test02()

 

 

print ‘from test01 return is [%s]‘ %t1

print ‘from test02 return is [%s]‘ %t2

 

总结:当一个函数/过程没有使用return显示的定义返回值时,python解释器会隐式的返回None,

 

所以在python中即便是过程也可以算作函数。

 

 

  def test01():

      pass

  

  def test02():

     return 0

  

 def test03():

     return 0,10,‘hello‘,[‘alex‘,‘lb‘],{‘WuDaLang‘:‘lb‘}

  

t1=test01()

t2=test02()

t3=test03()

  

print ‘from test01 return is [%s]: ‘ %type(t1),t1

print ‘from test02 return is [%s]: ‘ %type(t2),t2

print ‘from test03 return is [%s]: ‘ %type(t3),t3

 

总结:

 

   返回值数=0:返回None

 

   返回值数=1:返回object

 

   返回值数>1:返回tuple

 

 

四 函数参数

 

1.形参变量只有在被调用时才分配内存单元,在调用结束时,即刻释放所分配的内存单元。因此,形参只在函数内部有效。函数调用结束返回主调用函数后则不能再使用该形参变量

 

2.实参可以是常量、变量、表达式、函数等,无论实参是何种类型的量,在进行函数调用时,它们都必须有确定的值,以便把这些值传送给形参。因此应预先用赋值,输入等办法使参数获得确定值

 

 

 

 

 

3.位置参数和关键字(标准调用:实参与形参位置一一对应;关键字调用:位置无需固定)

 

4.默认参数

 

5.参数组

 

 

五 局部变量和全局变量

 

在子程序中定义的变量称为局部变量,在程序的一开始定义的变量称为全局变量。

全局变量作用域是整个程序,局部变量作用域是定义该变量的子程序。

当全局变量与局部变量同名时:

在定义局部变量的子程序内,局部变量起作用;在其它地方全局变量起作用。

 

 name=‘lhf‘

 

 def change_name():

     print(‘我的名字‘,name)

 

 change_name()

 

 

 def change_name():

    name=‘帅了一笔‘

    print(‘我的名字‘,name)

 

change_name()

print(name)

 

def change_name():

    global name

    name=‘帅了一笔‘

    print(‘我的名字‘,name)

 

change_name()

print(name)

 

 

六 前向引用之‘函数即变量‘

 

 

  def action():

      print ‘in the action‘

      logger()

  action()

  报错NameError: global name ‘logger‘ is not defined

 

 

 def logger():

     print ‘in the logger‘

def action():

    print ‘in the action‘

    logger()

 

action()

 

 

def action():

    print ‘in the action‘

    logger()

def logger():

    print ‘in the logger‘

 

action() 

 

 

七 嵌套函数和作用域

 

看上面的标题的意思是,函数还能套函数?of course

 

 

 name = "Alex"

  

 def change_name():

    name = "Alex2"

   

   def change_name2():

         name = "Alex3"

         print("第3层打印",name)

 

     change_name2() #调用内层函数

     print("第2层打印",name)  

 

 change_name()

 print("最外层打印",name)

 

此时,在最外层调用change_name2()会出现什么效果?

 

没错, 出错了, 为什么呢?

 

 

 

作用域在定义函数时就已经固定住了,不会随着调用位置的改变而改变

 

 

  例一:

  name=‘alex‘

  

  def foo():

      name=‘lhf‘

      def bar():

          print(name)

      return bar

 

 func=foo()

 func()

 

 

 例二:

   name=‘alex‘

 

 def foo():

     name=‘lhf‘

     def bar():

         name=‘wupeiqi‘

         def tt():

             print(name)

         return tt

     return bar

 

 func=foo()

 func()()

 

 

 

 

八 递归

 

在函数内部,可以调用其他函数。如果一个函数在内部调用自身本身,这个函数就是递归函数。

 

 

  def calc(n):

      print(n)

      if int(n/2) ==0:

          return n

      return calc(int(n/2))

   

  calc(10)

   

  输出:

 10

 5

 2

 1

 

 递归问路

 

 

递归特性:

 

1. 必须有一个明确的结束条件

 

2. 每次进入更深一层递归时,问题规模相比上次递归都应有所减少

 

3. 递归效率不高,递归层次过多会导致栈溢出(在计算机中,函数调用是通过栈(stack)这种数据结构实现的,每当进入一个函数调用,栈就会加一层栈帧,每当函数返回,栈就会减一层栈帧。由于栈的大小不是无限的,所以,递归调用的次数过多,会导致栈溢出)

 

堆栈扫盲http://www.cnblogs.com/lln7777/archive/2012/03/14/2396164.html 

 

 

 二分查找

 

九 匿名函数

 

匿名函数就是不需要显式的指定函数

 

 

#这段代码

def calc(n):

    return n**n

print(calc(10))

 

#换成匿名函数

calc = lambda n:n**n

print(calc(10))

 

 

 

l=[3,2,100,999,213,1111,31121,333]

 print(max(l))

 

dic={‘k1‘:10,‘k2‘:100,‘k3‘:30}

 

 

print(max(dic))

print(dic[max(dic,key=lambda k:dic[k])])

 

 

 

 

 res = map(lambda x:x**2,[1,5,7,4,8])

 for i in res:

     print(i)

 

 输出

 1

 25

 49

 16

 64

 

 

十 函数式编程

 

 

 高阶函数

 

满足俩个特性任意一个即为高阶函数

 

1.函数的传入参数是一个函数名

 

2.函数的返回值是一个函数名

 

 map函数

 reduce函数

 filter函数

 总结

第七篇 python基础之函数,递归,内置函数