Python基础教程

发布时间:2019-09-24 17:33:42编辑:auto阅读(1872)

    6.4.5 参数收集的逆过程

    假设有如下函数:

    def add(x,y): return x+y

    比如说有个包含由两个相加的数字组成的元组

    params = (1,2)

    使用*运算符对参数进行“分配”,不过是在调用而不是在定义时使用:

    >>> add(*params)
    3

    ======

    同样,可以使用 双星号 运算符来处理字典。

    假设之前定义了hello_3,那么可以这样使用:

    >>> params = {'name':Sir Robin','greeting':'Well met'}
    >>> hello_3(**params)
    Well met.Sir Robin

    星号只在 定义函数(允许使用不定数目的参数)或者 调用(“分割”字典或者序列)时才有用。

    6.5 作用域

    在执行x=1赋值语句后,名称x引用到值1。这就像是使用字典一样,键引用值。当然,变量和所对应的值用的是个“不可见”的字典。

    內建的vars函数可以返回这个字典:

    >>> x = 1
    >>> scope = vars()
    >>> scope['x']
    1
    >>> scope['x'] += 1
    >>> x
    2

    这类“不可见字典”叫做 命名空间 或者 作用域 。除了全局作用域外,每个函数调用都会创建一个新的作用域:

    >>> def foo(): x = 42
    ...
    >>> x = 1
    >>> foo()
    >>> x
    1

    这里的foo函数改变(重绑定)了变量x,但是在最后的时候,x并没有变。这是因为当调用foo的时候,新的命名空间就被创建了,它作用于foo内的代码块。赋值语句x=42只在内部作用域(局部命名空间)起作用,所以它并不影响外部(全局)作用域中的x

    函数内的变量被称为局部变量(local variable),这是与全局变量相反的概念。参数的工作原理类似于局部变量,所以用全局变量的名字作为参数名并没有问题。

    >>> def output(x): print x
    ...
    >>> x = 1
    >>> y = 2
    >>> output(y)
    2

    ======

    重绑定全局变量:

    如果在函数内部将值赋予一个变量,它将会自动成为局部变量——除非告知Python将其声明为全局变量:

    >>> x = 1
    >>> def change_global():
            global x
            x = x + 1
            
    >>> change_global()
    >>> x
    2

    ======

    嵌套作用域

    Python的函数是可以嵌套的:

    def foo():
        def bar():
            print "Hello,World!"
        bar()

    函数嵌套有一个很突出的应用,例如需要一个函数“创建”另一个。也就意味着可以像下面这样(在其他函数内)书写函数:

    def multiplier(factor):
        def multiplier(number):
            return number*factor
        returnmultiplyByFactor

    一个函数位于另外一个里面,外层函数返回里层函数。也就是说函数本身被返回了,但并没有被调用。重要的是返回的函数还可以访问它的定义所在的作用域。换句话说,它“带着”它的环境(和相关的局部变量)。

    每次调用外层函数,它内部的函数都被重新绑定。factor变量每次都有一个新的值。由于Python的嵌套作用域,来自(`multiplier的)外部作用域的这个变量,稍后会被内层函数访问:

    >>> double = multiplier(2)
    >>> double(5)
    10
    >>> triple = multiplier(3)
    >>> triple(3)
    9
    >>> multiplier(5)(4)
    20

    类似multiplayByFactor函数存储子封闭作用域的行为叫做闭包(closure)。

    6.6 递归

    递归的定义(包括递归函数定义)包括它们自身定义内容的引用。

    关于递归,一个类似的函数定义如下:

    def recursion():
        return recursion()

    理论上讲,上述程序应该永远地运行下去,然而每次调用函数都会用掉一点内存,在足够的函数调用发生后(在之前的调用返回后),空间就不够了,程序会以一个“超过最大递归深度”的错误信息结束。

    这类递归就做无穷递归(infinite recursion),类似于以while True开始的无穷循环,中间没有break或者return语句。因为(理论上讲)它永远不会结束。

    有用的递归函数包含以下几个部分:

    1. 当函数直接返回值时有基本实例(最小可能性问题)

    2. 递归实例,包括一个或者多个问题较小部分的递归调用。

    这里的关键就是将问题分解成小部分,递归不可能永远继续下去,因为它总是以最小可能性问题结束,而这些问题又存储在基本实例中的。

    当每次函数被调用时,针对这个调用的新命名空间会被创建,意味着当函数调用“自身”时,实际上运行的是两个不同的函数(或者说是同一个函数具有两个不同的命名空间)。实际上,可以将它想象成和同种类的一个生物进行对话的另一个生物对话。

    6.6.1 递归经典案例:阶乘和幂

    计算数n的的阶乘:

    def factorial(n):
        result = n
        for i in range(1,n):
            result *= 1
            return result

    递归实现:

    1. 1的阶乘是1;

    2. 大于1的数n的阶乘是nn-1的阶乘。

    def factorial(n):
        if n == 1:
            return 1
        else:
            return n * factorial(n-1)

    ======

    计算幂

    例子:power(x,n)(xn的幂次)是x自乘n-1次的结果(所以x用作乘数n次。

    def power(x,n):
        result = 1
        for i in range(n):
            result *= x
        return result

    递归实现:

    1. 对于任意数字x来说,`power(x,0)是1;

    2. 对于任何大于0的书来说,power(x,n)x乘以(x,n-1)的结果。

    def power(x,n):
        if n == 0:
            return 1
        else:
            return x * power(x,n-1)

    6.6.2 递归经典案例:二分法查找

    递归实现:

    1. 如果上下限相同,那么就是数字所在位置,返回;

    2. 否则找到两者的中点(上下限的平均值),查找数字是在左侧还是在右侧,继续查找数字所在的那半部分。

    def search(sequence,number,lower,upper):
        if lower == upper:
            assert number == sequence[upper]
            return upper
        else:
            #整数除法//,浮点数除法/
            middle = (lower + upper) // 2 
            if number > sequence[middle]:
                return search(sequence,number,middle+1,upper)
            else:
                return search(sequence,number,lower,middle)

    提示:标准库中的bisect模块可以非常有效地实现二分查找。

    补充:函数式编程

    Python在应对“函数式编程”方面有一些有用的函数:mapfilterreduce函数(Python3.0中都被移至fuctools模块中)。

    mapfilter在目前版本的Python并非特别有用,并且可以使用列表推导式代替。不过可以使用map函数将序列中的元素全部传递给一个函数:

    >>> map(str,range(10))        #Equivalent to [str(i) for i in range(10)]
    ['0','1','2','3','4','5','6','7','8','9']

    filter函数可以基于一个返回布尔值的函数对元素进行过滤。

    #island 判断字符变量是否为字母或数字,
    #若是则返回非零,否则返回零
    
    >>> def fun(x):
            return x.isalnum()
            
    >>> seq = ["foo","x41","?!","***"]
    >>> filter(func,seq)
    ['foo','x41']

    本例中,使用列表推导式可以不用专门定义一个函数:

    >>> [x for x in seq if x.isalnum()]
    ['foo','x41']

    事实上,还有个叫做lambda表达式的特性,可以创建短小的函数。

    >>> filter(lambda x: x.isalnum().seq)
    ['foo','x41']

    =======

    reduce函数一般来说不能轻松被列表推导式替代,但是通常用不到这个功能。它会将序列的前两个元素与给定的函数联合使用,并且将它们的返回值和第3个元素继续联合使用,直到整个序列都处理完毕,并且得到一个最终结果

    可以使用reduce函数加上lambda x,y:x+y(继续使用相同的数字):

    >>> numbers = [72,101,108,108,111,44,32,119,111,114,108,100,33]
    >>> reduce(lambda x,y:x+y,numbers)
    1161

    当然,这里也可以使用内建函数sum

    6.7 小结

    • 抽象。抽象是隐藏多余细节的艺术。定义处理细节的函数可以让程序更抽象。

    • 函数定义。函数使用def语句定义。它们是由语句组成的块,可以从“外部世界”获取值(参数),也可以返回一个或者多个值作为运算的结果。

    • 参数。函数从参数中得到需要的信息,也就是函数调用时设定的变量。Python中有两类参数:位置参数关键数参数。参数在给定默认值时是可选的。

    • 作用域。变量存储在作用域(也叫作命名空间)中。Python有两类主要的作用域——全局作用域局部作用域。作用域可以嵌套。

    • 递归。 函数可以调用自身即递归。一切用递归实现的功能都能用循环实现,但是有些时候递归函数更易读。

    • 函数式编程。Python有一些进行函数式编程的机制。包括lambda表达式以及mapfilterreduce函数。

    6.7.1 本章的新函数

    | 函数 | 描述 |
    | ------------- |:-------------|
    | map(func,seq[,seq,...])| 对序列中的每个元素应用函数 |
    | filter(fuc,seq) | 返回其函数为真的元素的列表 |
    | reduce(func,seq[,initial]) | 等同于func(func(func(seq[0],seq[1],se1[2]... |
    | sum(seq) | 返回seq所有元素的和 |
    | apply(func,args[,kwargs]] | 调用函数,可以提供参数 |


    第7章 更加抽象

    在面对对象程序设计中,术语对象(object)基本上可以看做数据(特性)以及由一系列可以存取、操作这些数据的方法所组成的集合。使用对象替代全局变量和函数的原因可能有很多,其中对象最重要的优点包括以下几方面:

    • 多态(Polymorphism):意味着可以对不同类的对象使用同样的操作,它们会像“被施了魔法一般”工作。

    • 封装(Encapsulation):对外部世界隐藏对象的工作细节。

    • 继承(Inheritance):以通用的类为基础建立专门的类对象。

    7.1.1 多态

    术语多态的意思是“有多种形式”。多态意味着就算不知道变量所引用的对象类型是什么,还是能它进行操作,而它也会根据对象(或类)类型的不同而表现出不同的行为。

    repr函数是多态特性的代表之一,可以对任何东西使用:

    def length_message(x):
        print "The length of",repr(x),"is",len(x)
    >>> length_message('Fnord')
    The length of 'Fnord' is 5
    >>> length_message([1,2,3])
    The length of [1,2,3] is 3

    很多函数和运算符都是多态的——你写的绝大多数程序可能都是,只要使用多态函数和运算符,就会与“多态”发生关联。事实上,唯一能毁掉多态的就是使用函数显式地检查类型,比如typeisinstance以及issubclass函数等等。如果可能的话,应该尽力避免使用这些毁掉多态的方式。真正重要的是如何让对象按照你所希望的方式工作,不管它是不是真正的类型(或者类)。

    7.1.2 封装

    封装是指向程序中的其他部分隐藏对象的具体实现细节的原则。

    但是封装并不等同于多态,多态可以让用户对于不知道什么是类(对象类型)的对象进行方法调用,而封装是可以不用关心对象是如何构建的而直接进行使用。

    基本上,需要将对象进行抽象,调用方法的时候不用关心其他的东西,比如它是否干扰了全局变量。

    可以将其作为 特性(attribute) 存储。特性是作为变量构成对象的一部分,事实上方法更像是绑定到函数上的属性。

    对象有着自己的状态(state)。对象的状态由它的特性(比如名称)来描述。对象的方法可以改变它的特性。所以就像是将一大堆函数(方法)捆在一起,并且给予他们访问变量(特性)的权力。它们可以在函数调用之间保持保存的值。

    7.1.3 继承

    7.2 类和类型

    7.2.1 类到底是什么

    类是一种对象,所有的对象都属于某一个类,称为类的实例(instance)

    当一个对象所属的类是另外一个对象所属类的子集时,前者就被称为后者的 子类(subclass),所以“百灵鸟类”是“鸟类”的子类。相反,“鸟类”是“百灵鸟类”的“超类”(superclass)。但是,在面向程序设计中,子类的关系是隐式的,因为一个类的定义取决于它所支持的方法。类的所有实例都会包含这些方法,所以所有子类的所有实例都有这些方法。定义子类只是个定义更多(也有可能是重载已经存在的)方法的过程。

    7.2.2 创建自己的类

    7.2.3 特性、函数和方法

    事实上,self参数正是方法和参数的区别。方法(更专业一点可以称为绑定方法)将它们的第一个参数绑定到所属的实例上,因此无需显式提供该参数。当然也可以将特性绑定到一个普通函数上,这样就不会有特殊的self参数了:

    >>> class Class:
        def method(self):
            print 'I hava a self'
    
    >>> def function():
        print "I don't..."
     
    >>> instance = Class()
    >>> instance.method()
    I hava a self!
    >>> instance.method =function
    >>> instance.method()
    I don't...   

    注意,self参数并不依赖于调用方法的方式,前面使用的是instance.method(实例.方法)的形式,可以随意使用其他变量引用同一个方法:

    >>> class Bird:
        song = 'Squaawk!'
        def sing(self):
            print self.song
    
    >>> bird = Bird()
    >>> bird.sing()
    Squaawk!
    
    >>> birdsong = bird.sing
    >>> birdsong()
    Squaawk!

    尽管最后一个方法调用看起来与函数调用十分相似,但是变量birdsongs引用绑定方法bird.sing上,也就意味着这还是会对self参数进行访问(也就是说,它仍旧绑定到类的相同实例上)。

    再论私有化

    默认情况下,程序可以从外部访问一个对象的特性:

    >>> c.name
    'Sir Lancelot'
    >>> c.name = 'Sir Gumby'
    >>> c.getName()
    'Sir Gumby'

    为了避免这类事情的发生,应该使用私有(private)特性,这是外部对象无法访问到,但getNamesetName访问器(accessor)能够访问的特性。

    Python并不直接支持私有防暑,为了让方法或者特性变为私有(从外部无法访问),只要在它的名字前面加上双下划线即可。

    class Secretive:
        def __inacessible(self):
            print "Bet you can't see me.."
           
        def accessible(self):
            print "The secret message is:"
            self.__inaccessible

    现在,__inaccessible从外界是无法访问的,而在类内部还能使用(比如从accessible)访问:

    >>> s = Secretive()
    >>> s.__inaccessible()
    Traceback (most recent call last):
        File "<pyshell#112>",;ine 1, in ?
        s.__inaccessible()
    AttributeError: Secretive instance has no attribute '__inaccessible'
    >>> s.accessible()
    The secret message is:
    Bet you can't see me...

    尽管双下划线有些奇怪,但是看起来像是其他鱼鱼中的标准的私有方法。而在类的内部定义中,所有以双下划线开始的名字都被“翻译”成前面加上单下划线类名的形式。

    >>> Secretive._Secret__inaccsible
    <unboud method Secretive.__inaccessible>

    但实际上还是能够在类外访问这些私有方法,尽管不应该这么做:

    >>> s._Secretive.__inaccessible
    Bet you can't see me..

    简而言之,确保他人不会访问对象的方法和特性是不可能的,但是这类“名称变化”是提醒他们不应该访问这些函数或者特性的强有力信号。

    如果不需要使用这种方法但是又想让其他对象不要访问内部数据,那么可以使用单下划线,这不过是个习惯,但的确有实际效果。例如,前面有下划线的名字都不会被带星号的import语句(from module import *)导入。

    7.2.4 类的命名空间

    下面的两个语句几乎等价:

    def foo(x):return x*x
    foo = lambda X:x*x

    两者都创建了返回参数平方的函数,而且都将变量foo绑定到函数上。变量foo可以在全局(模块)范围内进行定义,也可处在局部的函数或方法内。定义类时,太阳的事情也会发生,所有位于class语句中的代码块都在特殊的命名空间中执行——类命名空间(class namespace)。这个命名空间可由类内所有成员访问。但并不是所有Python程序员都知道类的定义其实就是执行代码块。

    7.2.5 指定超类

    子类可以拓展超类的定义。将其他类名写在class语句后的圆括号内可以指定超类。

    7.2.6 检查继承

    如果想要查看一个类是否是另一个的子类,可以使用内建的issubclass函数。

    如果想要知道已知类的基类(们),可以直接使用它的特殊特性__base__:

    同样,还能使用isinstance方法检查一个对象是否是一个类的实例:

    7.2.7 多个超类

    7.2.8 接口和内省

    7.3 一些关于面向对象设计的思考

    7.4 小结


    第8章 异常

    8.1 什么是异常

关键字