Python 進(jìn)階篇之面向?qū)ο蠡A(chǔ)

定義類并創(chuàng)建實(shí)例

在Python中芒划,類通過 class 關(guān)鍵字定義。以 Person 為例贼穆,定義一個(gè)Person類如下:

class Person(object):
    pass

按照 Python 的編程習(xí)慣哼御,類名以大寫字母開頭轧葛,緊接著是(object),表示該類是從哪個(gè)類繼承下來的艇搀。類的繼承將在后面的章節(jié)講解,現(xiàn)在我們只需要簡單地從object類繼承求晶。
有了Person類的定義焰雕,就可以創(chuàng)建出具體的xiaoming、xiaohong等實(shí)例芳杏。創(chuàng)建實(shí)例使用 類名+()矩屁,類似函數(shù)調(diào)用的形式創(chuàng)建:

xiaoming = Person()
xiaohong = Person()

創(chuàng)建實(shí)例屬性

雖然可以通過Person類創(chuàng)建出xiaoming、xiaohong等實(shí)例爵赵,但是這些實(shí)例看上除了地址不同外吝秕,沒有什么其他不同。
在現(xiàn)實(shí)世界中空幻,區(qū)分xiaoming烁峭、xiaohong要依靠他們各自的名字、性別秕铛、生日等屬性约郁。
如何讓每個(gè)實(shí)例擁有各自不同的屬性?由于Python是動(dòng)態(tài)語言但两,對(duì)每一個(gè)實(shí)例鬓梅,都可以直接給他們的屬性賦值,
例如谨湘,給xiaoming這個(gè)實(shí)例加上name绽快、genderbirth屬性:

xiaoming = Person()
xiaoming.name = 'Xiao Ming'
xiaoming.gender = 'Male'
xiaoming.birth = '1990-1-1'

給xiaohong加上的屬性不一定要和xiaoming相同:

xiaohong = Person()
xiaohong.name = 'Xiao Hong'
xiaohong.school = 'No. 1 High School'
xiaohong.grade = 2

實(shí)例的屬性可以像普通變量一樣進(jìn)行操作:

xiaohong.grade = xiaohong.grade + 1

初始化實(shí)例屬性

雖然我們可以自由地給一個(gè)實(shí)例綁定各種屬性芥丧,但是,現(xiàn)實(shí)世界中坊罢,一種類型的實(shí)例應(yīng)該擁有相同名字的屬性续担。
例如,Person類應(yīng)該在創(chuàng)建的時(shí)候就擁有 name艘绍、genderbirth 屬性赤拒,怎么辦?
在定義 Person 類時(shí)诱鞠,可以為Person類添加一個(gè)特殊的__init__()方法挎挖,當(dāng)創(chuàng)建實(shí)例時(shí),__init__()方法被自動(dòng)調(diào)用航夺,我們就能在此為每個(gè)實(shí)例都統(tǒng)一加上以下屬性:

class Person(object):
    def __init__(self, name, gender, birth):
        self.name = name
        self.gender = gender
        self.birth = birth

__init__() 方法的第一個(gè)參數(shù)必須是self(也可以用別的名字蕉朵,但建議使用習(xí)慣用法),后續(xù)參數(shù)則可以自由指定阳掐,和定義函數(shù)沒有任何區(qū)別始衅。
相應(yīng)地,創(chuàng)建實(shí)例時(shí)缭保,就必須要提供除 self 以外的參數(shù):

xiaoming = Person('Xiao Ming', 'Male', '1991-1-1')
xiaohong = Person('Xiao Hong', 'Female', '1992-2-2')

有了__init__()方法汛闸,每個(gè)Person實(shí)例在創(chuàng)建時(shí),都會(huì)有 name艺骂、genderbirth 這3個(gè)屬性诸老,并且,被賦予不同的屬性值钳恕,訪問屬性使用.操作符:

print xiaoming.name
# 輸出 'Xiao Ming'
print xiaohong.birth
# 輸出 '1992-2-2'

要特別注意的是别伏,初學(xué)者定義__init__()方法常常忘記了 self 參數(shù):

>>> class Person(object):
...     def __init__(name, gender, birth):
...         pass
... 
>>> xiaoming = Person('Xiao Ming', 'Male', '1990-1-1')
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: __init__() takes exactly 3 arguments (4 given)

這會(huì)導(dǎo)致創(chuàng)建失敗或運(yùn)行不正常,因?yàn)榈谝粋€(gè)參數(shù)name被Python解釋器傳入了實(shí)例的引用忧额,從而導(dǎo)致整個(gè)方法的調(diào)用參數(shù)位置全部沒有對(duì)上厘肮。

訪問限制

我們可以給一個(gè)實(shí)例綁定很多屬性,如果有些屬性不希望被外部訪問到怎么辦睦番?
Python對(duì)屬性權(quán)限的控制是通過屬性名來實(shí)現(xiàn)的类茂,如果一個(gè)屬性由雙下劃線開頭(__),該屬性就無法被外部訪問抡砂〈笤郏看例子:

class Person(object):
    def __init__(self, name):
        self.name = name
        self._title = 'Mr'
        self.__job = 'Student'
p = Person('Bob')
print p.name
# => Bob
print p._title
# => Mr
print p.__job
# => Error
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'Person' object has no attribute '__job'

可見,只有以雙下劃線開頭的"__job"不能直接被外部訪問注益。
但是碴巾,如果一個(gè)屬性以"__xxx__"的形式定義,那它又可以被外部訪問了丑搔,以"__xxx__"定義的屬性在Python的類中被稱為特殊屬性厦瓢,
有很多預(yù)定義的特殊屬性可以使用提揍,通常我們不要把普通屬性用"xxx"定義。
以單下劃線開頭的屬性"_xxx"雖然也可以被外部訪問煮仇,但是贸桶,按照習(xí)慣蚀瘸,他們不應(yīng)該被外部訪問。

創(chuàng)建類屬性

類是模板,而實(shí)例則是根據(jù)類創(chuàng)建的對(duì)象蝇恶。
綁定在一個(gè)實(shí)例上的屬性不會(huì)影響其他實(shí)例羡鸥,但是咬摇,類本身也是一個(gè)對(duì)象儡陨,如果在類上綁定一個(gè)屬性,則所有實(shí)例都可以訪問類的屬性辙售,并且轻抱,所有實(shí)例訪問的類屬性都是同一個(gè)!
也就是說旦部,實(shí)例屬性每個(gè)實(shí)例各自擁有祈搜,互相獨(dú)立,而類屬性有且只有一份士八。
定義類屬性可以直接在 class 中定義:

class Person(object):
    address = 'Earth'
    def __init__(self, name):
        self.name = name

因?yàn)轭悓傩允侵苯咏壎ㄔ陬惿系娜菅啵裕L問類屬性不需要?jiǎng)?chuàng)建實(shí)例婚度,就可以直接訪問:

print Person.address
# => Earth

對(duì)一個(gè)實(shí)例調(diào)用類的屬性也是可以訪問的缰趋,所有實(shí)例都可以訪問到它所屬的類的屬性:

p1 = Person('Bob')
p2 = Person('Alice')
print p1.address
# => Earth
print p2.address
# => Earth

由于Python是動(dòng)態(tài)語言,類屬性也是可以動(dòng)態(tài)添加和修改的:

Person.address = 'China'
print p1.address
# => 'China'
print p2.address
# => 'China'

因?yàn)轭悓傩灾挥幸环萆录裕?dāng)Person類的address改變時(shí)味抖,所有實(shí)例訪問到的類屬性都改變了评甜。
類屬性和實(shí)例屬性名字沖突怎么辦
修改類屬性會(huì)導(dǎo)致所有實(shí)例訪問到的類屬性全部都受影響,但是仔涩,如果在實(shí)例變量上修改類屬性會(huì)發(fā)生什么問題呢忍坷?

class Person(object):
    address = 'Earth'
    def __init__(self, name):
        self.name = name

p1 = Person('Bob')
p2 = Person('Alice')

print 'Person.address = ' + Person.address

p1.address = 'China'
print 'p1.address = ' + p1.address

print 'Person.address = ' + Person.address
print 'p2.address = ' + p2.address
結(jié)果如下:
Person.address = Earth
p1.address = China
Person.address = Earth
p2.address = Earth

我們發(fā)現(xiàn),在設(shè)置了 p1.address = 'China' 后熔脂,p1訪問 address 確實(shí)變成了 'China'佩研,但是,Person.addressp2.address仍然是'Earch'霞揉,怎么回事旬薯?
原因是 p1.address = 'China'并沒有改變 Personaddress,而是給 p1這個(gè)實(shí)例綁定了實(shí)例屬性address 适秩,對(duì)p1來說绊序,
它有一個(gè)實(shí)例屬性address(值是'China')硕舆,而它所屬的類Person也有一個(gè)類屬性address,所以:
訪問 p1.address 時(shí)骤公,優(yōu)先查找實(shí)例屬性抚官,返回'China'
訪問 p2.address 時(shí)阶捆,p2沒有實(shí)例屬性address凌节,但是有類屬性address,因此返回'Earth'洒试。
可見倍奢,當(dāng)實(shí)例屬性和類屬性重名時(shí),實(shí)例屬性優(yōu)先級(jí)高儡司,它將屏蔽掉對(duì)類屬性的訪問娱挨。
當(dāng)我們把 p1address 實(shí)例屬性刪除后,訪問 p1.address 就又返回類屬性的值 'Earth'了:

del p1.address
print p1.address
# => Earth

可見捕犬,千萬不要在實(shí)例上修改類屬性跷坝,它實(shí)際上并沒有修改類屬性,而是給實(shí)例綁定了一個(gè)實(shí)例屬性碉碉。

定義實(shí)例方法

一個(gè)實(shí)例的私有屬性就是以__開頭的屬性柴钻,無法被外部訪問,那這些屬性定義有什么用垢粮?
雖然私有屬性無法從外部訪問贴届,但是,從類的內(nèi)部是可以訪問的蜡吧。除了可以定義實(shí)例的屬性外毫蚓,還可以定義實(shí)例的方法。
實(shí)例的方法就是在類中定義的函數(shù)昔善,它的第一個(gè)參數(shù)永遠(yuǎn)是 self元潘,指向調(diào)用該方法的實(shí)例本身,其他參數(shù)和一個(gè)普通函數(shù)是完全一樣的:

class Person(object):

    def __init__(self, name):
        self.__name = name

    def get_name(self):
        return self.__name

get_name(self) 就是一個(gè)實(shí)例方法君仆,它的第一個(gè)參數(shù)是self翩概。__init__(self, name)其實(shí)也可看做是一個(gè)特殊的實(shí)例方法。
調(diào)用實(shí)例方法必須在實(shí)例上調(diào)用:

p1 = Person('Bob')
print p1.get_name()  # self不需要顯式傳入# => Bob

在實(shí)例方法內(nèi)部返咱,可以訪問所有實(shí)例屬性钥庇,這樣,如果外部需要訪問私有屬性咖摹,可以通過方法調(diào)用獲得评姨,這種數(shù)據(jù)封裝的形式除了能保護(hù)內(nèi)部數(shù)據(jù)一致性外,還可以簡化外部調(diào)用的難度萤晴。

方法也是屬性

我們?cè)?class 中定義的實(shí)例方法其實(shí)也是屬性参咙,它實(shí)際上是一個(gè)函數(shù)對(duì)象:

class Person(object):
    def __init__(self, name, score):
        self.name = name
        self.score = score
    def get_grade(self):
        return 'A'

p1 = Person('Bob', 90)
print p1.get_grade
# => <bound method Person.get_grade of <__main__.Person object at 0x109e58510>>
print p1.get_grade()
# => A

也就是說龄广,p1.get_grade 返回的是一個(gè)函數(shù)對(duì)象,但這個(gè)函數(shù)是一個(gè)綁定到實(shí)例的函數(shù)蕴侧,p1.get_grade() 才是方法調(diào)用择同。
因?yàn)榉椒ㄒ彩且粋€(gè)屬性,所以净宵,它也可以動(dòng)態(tài)地添加到實(shí)例上敲才,只是需要用 types.MethodType() 把一個(gè)函數(shù)變?yōu)橐粋€(gè)方法:

import types
def fn_get_grade(self):
    if self.score >= 80:
        return 'A'
    if self.score >= 60:
        return 'B'
    return 'C'

class Person(object):
    def __init__(self, name, score):
        self.name = name
        self.score = score

p1 = Person('Bob', 90)
p1.get_grade = types.MethodType(fn_get_grade, p1, Person)
print p1.get_grade()
# => A
p2 = Person('Alice', 65)
print p2.get_grade()
# ERROR: AttributeError: 'Person' object has no attribute 'get_grade'
# 因?yàn)閜2實(shí)例并沒有綁定get_grade

給一個(gè)實(shí)例動(dòng)態(tài)添加方法并不常見,直接在class中定義要更直觀择葡。

定義類方法

和屬性類似紧武,方法也分實(shí)例方法和類方法。
在class中定義的全部是實(shí)例方法敏储,實(shí)例方法第一個(gè)參數(shù) self 是實(shí)例本身阻星。
要在class中定義類方法,需要這么寫:

class Person(object):
    count = 0
    @classmethod
    def how_many(cls):
        return cls.count
    def __init__(self, name):
        self.name = name
        Person.count = Person.count + 1

print Person.how_many()
p1 = Person('Bob')
print Person.how_many()

通過標(biāo)記一個(gè) @classmethod已添,該方法將綁定到 Person 類上妥箕,而非類的實(shí)例。類方法的第一個(gè)參數(shù)將傳入類本身更舞,通常將參數(shù)名命名為 cls畦幢,上面的 cls.count 實(shí)際上相當(dāng)于 Person.count
因?yàn)槭窃陬惿险{(diào)用缆蝉,而非實(shí)例上調(diào)用宇葱,因此類方法無法獲得任何實(shí)例變量,只能獲得類的引用刊头。

類繼承

繼承一個(gè)類
如果已經(jīng)定義了Person類黍瞧,需要定義新的Student和Teacher類時(shí),可以直接從Person類繼承:

class Person(object):
    def __init__(self, name, gender):
        self.name = name
        self.gender = gender

定義Student類時(shí)原杂,只需要把額外的屬性加上雷逆,例如score

class Student(Person):
    def __init__(self, name, gender, score):
        super(Student, self).__init__(name, gender)
        self.score = score

一定要用 super(Student, self).__init__(name, gender) 去初始化父類,否則污尉,繼承自 PersonStudent 將沒有 namegender
函數(shù)super(Student, self)將返回當(dāng)前類繼承的父類往产,即 Person 被碗,然后調(diào)用__init__()方法,注意self參數(shù)已在super()中傳入仿村,在__init__()中將隱式傳遞锐朴,不需要寫出(也不能寫)。

判斷類型

函數(shù)isinstance()可以判斷一個(gè)變量的類型蔼囊,既可以用在Python內(nèi)置的數(shù)據(jù)類型如str焚志、list衣迷、dict,也可以用在我們自定義的類酱酬,它們本質(zhì)上都是數(shù)據(jù)類型壶谒。
假設(shè)有如下的 Person、StudentTeacher 的定義及繼承關(guān)系如下:

class Person(object):
    def __init__(self, name, gender):
        self.name = name
        self.gender = gender

class Student(Person):
    def __init__(self, name, gender, score):
        super(Student, self).__init__(name, gender)
        self.score = score

class Teacher(Person):
    def __init__(self, name, gender, course):
        super(Teacher, self).__init__(name, gender)
        self.course = course

p = Person('Tim', 'Male')
s = Student('Bob', 'Male', 88)
t = Teacher('Alice', 'Female', 'English')

當(dāng)我們拿到變量 p膳沽、s汗菜、t 時(shí),可以使用 isinstance 判斷類型:

>>> isinstance(p, Person)
True    # p是Person類型
>>> isinstance(p, Student)
False   # p不是Student類型
>>> isinstance(p, Teacher)
False   # p不是Teacher類型

這說明在繼承鏈上挑社,一個(gè)父類的實(shí)例不能是子類類型陨界,因?yàn)樽宇惐雀割惗嗔艘恍傩院头椒ā?br> 我們?cè)倏疾?s :

>>> isinstance(s, Person)
True    # s是Person類型
>>> isinstance(s, Student)
True    # s是Student類型
>>> isinstance(s, Teacher)
False   # s不是Teacher類型

s 是Student類型,不是Teacher類型痛阻,這很容易理解菌瘪。但是,s 也是Person類型阱当,因?yàn)镾tudent繼承自Person俏扩,雖然它比Person多了一些屬性和方法,但是斗这,把 s 看成Person的實(shí)例也是可以的动猬。
這說明在一條繼承鏈上,一個(gè)實(shí)例可以看成它本身的類型表箭,也可以看成它父類的類型赁咙。

多態(tài)

類具有繼承關(guān)系,并且子類類型可以向上轉(zhuǎn)型看做父類類型免钻,如果我們從 Person 派生出 Student和Teacher 彼水,并都寫了一個(gè) whoAmI() 方法:

class Person(object):
    def __init__(self, name, gender):
        self.name = name
        self.gender = gender
    def whoAmI(self):
        return 'I am a Person, my name is %s' % self.name

class Student(Person):
    def __init__(self, name, gender, score):
        super(Student, self).__init__(name, gender)
        self.score = score
    def whoAmI(self):
        return 'I am a Student, my name is %s' % self.name

class Teacher(Person):
    def __init__(self, name, gender, course):
        super(Teacher, self).__init__(name, gender)
        self.course = course
    def whoAmI(self):
        return 'I am a Teacher, my name is %s' % self.name

在一個(gè)函數(shù)中,如果我們接收一個(gè)變量 x极舔,則無論該 x 是 Person凤覆、Student還是 Teacher,都可以正確打印出結(jié)果:

def who_am_i(x):
    print x.whoAmI()

p = Person('Tim', 'Male')
s = Student('Bob', 'Male', 88)
t = Teacher('Alice', 'Female', 'English')

who_am_i(p)
who_am_i(s)
who_am_i(t)
運(yùn)行結(jié)果:
I am a Person, my name is Tim
I am a Student, my name is Bob
I am a Teacher, my name is Alice

這種行為稱為多態(tài)拆魏。也就是說盯桦,方法調(diào)用將作用在 x 的實(shí)際類型上。
s 是Student類型渤刃,它實(shí)際上擁有自己的 whoAmI()方法以及從 Person繼承的 whoAmI方法拥峦,但調(diào)用 s.whoAmI()總是先查找它自身的定義,如果沒有定義卖子,則順著繼承鏈向上查找略号,直到在某個(gè)父類中找到為止。
由于Python是動(dòng)態(tài)語言,所以玄柠,傳遞給函數(shù) who_am_i(x)的參數(shù) x 不一定是 Person 或 Person 的子類型突梦。任何數(shù)據(jù)類型的實(shí)例都可以,只要它有一個(gè)whoAmI()的方法即可:

class Book(object):
    def whoAmI(self):
        return 'I am a book'

這是動(dòng)態(tài)語言和靜態(tài)語言(例如Java)最大的差別之一羽利。動(dòng)態(tài)語言調(diào)用實(shí)例方法宫患,不檢查類型,只要方法存在铐伴,參數(shù)正確撮奏,就可以調(diào)用。

多重繼承

除了從一個(gè)父類繼承外当宴,Python允許從多個(gè)父類繼承畜吊,稱為多重繼承。
多重繼承的繼承鏈就不是一棵樹了户矢,它像這樣:

class A(object):
    def __init__(self, a):
        print 'init A...'
        self.a = a

class B(A):
    def __init__(self, a):
        super(B, self).__init__(a)
        print 'init B...'

class C(A):
    def __init__(self, a):
        super(C, self).__init__(a)
        print 'init C...'

class D(B, C):
    def __init__(self, a):
        super(D, self).__init__(a)
        print 'init D...'

看下圖:

繼承

像這樣玲献,D 同時(shí)繼承自 BC,也就是 D 擁有了 A梯浪、B捌年、C 的全部功能。多重繼承通過 super()調(diào)用__init__()方法時(shí)挂洛,A 雖然被繼承了兩次礼预,但__init__()只調(diào)用一次:

>>> d = D('d')
init A...
init C...
init B...
init D...

多重繼承的目的是從兩種繼承樹中分別選擇并繼承出子類,以便組合功能使用虏劲。
舉個(gè)例子托酸,Python的網(wǎng)絡(luò)服務(wù)器有TCPServer、UDPServer柒巫、UnixStreamServer励堡、UnixDatagramServer,而服務(wù)器運(yùn)行模式有 多進(jìn)程ForkingMixin 和 多線程ThreadingMixin兩種堡掏。
要?jiǎng)?chuàng)建多進(jìn)程模式的 TCPServer:

class MyTCPServer(TCPServer, ForkingMixin)
    pass

要?jiǎng)?chuàng)建多線程模式的 UDPServer:

class MyUDPServer(UDPServer, ThreadingMixin):
    pass

如果沒有多重繼承应结,要實(shí)現(xiàn)上述所有可能的組合需要 4x2=8 個(gè)子類。
獲取對(duì)象信息
拿到一個(gè)變量泉唁,除了用 isinstance() 判斷它是否是某種類型的實(shí)例外鹅龄,還有沒有別的方法獲取到更多的信息呢?
例如亭畜,已有定義:

class Person(object):
    def __init__(self, name, gender):
        self.name = name
        self.gender = gender

class Student(Person):
    def __init__(self, name, gender, score):
        super(Student, self).__init__(name, gender)
        self.score = score
    def whoAmI(self):
        return 'I am a Student, my name is %s' % self.name
首先可以用 type() 函數(shù)獲取變量的類型扮休,它返回一個(gè) Type 對(duì)象:
>>> type(123)
<type 'int'>
>>> s = Student('Bob', 'Male', 88)
>>> type(s)
<class '__main__.Student'>

其次,可以用 dir() 函數(shù)獲取變量的所有屬性:

>>> dir(123)   # 整數(shù)也有很多屬性...
['__abs__', '__add__', '__and__', '__class__', '__cmp__', ...]

>>> dir(s)
['__class__', '__delattr__', '__dict__', '__doc__', '__format__', '__getattribute__', '__hash__', '__init__', '__module__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'gender', 'name', 'score', 'whoAmI']

對(duì)于實(shí)例變量贱案,dir()返回所有實(shí)例屬性,包括__class__這類有特殊意義的屬性。注意到方法whoAmI也是 s 的一個(gè)屬性宝踪。
如何去掉__xxx__這類的特殊屬性侨糟,只保留我們自己定義的屬性?回顧一下filter()函數(shù)的用法瘩燥。
dir()返回的屬性是字符串列表秕重,如果已知一個(gè)屬性名稱,要獲取或者設(shè)置對(duì)象的屬性厉膀,就需要用 getattr() 和 setattr( )函數(shù)了:

>>> getattr(s, 'name')  # 獲取name屬性
'Bob'

>>> setattr(s, 'name', 'Adam')  # 設(shè)置新的name屬性

>>> s.name
'Adam'

>>> getattr(s, 'age')  # 獲取age屬性溶耘,但是屬性不存在,報(bào)錯(cuò):
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'Student' object has no attribute 'age'

>>> getattr(s, 'age', 20)  # 獲取age屬性服鹅,如果屬性不存在凳兵,就返回默認(rèn)值20:

定制類

__str____repr__
如果要把一個(gè)類的實(shí)例變成 str,就需要實(shí)現(xiàn)特殊方法__str__()

class Person(object):
    def __init__(self, name, gender):
        self.name = name
        self.gender = gender
    def __str__(self):
        return '(Person: %s, %s)' % (self.name, self.gender)

現(xiàn)在企软,在交互式命令行下用 print 試試:

>>> p = Person('Bob', 'male')
>>> print p
(Person: Bob, male)
但是庐扫,如果直接敲變量 p:
>>> p
<main.Person object at 0x10c941890>

似乎__str__() 不會(huì)被調(diào)用。
因?yàn)?Python 定義了__str__()__repr__()兩種方法仗哨,__str__()用于顯示給用戶形庭,而__repr__()用于顯示給開發(fā)人員。
有一個(gè)偷懶的定義repr的方法:

class Person(object):
    def __init__(self, name, gender):
        self.name = name
        self.gender = gender
    def __str__(self):
        return '(Person: %s, %s)' % (self.name, self.gender)
    __repr__ = __str__          
__cmp__

對(duì) int厌漂、str等內(nèi)置數(shù)據(jù)類型排序時(shí)萨醒,Python的 sorted() 按照默認(rèn)的比較函數(shù) cmp 排序,但是苇倡,如果對(duì)一組 Student 類的實(shí)例排序時(shí)富纸,就必須提供我們自己的特殊方法 __cmp__()

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.score = score
    def __str__(self):
        return '(%s: %s)' % (self.name, self.score)
    __repr__ = __str__

    def __cmp__(self, s):
        if self.name < s.name:
            return -1
        elif self.name > s.name:
            return 1
        else:
            return 0

上述 Student 類實(shí)現(xiàn)了__cmp__()方法,__cmp__用實(shí)例自身self和傳入的實(shí)例 s 進(jìn)行比較雏节,如果 self 應(yīng)該排在前面胜嗓,就返回 -1,如果 s 應(yīng)該排在前面钩乍,就返回1辞州,如果兩者相當(dāng),返回 0寥粹。
Student類實(shí)現(xiàn)了按name進(jìn)行排序:

>>> L = [Student('Tim', 99), Student('Bob', 88), Student('Alice', 77)]
>>> print sorted(L)
[(Alice: 77), (Bob: 88), (Tim: 99)]

注意: 如果list不僅僅包含 Student 類变过,則 __cmp__ 可能會(huì)報(bào)錯(cuò):

L = [Student('Tim', 99), Student('Bob', 88), 100, 'Hello']
print sorted(L)

請(qǐng)思考如何解決。
__len__
如果一個(gè)類表現(xiàn)得像一個(gè)list涝涤,要獲取有多少個(gè)元素媚狰,就得用 len() 函數(shù)。
要讓 len() 函數(shù)工作正常阔拳,類必須提供一個(gè)特殊方法__len__()崭孤,它返回元素的個(gè)數(shù)。
例如,我們寫一個(gè) Students 類辨宠,把名字傳進(jìn)去:

class Students(object):
    def __init__(self, *args):
        self.names = args
    def __len__(self):
        return len(self.names)

只要正確實(shí)現(xiàn)了__len__()方法遗锣,就可以用len()函數(shù)返回Students實(shí)例的“長度”:

>>> ss = Students('Bob', 'Alice', 'Tim')
>>> print len(ss)
3

數(shù)學(xué)運(yùn)算

Python 提供的基本數(shù)據(jù)類型 int、float 可以做整數(shù)和浮點(diǎn)的四則運(yùn)算以及乘方等運(yùn)算嗤形。
但是精偿,四則運(yùn)算不局限于int和float,還可以是有理數(shù)赋兵、矩陣等笔咽。
要表示有理數(shù),可以用一個(gè)Rational類來表示:

class Rational(object):
    def __init__(self, p, q):
        self.p = p
        self.q = q

p霹期、q 都是整數(shù)叶组,表示有理數(shù) p/q
如果要讓Rational進(jìn)行+運(yùn)算经伙,需要正確實(shí)現(xiàn)__add__

class Rational(object):
    def __init__(self, p, q):
        self.p = p
        self.q = q
    def __add__(self, r):
        return Rational(self.p * r.q + self.q * r.p, self.q * r.q)
    def __str__(self):
        return '%s/%s' % (self.p, self.q)
    __repr__ = __str__

現(xiàn)在可以試試有理數(shù)加法:

>>> r1 = Rational(1, 3)
>>> r2 = Rational(1, 2)
>>> print r1 + r2
5/6

類型轉(zhuǎn)換

Rational類實(shí)現(xiàn)了有理數(shù)運(yùn)算扶叉,但是,如果要把結(jié)果轉(zhuǎn)為 int 或 float 怎么辦帕膜?
考察整數(shù)和浮點(diǎn)數(shù)的轉(zhuǎn)換:

>>> int(12.34)
12
>>> float(12)
12.0

如果要把 Rational 轉(zhuǎn)為 int枣氧,應(yīng)該使用:

r = Rational(12, 5)
n = int(r)

要讓int()函數(shù)正常工作,只需要實(shí)現(xiàn)特殊方法__int__():

class Rational(object):
    def __init__(self, p, q):
        self.p = p
        self.q = q
    def __int__(self):
        return self.p // self.q
結(jié)果如下:
>>> print int(Rational(7, 2))
3
>>> print int(Rational(1, 3))
0

同理垮刹,要讓float()函數(shù)正常工作达吞,只需要實(shí)現(xiàn)特殊方法__float__()

@property

考察 Student 類:

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.score = score

當(dāng)我們想要修改一個(gè) Student 的 scroe 屬性時(shí)荒典,可以這么寫:

s = Student('Bob', 59)
s.score = 60

但是也可以這么寫:

s.score = 1000

顯然酪劫,直接給屬性賦值無法檢查分?jǐn)?shù)的有效性。
如果利用兩個(gè)方法:

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.__score = score
    def get_score(self):
        return self.__score
    def set_score(self, score):
        if score < 0 or score > 100:
            raise ValueError('invalid score')
        self.__score = score

這樣一來寺董,s.set_score(1000) 就會(huì)報(bào)錯(cuò)覆糟。
這種使用 get/set 方法來封裝對(duì)一個(gè)屬性的訪問在許多面向?qū)ο缶幊痰恼Z言中都很常見。
但是寫 s.get_score()s.set_score() 沒有直接寫 s.score 來得直接遮咖。
有沒有兩全其美的方法滩字?----有。
因?yàn)镻ython支持高階函數(shù)御吞,在函數(shù)式編程中我們介紹了裝飾器函數(shù)麦箍,可以用裝飾器函數(shù)把 get/set 方法“裝飾”成屬性調(diào)用:

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.__score = score
    @property
    def score(self):
        return self.__score
    @score.setter
    def score(self, score):
        if score < 0 or score > 100:
            raise ValueError('invalid score')
        self.__score = score

注意: 第一個(gè)score(self)是get方法,用@property裝飾陶珠,第二個(gè)score(self, score)是set方法挟裂,用@score.setter裝飾,@score.setter是前一個(gè)@property裝飾后的副產(chǎn)品揍诽。
現(xiàn)在诀蓉,就可以像使用屬性一樣設(shè)置score了:

>>> s = Student('Bob', 59)
>>> s.score = 60
>>> print s.score
60
>>> s.score = 1000
Traceback (most recent call last):
  ...
ValueError: invalid score

說明對(duì) score 賦值實(shí)際調(diào)用的是 set方法栗竖。

slots

由于Python是動(dòng)態(tài)語言,任何實(shí)例在運(yùn)行期都可以動(dòng)態(tài)地添加屬性渠啤。
如果要限制添加的屬性划滋,例如,Student類只允許添加 name埃篓、genderscore 這3個(gè)屬性,就可以利用Python的一個(gè)特殊的__slots__來實(shí)現(xiàn)根资。
顧名思義架专,__slots__是指一個(gè)類允許的屬性列表:

class Student(object):
    __slots__ = ('name', 'gender', 'score')
    def __init__(self, name, gender, score):
        self.name = name
        self.gender = gender
        self.score = score
現(xiàn)在,對(duì)實(shí)例進(jìn)行操作:
>>> s = Student('Bob', 'male', 59)
>>> s.name = 'Tim' # OK
>>> s.score = 99 # OK
>>> s.grade = 'A'
Traceback (most recent call last):
  ...
AttributeError: 'Student' object has no attribute 'grade'

__slots__的目的是限制當(dāng)前類所能擁有的屬性玄帕,如果不需要添加任意動(dòng)態(tài)的屬性部脚,使用__slots__也能節(jié)省內(nèi)存。

call

在Python中裤纹,函數(shù)其實(shí)是一個(gè)對(duì)象:

>>> f = abs
>>> f.__name__
'abs'
>>> f(-123)
123

由于 f 可以被調(diào)用委刘,所以,f 被稱為可調(diào)用對(duì)象鹰椒。
所有的函數(shù)都是可調(diào)用對(duì)象锡移。
一個(gè)類實(shí)例也可以變成一個(gè)可調(diào)用對(duì)象,只需要實(shí)現(xiàn)一個(gè)特殊方法__call__()漆际。
我們把 Person 類變成一個(gè)可調(diào)用對(duì)象:

class Person(object):
    def __init__(self, name, gender):
        self.name = name
        self.gender = gender

    def __call__(self, friend):
        print 'My name is %s...' % self.name
        print 'My friend is %s...' % friend

現(xiàn)在可以對(duì) Person 實(shí)例直接調(diào)用:

>>> p = Person('Bob', 'male')
>>> p('Tim')
My name is Bob...
My friend is Tim...

單看 p('Tim') 你無法確定 p 是一個(gè)函數(shù)還是一個(gè)類實(shí)例淆珊,所以,在Python中奸汇,函數(shù)也是對(duì)象施符,對(duì)象和函數(shù)的區(qū)別并不顯著。

模塊

模塊讓你能夠有邏輯地組織你的Python代碼段擂找。
把相關(guān)的代碼分配到一個(gè) 模塊里能讓你的代碼更好用戳吝,更易懂。
模塊也是Python對(duì)象贯涎,具有隨機(jī)的名字屬性用來綁定或引用听哭。
簡單地說,模塊就是一個(gè)保存了Python代碼的文件柬采。模塊能定義函數(shù)欢唾,類和變量。模塊里也能包含可執(zhí)行的代碼粉捻。

例子
一個(gè)叫做aname的模塊里的Python代碼一般都能在一個(gè)叫aname.py的文件中找到礁遣。下例是個(gè)簡單的模塊support.py。

def print_func( par ):
   print "Hello : ", par
   return

import 語句

想使用Python源文件肩刃,只需在另一個(gè)源文件里執(zhí)行import語句祟霍,語法如下:

import module1[, module2[,... moduleN]

當(dāng)解釋器遇到import語句杏头,如果模塊在當(dāng)前的搜索路徑就會(huì)被導(dǎo)入。
搜索路徑是一個(gè)解釋器會(huì)先進(jìn)行搜索的所有目錄的列表沸呐。如想要導(dǎo)入模塊hello.py醇王,需要把命令放在腳本的頂端:

#coding=utf-8#!/usr/bin/python
# 導(dǎo)入模塊import support
# 現(xiàn)在可以調(diào)用模塊里包含的函數(shù)了
support.print_func("Zara")
以上實(shí)例輸出結(jié)果:

Hello : Zara

一個(gè)模塊只會(huì)被導(dǎo)入一次,不管你執(zhí)行了多少次import崭添。這樣可以防止導(dǎo)入模塊被一遍又一遍地執(zhí)行寓娩。

From…import 語句

Python的from語句讓你從模塊中導(dǎo)入一個(gè)指定的部分到當(dāng)前命名空間中。語法如下:

from modname import name1[, name2[, ... nameN]]

例如呼渣,要導(dǎo)入模塊fibfibonacci函數(shù)棘伴,使用如下語句:

from fib import fibonacci

這個(gè)聲明不會(huì)把整個(gè)fib模塊導(dǎo)入到當(dāng)前的命名空間中,它只會(huì)將fib里的fibonacci單個(gè)引入到執(zhí)行這個(gè)聲明的模塊的全局符號(hào)表屁置。

From…import* 語句

把一個(gè)模塊的所有內(nèi)容全都導(dǎo)入到當(dāng)前的命名空間也是可行的焊夸,只需使用如下聲明:

from modname import *

這提供了一個(gè)簡單的方法來導(dǎo)入一個(gè)模塊中的所有項(xiàng)目。然而這種聲明不該被過多地使用蓝角。

定位模塊

當(dāng)你導(dǎo)入一個(gè)模塊阱穗,Python解析器對(duì)模塊位置的搜索順序是:

  • 當(dāng)前目錄
  • 如果不在當(dāng)前目錄,Python則搜索在shell變量PYTHONPATH下的每個(gè)目錄使鹅。
  • 如果都找不到揪阶,Python會(huì)察看默認(rèn)路徑。UNIX下患朱,默認(rèn)路徑一般為/usr/local/lib/python/
    模塊搜索路徑存儲(chǔ)在system模塊的sys.path變量中遣钳。變量里包含當(dāng)前目錄,PYTHONPATH和由安裝過程決定的默認(rèn)目錄麦乞。

PYTHONPATH變量

作為環(huán)境變量蕴茴,PYTHONPATH由裝在一個(gè)列表里的許多目錄組成。PYTHONPATH的語法和shell變量PATH的一樣姐直。
在Windows系統(tǒng)倦淀,典型的PYTHONPATH如下:

set PYTHONPATH=c:\python20\lib;

在UNIX系統(tǒng),典型的PYTHONPATH如下:

set PYTHONPATH=/usr/local/lib/python

命名空間和作用域

變量是擁有匹配對(duì)象的名字(標(biāo)識(shí)符)声畏。命名空間是一個(gè)包含了變量名稱們(鍵)和它們各自相應(yīng)的對(duì)象們(值)的字典撞叽。
一個(gè)Python表達(dá)式可以訪問局部命名空間和全局命名空間里的變量。如果一個(gè)局部變量和一個(gè)全局變量重名插龄,則局部變量會(huì)覆蓋全局變量愿棋。
每個(gè)函數(shù)都有自己的命名空間。類的方法的作用域規(guī)則和通常函數(shù)的一樣均牢。
Python會(huì)智能地猜測一個(gè)變量是局部的還是全局的糠雨,它假設(shè)任何在函數(shù)內(nèi)賦值的變量都是局部的。
因此徘跪,如果要給全局變量在一個(gè)函數(shù)里賦值甘邀,必須使用global語句琅攘。
global VarName的表達(dá)式會(huì)告訴Python,VarName是一個(gè)全局變量松邪,這樣Python就不會(huì)在局部命名空間里尋找這個(gè)變量了坞琴。
例如,我們?cè)谌置臻g里定義一個(gè)變量money逗抑。我們?cè)僭诤瘮?shù)內(nèi)給變量money賦值剧辐,然后Python會(huì)假定money是一個(gè)局部變量。然而邮府,我們并沒有在訪問前聲明一個(gè)局部變量money浙于,結(jié)果就是會(huì)出現(xiàn)一個(gè)UnboundLocalError的錯(cuò)誤。取消global語句的注釋就能解決這個(gè)問題挟纱。

#coding=utf-8#!/usr/bin/python
 
Money = 2000
def AddMoney():
   # 想改正代碼就取消以下注釋:
   # global Money
   Money = Money + 1
 
print MoneyAddMoney()
print Money

dir()函數(shù)

dir()函數(shù)一個(gè)排好序的字符串列表,內(nèi)容是一個(gè)模塊里定義過的名字腐宋。
返回的列表容納了在一個(gè)模塊里定義的所有模塊紊服,變量和函數(shù)。如下一個(gè)簡單的實(shí)例:

#coding=utf-8#!/usr/bin/python
# 導(dǎo)入內(nèi)置math模塊import math 
content = dir(math)
print content;
以上實(shí)例輸出結(jié)果:
['__doc__', '__file__', '__name__', 'acos', 'asin', 'atan', 
'atan2', 'ceil', 'cos', 'cosh', 'degrees', 'e', 'exp', 
'fabs', 'floor', 'fmod', 'frexp', 'hypot', 'ldexp', 'log','log10', 'modf', 'pi', 'pow', 'radians', 'sin', 'sinh', 
'sqrt', 'tan', 'tanh']

在這里胸竞,特殊字符串變量__name__指向模塊的名字欺嗤,__file__指向該模塊的導(dǎo)入文件名。

globals()和locals()函數(shù)

根據(jù)調(diào)用地方的不同卫枝,globals()locals()函數(shù)可被用來返回全局和局部命名空間里的名字煎饼。
如果在函數(shù)內(nèi)部調(diào)用locals(),返回的是所有能在該函數(shù)里訪問的命名校赤。
如果在函數(shù)內(nèi)部調(diào)用globals()吆玖,返回的是所有在該函數(shù)里能訪問的全局名字。
兩個(gè)函數(shù)的返回類型都是字典马篮。所以名字們能用keys()函數(shù)摘取。

reload()函數(shù)

當(dāng)一個(gè)模塊被導(dǎo)入到一個(gè)腳本,模塊頂層部分的代碼只會(huì)被執(zhí)行一次峦剔。
因此赎败,如果你想重新執(zhí)行模塊里頂層部分的代碼,可以用reload()函數(shù)迁央。該函數(shù)會(huì)重新導(dǎo)入之前導(dǎo)入過的模塊掷匠。語法如下:

reload(module_name)

在這里,module_name要直接放模塊的名字岖圈,而不是一個(gè)字符串形式讹语。比如想重載hello模塊,如下:

reload(hello)

Python中的包

包是一個(gè)分層次的文件目錄結(jié)構(gòu)蜂科,它定義了一個(gè)由模塊及子包募强,和子包下的子包等組成的Python的應(yīng)用環(huán)境株灸。
考慮一個(gè)在Phone目錄下的pots.py文件。這個(gè)文件有如下源代碼:

#coding=utf-8#!/usr/bin/python
 
def Pots():
   print "I'm Pots Phone"

同樣地擎值,我們有另外兩個(gè)保存了不同函數(shù)的文件:

  • Phone/Isdn.py 含有函數(shù)Isdn()
  • Phone/G3.py 含有函數(shù)G3()
    現(xiàn)在慌烧,在Phone目錄下創(chuàng)建file __init__.py
  • Phone/init.py

當(dāng)你導(dǎo)入Phone時(shí),為了能夠使用所有函數(shù)鸠儿,你需要在__init__.py里使用顯式的導(dǎo)入語句屹蚊,如下:

from Pots import Potsfrom Isdn import Isdnfrom G3 import G3

當(dāng)你把這些代碼添加到init.py之后,導(dǎo)入Phone包的時(shí)候這些類就全都是可用的了进每。

#coding=utf-8#!/usr/bin/python
# Now import your Phone Package.import Phone
 
Phone.Pots()Phone.Isdn()Phone.G3()

以上實(shí)例輸出結(jié)果:

I'm Pots Phone
I'm 3G Phone
I'm ISDN Phone

如上汹粤,為了舉例,我們只在每個(gè)文件里放置了一個(gè)函數(shù)田晚,但其實(shí)你可以放置許多函數(shù)嘱兼。你也可以在這些文件里定義Python的類,然后為這些類建一個(gè)包贤徒。

Python 文件I/O

本章只講述所有基本的的I/O函數(shù)芹壕,更多函數(shù)請(qǐng)參考Python標(biāo)準(zhǔn)文檔。

打印到屏幕

最簡單的輸出方法是用print語句接奈,你可以給它傳遞零個(gè)或多個(gè)用逗號(hào)隔開的表達(dá)式踢涌。此函數(shù)把你傳遞的表達(dá)式轉(zhuǎn)換成一個(gè)字符串表達(dá)式,并將結(jié)果寫到標(biāo)準(zhǔn)輸出如下:

#!/usr/bin/python
print "Python is really a great language,", "isn't it?";

你的標(biāo)準(zhǔn)屏幕上會(huì)產(chǎn)生以下結(jié)果:

Python is really a great language, isn't it?

讀取鍵盤輸入

Python提供了兩個(gè)內(nèi)置函數(shù)從標(biāo)準(zhǔn)輸入讀入一行文本序宦,默認(rèn)的標(biāo)準(zhǔn)輸入是鍵盤睁壁。如下:

  • raw_input
  • input

raw_input函數(shù)

raw_input([prompt]) 函數(shù)從標(biāo)準(zhǔn)輸入讀取一個(gè)行,并返回一個(gè)字符串(去掉結(jié)尾的換行符):

#!/usr/bin/python
str = raw_input("Enter your input: ");
print "Received input is : ", str

這將提示你輸入任意字符串互捌,然后在屏幕上顯示相同的字符串潘明。當(dāng)我輸入"Hello Python!"秕噪,它的輸出如下:

Enter your input: Hello PythonReceived 
input is :  Hello Python

input函數(shù)

input([prompt]) 函數(shù)和raw_input([prompt]) 函數(shù)基本可以互換钉疫,但是input會(huì)假設(shè)你的輸入是一個(gè)有效的Python表達(dá)式,并返回運(yùn)算結(jié)果巢价。

#!/usr/bin/python
str = input("Enter your input: ");
print "Received input is : ", str

這會(huì)產(chǎn)生如下的對(duì)應(yīng)著輸入的結(jié)果:

Enter your input: [x*5 for x in range(2,10,2)]Recieved 
input is :  [10, 20, 30, 40]

打開和關(guān)閉文件

到現(xiàn)在為止牲阁,您已經(jīng)可以向標(biāo)準(zhǔn)輸入和輸進(jìn)行讀寫。現(xiàn)在壤躲,來看看怎么讀寫實(shí)際的數(shù)據(jù)文件城菊。
Python提供了必要的函數(shù)和方法進(jìn)行默認(rèn)情況下的文件基本操作。你可以用file對(duì)象做大部分的文件操作碉克。

open函數(shù)

你必須先用Python內(nèi)置的open()函數(shù)打開一個(gè)文件凌唬,創(chuàng)建一個(gè)file對(duì)象,相關(guān)的輔助方法才可以調(diào)用它進(jìn)行讀寫漏麦。
語法:

file object = open(file_name [, access_mode][, buffering])

各個(gè)參數(shù)的細(xì)節(jié)如下:

  • file_name:file_name變量是一個(gè)包含了你要訪問的文件名稱的字符串值客税。
  • access_mode:access_mode決定了打開文件的模式:只讀况褪,寫入,追加等更耻。所有可取值見如下的完全列表测垛。這個(gè)參數(shù)是非強(qiáng)制的,默認(rèn)文件訪問模式為只讀(r)秧均。
  • buffering:如果buffering的值被設(shè)為0食侮,就不會(huì)有寄存。如果buffering的值取1目胡,訪問文件時(shí)會(huì)寄存行锯七。如果將buffering的值設(shè)為大于1的整數(shù),表明了這就是的寄存區(qū)的緩沖大小誉己。如果取負(fù)值眉尸,寄存區(qū)的緩沖大小則為系統(tǒng)默認(rèn)。
    不同模式打開文件的完全列表:

模式 描述

  • r 以只讀方式打開文件巨双。文件的指針將會(huì)放在文件的開頭噪猾。這是默認(rèn)模式。
  • rb 以二進(jìn)制格式打開一個(gè)文件用于只讀炉峰。文件指針將會(huì)放在文件的開頭。這是默認(rèn)模式脉执。
  • r+ 打開一個(gè)文件用于讀寫疼阔。文件指針將會(huì)放在文件的開頭。
  • rb+ 以二進(jìn)制格式打開一個(gè)文件用于讀寫半夷。文件指針將會(huì)放在文件的開頭婆廊。
  • w 打開一個(gè)文件只用于寫入。如果該文件已存在則將其覆蓋巫橄。如果該文件不存在淘邻,創(chuàng)建新文件。
  • wb 以二進(jìn)制格式打開一個(gè)文件只用于寫入湘换。如果該文件已存在則將其覆蓋宾舅。如果該文件不存在,創(chuàng)建新文件彩倚。
  • w+ 打開一個(gè)文件用于讀寫筹我。如果該文件已存在則將其覆蓋。如果該文件不存在帆离,創(chuàng)建新文件蔬蕊。
  • wb+ 以二進(jìn)制格式打開一個(gè)文件用于讀寫。如果該文件已存在則將其覆蓋哥谷。如果該文件不存在岸夯,創(chuàng)建新文件麻献。
  • a 打開一個(gè)文件用于追加。如果該文件已存在猜扮,文件指針將會(huì)放在文件的結(jié)尾勉吻。也就是說,新的內(nèi)容將會(huì)被寫入到已有內(nèi)容之后破镰。如果該文件不存在餐曼,創(chuàng)建新文件進(jìn)行寫入。
  • ab 以二進(jìn)制格式打開一個(gè)文件用于追加鲜漩。如果該文件已存在源譬,文件指針將會(huì)放在文件的結(jié)尾。也就是說孕似,新的內(nèi)容將會(huì)被寫入到已有內(nèi)容之后踩娘。如果該文件不存在,創(chuàng)建新文件進(jìn)行寫入喉祭。
  • a+ 打開一個(gè)文件用于讀寫养渴。如果該文件已存在,文件指針將會(huì)放在文件的結(jié)尾泛烙。文件打開時(shí)會(huì)是追加模式理卑。如果該文件不存在,創(chuàng)建新文件用于讀寫蔽氨。
  • ab+ 以二進(jìn)制格式打開一個(gè)文件用于追加藐唠。如果該文件已存在,文件指針將會(huì)放在文件的結(jié)尾鹉究。如果該文件不存在宇立,創(chuàng)建新文件用于讀寫。

File對(duì)象的屬性
一個(gè)文件被打開后自赔,你有一個(gè)file對(duì)象妈嘹,你可以得到有關(guān)該文件的各種信息。
以下是和file對(duì)象相關(guān)的所有屬性的列表:

屬性 描述

  • file.closed 返回true如果文件已被關(guān)閉绍妨,否則返回false润脸。
  • file.mode 返回被打開文件的訪問模式。
  • file.name 返回文件的名稱他去。
  • file.softspace 如果用print輸出后津函,必須跟一個(gè)空格符,則返回false孤页。否則返回true尔苦。
    如下實(shí)例:
#coding=utf-8#!/usr/bin/python
# 打開一個(gè)文件
fo = open("foo.txt", "wb")print "Name of the file: ", fo.name
print "Closed or not : ", fo.closed
print "Opening mode : ", fo.mode
print "Softspace flag : ", fo.softspace

以上實(shí)例輸出結(jié)果:

Name of the file:  foo.txt
Closed or not :  FalseOpening mode :  wb
Softspace flag :  0

Close()方法

File對(duì)象的close()方法刷新緩沖區(qū)里任何還沒寫入的信息,并關(guān)閉該文件,這之后便不能再進(jìn)行寫入允坚。
當(dāng)一個(gè)文件對(duì)象的引用被重新指定給另一個(gè)文件時(shí)魂那,Python會(huì)關(guān)閉之前的文件。用close()方法關(guān)閉文件是一個(gè)很好的習(xí)慣稠项。
語法:

fileObject.close();

例子:

#coding=utf-8#!/usr/bin/python
 
# 打開一個(gè)文件
fo = open("foo.txt", "wb")print "Name of the file: ", fo.name
# 關(guān)閉打開的文件
fo.close()

以上實(shí)例輸出結(jié)果:

Name of the file:  foo.txt

讀寫文件:
file對(duì)象提供了一系列方法涯雅,能讓我們的文件訪問更輕松。來看看如何使用read()和write()方法來讀取和寫入文件展运。

Write()方法

Write()方法可將任何字符串寫入一個(gè)打開的文件活逆。需要重點(diǎn)注意的是,Python字符串可以是二進(jìn)制數(shù)據(jù)拗胜,而不是僅僅是文字蔗候。
Write()方法不在字符串的結(jié)尾不添加換行符('\n'):
語法:

fileObject.write(string);

在這里,被傳遞的參數(shù)是要寫入到已打開文件的內(nèi)容埂软。
例子:

#coding=utf-8#!/usr/bin/python
# 打開一個(gè)文件
fo = open("/tmp/foo.txt", "wb")
fo.write( "Python is a great language.\nYeah its great!!\n");
 
# 關(guān)閉打開的文件
fo.close()

上述方法會(huì)創(chuàng)建foo.txt文件锈遥,并將收到的內(nèi)容寫入該文件,并最終關(guān)閉文件勘畔。如果你打開這個(gè)文件所灸,將看到以下內(nèi)容:

Python is a great language.Yeah its great!!

read()方法

read()方法從一個(gè)打開的文件中讀取一個(gè)字符串。需要重點(diǎn)注意的是炫七,Python字符串可以是二進(jìn)制數(shù)據(jù)爬立,而不是僅僅是文字。
語法:

fileObject.read([count]);

在這里万哪,被傳遞的參數(shù)是要從已打開文件中讀取的字節(jié)計(jì)數(shù)侠驯。該方法從文件的開頭開始讀入,如果沒有傳入count壤圃,它會(huì)嘗試盡可能多地讀取更多的內(nèi)容陵霉,很可能是直到文件的末尾琅轧。
例子:
就用我們上面創(chuàng)建的文件foo.txt伍绳。

#coding=utf-8#!/usr/bin/python
 
# 打開一個(gè)文件
fo = open("/tmp/foo.txt", "r+")
str = fo.read(10);print "Read String is : ", str
# 關(guān)閉打開的文件
fo.close()
以上實(shí)例輸出結(jié)果:
Read String is :  Python is

文件位置:

Tell()方法告訴你文件內(nèi)的當(dāng)前位置;換句話說乍桂,下一次的讀寫會(huì)發(fā)生在文件開頭這么多字節(jié)之后:
seek(offset [,from])方法改變當(dāng)前文件的位置冲杀。Offset變量表示要移動(dòng)的字節(jié)數(shù)。From變量指定開始移動(dòng)字節(jié)的參考位置睹酌。
如果from被設(shè)為0权谁,這意味著將文件的開頭作為移動(dòng)字節(jié)的參考位置。如果設(shè)為1憋沿,則使用當(dāng)前的位置作為參考位置旺芽。如果它被設(shè)為2,那么該文件的末尾將作為參考位置。
例子:
就用我們上面創(chuàng)建的文件foo.txt采章。

#coding=utf-8#!/usr/bin/python
 
# 打開一個(gè)文件
fo = open("/tmp/foo.txt", "r+")
str = fo.read(10)
print "Read String is : ", str
 
# 查找當(dāng)前位置
position = fo.tell();print "Current file position : ", position
 
# 把指針再次重新定位到文件開頭
position = fo.seek(0, 0);
str = fo.read(10);print "Again read String is : ", str
# 關(guān)閉打開的文件
fo.close()
以上實(shí)例輸出結(jié)果:

Read String is :  Python isCurrent file position :  10Again read String is :  Python is

重命名和刪除文件

Python的os模塊提供了幫你執(zhí)行文件處理操作的方法运嗜,比如重命名和刪除文件。
要使用這個(gè)模塊悯舟,你必須先導(dǎo)入它担租,然后可以調(diào)用相關(guān)的各種功能。

rename()方法:

rename()方法需要兩個(gè)參數(shù)抵怎,當(dāng)前的文件名和新文件名奋救。
語法:

os.rename(current_file_name, new_file_name)

例子:
下例將重命名一個(gè)已經(jīng)存在的文件test1.txt。

#coding=utf-8#!/usr/bin/pythonimport os
 
# 重命名文件test1.txt到test2.txt反惕。
os.rename( "test1.txt", "test2.txt" )

remove()方法

你可以用remove()方法刪除文件尝艘,需要提供要?jiǎng)h除的文件名作為參數(shù)。
語法:

os.remove(file_name)

例子:
下例將刪除一個(gè)已經(jīng)存在的文件test2.txt承璃。

#coding=utf-8#!/usr/bin/pythonimport os
 
# 刪除一個(gè)已經(jīng)存在的文件test2.txt
os.remove("text2.txt")

Python里的目錄:
所有文件都包含在各個(gè)不同的目錄下利耍,不過Python也能輕松處理。os模塊有許多方法能幫你創(chuàng)建盔粹,刪除和更改目錄隘梨。

mkdir()方法

可以使用os模塊的mkdir()方法在當(dāng)前目錄下創(chuàng)建新的目錄們。你需要提供一個(gè)包含了要?jiǎng)?chuàng)建的目錄名稱的參數(shù)舷嗡。
語法:

os.mkdir("newdir")

例子:
下例將在當(dāng)前目錄下創(chuàng)建一個(gè)新目錄test轴猎。

#coding=utf-8#!/usr/bin/pythonimport os
 
# 創(chuàng)建目錄test
os.mkdir("test")

chdir()方法

可以用chdir()方法來改變當(dāng)前的目錄。chdir()方法需要的一個(gè)參數(shù)是你想設(shè)成當(dāng)前目錄的目錄名稱进萄。
語法:

os.chdir("newdir")

例子:
下例將進(jìn)入"/home/newdir"目錄捻脖。

#coding=utf-8#!/usr/bin/pythonimport os
 
# 將當(dāng)前目錄改為"/home/newdir"
os.chdir("/home/newdir")

getcwd()方法:

getcwd()方法顯示當(dāng)前的工作目錄。
語法:

os.getcwd()

例子:
下例給出當(dāng)前目錄:

#coding=utf-8#!/usr/bin/pythonimport os
 
# 給出當(dāng)前的目錄
os.getcwd()

rmdir()方法

rmdir()方法刪除目錄中鼠,目錄名稱以參數(shù)傳遞可婶。
在刪除這個(gè)目錄之前,它的所有內(nèi)容應(yīng)該先被清除援雇。
語法:

os.rmdir('dirname')

例子:
以下是刪除" /tmp/test"目錄的例子矛渴。目錄的完全合規(guī)的名稱必須被給出,否則會(huì)在當(dāng)前目錄下搜索該目錄惫搏。

#coding=utf-8#!/usr/bin/pythonimport os
 
# 刪除”/tmp/test”目錄
os.rmdir( "/tmp/test"  )

文件具温、目錄相關(guān)的方法

三個(gè)重要的方法來源能對(duì)Windows和Unix操作系統(tǒng)上的文件及目錄進(jìn)行一個(gè)廣泛且實(shí)用的處理及操控,如下:

  • File 對(duì)象方法: file對(duì)象提供了操作文件的一系列方法筐赔。
  • OS 對(duì)象方法: 提供了處理文件及目錄的一系列方法铣猩。

Python 異常處理

python提供了兩個(gè)非常重要的功能來處理python程序在運(yùn)行中出現(xiàn)的異常和錯(cuò)誤。你可以使用該功能來調(diào)試python程序茴丰。

  • 異常處理: 本站Python教程會(huì)具體介紹达皿。
  • 斷言(Assertions):本站Python教程會(huì)具體介紹天吓。

python標(biāo)準(zhǔn)異常

異常名稱 描述

  • BaseException 所有異常的基類
  • SystemExit 解釋器請(qǐng)求退出
  • KeyboardInterrupt 用戶中斷執(zhí)行(通常是輸入^C)
  • Exception 常規(guī)錯(cuò)誤的基類
  • StopIteration 迭代器沒有更多的值
  • GeneratorExit 生成器(generator)發(fā)生異常來通知退出
  • StandardError 所有的內(nèi)建標(biāo)準(zhǔn)異常的基類
  • ArithmeticError 所有數(shù)值計(jì)算錯(cuò)誤的基類
  • FloatingPointError 浮點(diǎn)計(jì)算錯(cuò)誤
  • OverflowError 數(shù)值運(yùn)算超出最大限制
  • ZeroDivisionError 除(或取模)零 (所有數(shù)據(jù)類型)
  • AssertionError 斷言語句失敗
  • AttributeError 對(duì)象沒有這個(gè)屬性
  • EOFError 沒有內(nèi)建輸入,到達(dá)EOF 標(biāo)記
  • EnvironmentError 操作系統(tǒng)錯(cuò)誤的基類
  • IOError 輸入/輸出操作失敗
  • OSError 操作系統(tǒng)錯(cuò)誤
  • WindowsError 系統(tǒng)調(diào)用失敗
  • ImportError 導(dǎo)入模塊/對(duì)象失敗
  • LookupError 無效數(shù)據(jù)查詢的基類
  • IndexError 序列中沒有此索引(index)
  • KeyError 映射中沒有這個(gè)鍵
  • MemoryError 內(nèi)存溢出錯(cuò)誤(對(duì)于Python 解釋器不是致命的)
  • NameError 未聲明/初始化對(duì)象 (沒有屬性)
  • UnboundLocalError 訪問未初始化的本地變量
  • ReferenceError 弱引用(Weak reference)試圖訪問已經(jīng)垃圾回收了的對(duì)象
  • RuntimeError 一般的運(yùn)行時(shí)錯(cuò)誤
  • NotImplementedError 尚未實(shí)現(xiàn)的方法
  • SyntaxError Python 語法錯(cuò)誤
  • IndentationError 縮進(jìn)錯(cuò)誤
  • TabError Tab 和空格混用
  • SystemError 一般的解釋器系統(tǒng)錯(cuò)誤
  • TypeError 對(duì)類型無效的操作
  • ValueError 傳入無效的參數(shù)
  • UnicodeError Unicode 相關(guān)的錯(cuò)誤
  • UnicodeDecodeError Unicode 解碼時(shí)的錯(cuò)誤
  • UnicodeEncodeError Unicode 編碼時(shí)錯(cuò)誤
  • UnicodeTranslateError Unicode 轉(zhuǎn)換時(shí)錯(cuò)誤
  • Warning 警告的基類
  • DeprecationWarning 關(guān)于被棄用的特征的警告
  • FutureWarning 關(guān)于構(gòu)造將來語義會(huì)有改變的警告
  • OverflowWarning 舊的關(guān)于自動(dòng)提升為長整型(long)的警告
  • PendingDeprecationWarning 關(guān)于特性將會(huì)被廢棄的警告
  • RuntimeWarning 可疑的運(yùn)行時(shí)行為(runtime behavior)的警告
  • SyntaxWarning 可疑的語法的警告
  • UserWarning 用戶代碼生成的警告

什么是異常?
異常即是一個(gè)事件峦椰,該事件會(huì)在程序執(zhí)行過程中發(fā)生失仁,影響了程序的正常執(zhí)行。
一般情況下们何,在Python無法正常處理程序時(shí)就會(huì)發(fā)生一個(gè)異常萄焦。
異常是Python對(duì)象,表示一個(gè)錯(cuò)誤冤竹。
當(dāng)Python腳本發(fā)生異常時(shí)我們需要捕獲處理它拂封,否則程序會(huì)終止執(zhí)行。

異常處理

捕捉異仇腥洌可以使用try/except語句冒签。
try/except語句用來檢測try語句塊中的錯(cuò)誤,從而讓except語句捕獲異常信息并處理钟病。
如果你不想在異常發(fā)生時(shí)結(jié)束你的程序萧恕,只需在try里捕獲它。
語法:
以下為簡單的

try....
except...
else

的語法:

try:<語句>   
#運(yùn)行別的代碼
except <名字>:
<語句>        
#如果在try部份引發(fā)了'name'異常
except <名字>肠阱,<數(shù)據(jù)>:<語句>        
#如果引發(fā)了'name'異常票唆,獲得附加的數(shù)據(jù)
else:<語句>        
#如果沒有異常發(fā)生

try的工作原理是,當(dāng)開始一個(gè)try語句后屹徘,python就在當(dāng)前程序的上下文中作標(biāo)記走趋,這樣當(dāng)異常出現(xiàn)時(shí)就可以回到這里,try子句先執(zhí)行噪伊,接下來會(huì)發(fā)生什么依賴于執(zhí)行時(shí)是否出現(xiàn)異常簿煌。

  • 如果當(dāng)try后的語句執(zhí)行時(shí)發(fā)生異常,python就跳回到try并執(zhí)行第一個(gè)匹配該異常的except子句鉴吹,異常處理完畢姨伟,控制流就通過整個(gè)try語句(除非在處理異常時(shí)又引發(fā)新的異常)。
  • 如果在try后的語句里發(fā)生了異常豆励,卻沒有匹配的except子句夺荒,異常將被遞交到上層的try,或者到程序的最上層(這樣將結(jié)束程序肆糕,并打印缺省的出錯(cuò)信息)般堆。
  • 如果在try子句執(zhí)行時(shí)沒有發(fā)生異常在孝,python將執(zhí)行else語句后的語句(如果有else的話)诚啃,然后控制流通過整個(gè)try語句。
    實(shí)例

下面是簡單的例子私沮,它打開一個(gè)文件始赎,在該文件中的內(nèi)容寫入內(nèi)容,且并未發(fā)生異常:

#!/usr/bin/python

try:
   fh = open("testfile", "w")
   fh.write("This is my test file for exception handling!!")
except IOError:
   print "Error: can\'t find file or read data"else:
   print "Written content in the file successfully"
   fh.close()
以上程序輸出結(jié)果:

 Written content in the file successfully

實(shí)例
下面是簡單的例子,它打開一個(gè)文件造垛,在該文件中的內(nèi)容寫入內(nèi)容魔招,但文件沒有寫入權(quán)限,發(fā)生了異常:

#!/usr/bin/python

try:
   fh = open("testfile", "w")
   fh.write("This is my test file for exception handling!!")
except IOError:
   print "Error: can\'t find file or read data"else:
   print "Written content in the file successfully"
以上程序輸出結(jié)果:

Error: can't find file or read data

使用except而不帶任何異常類型
你可以不帶任何異常類型使用except五辽,如下實(shí)例:

try:
   You do your operations here;
   ......................
except:
   If there is any exception, then execute this block.
   ......................
else:
   If there is no exception then execute this block. 

以上方式try-except語句捕獲所有發(fā)生的異常办斑。但這不是一個(gè)很好的方式,我們不能通過該程序識(shí)別出具體的異常信息杆逗。因?yàn)樗东@所有的異常乡翅。

使用except而帶多種異常類型
你也可以使用相同的except語句來處理多個(gè)異常信息,如下所示:

try:
   You do your operations here;
   ......................
except(Exception1[, Exception2[,...ExceptionN]]]):
   If there is any exception from the given exception list, 
   then execute this block.
   ......................
else:
   If there is no exception then execute this block.  

try-finally 語句

try-finally 語句無論是否發(fā)生異常都將執(zhí)行最后的代碼罪郊。  
try:<語句>finally:<語句>    
#退出try時(shí)總會(huì)執(zhí)行raise

注意:你可以使用except語句或者finally語句蠕蚜,但是兩者不能同時(shí)使用。else語句也不能與finally語句同時(shí)使用
實(shí)例

#!/usr/bin/python

try:
   fh = open("testfile", "w")
   fh.write("This is my test file for exception handling!!")
finally:
   print "Error: can\'t find file or read data"

如果打開的文件沒有可寫權(quán)限悔橄,輸出如下所示:

Error: can't find file or read data

同樣的例子也可以寫成如下方式:

#!/usr/bin/python

try:
    fh = open("testfile", "w")
    try:
        fh.write("This is my test file for exception handling!!")
    finally:
        print "Going to close the file"
        fh.close()
    except IOError:
    print "Error: can\'t find file or read data"

當(dāng)在try塊中拋出一個(gè)異常靶累,立即執(zhí)行finally塊代碼。
finally塊中的所有語句執(zhí)行后癣疟,異常被再次提出挣柬,并執(zhí)行except塊代碼。
參數(shù)的內(nèi)容不同于異常睛挚。

異常的參數(shù)

一個(gè)異沉莘蓿可以帶上參數(shù),可作為輸出的異常信息參數(shù)竞川。
你可以通過except語句來捕獲異常的參數(shù)店溢,如下所示:

try:
   You do your operations here;
   ......................
except ExceptionType, Argument:
   You can print value of Argument here...

變量接收的異常值通常包含在異常的語句中。在元組的表單中變量可以接收一個(gè)或者多個(gè)值委乌。
元組通常包含錯(cuò)誤字符串床牧,錯(cuò)誤數(shù)字,錯(cuò)誤位置遭贸。
實(shí)例
以下為單個(gè)異常的實(shí)例:

#!/usr/bin/python

# Define a function here.def temp_convert(var):
   try:
      return int(var)
   except ValueError, Argument:
      print "The argument does not contain numbers\n", Argument

# Call above function here.
temp_convert("xyz");

以上程序執(zhí)行結(jié)果如下:

The argument does not contain numbers
invalid literal for int() with base 10: 'xyz'

觸發(fā)異常

我們可以使用raise語句自己觸發(fā)異常
raise語法格式如下:

raise [Exception [, args [, traceback]]]

語句中Exception是異常的類型(例如戈咳,NameError)參數(shù)是一個(gè)異常參數(shù)值。該參數(shù)是可選的壕吹,如果不提供著蛙,異常的參數(shù)是"None"。
最后一個(gè)參數(shù)是可選的(在實(shí)踐中很少使用)耳贬,如果存在踏堡,是跟蹤異常對(duì)象。
實(shí)例
一個(gè)異持渚ⅲ可以是一個(gè)字符串顷蟆,類或?qū)ο蟆?Python的內(nèi)核提供的異常诫隅,大多數(shù)都是實(shí)例化的類,這是一個(gè)類的實(shí)例的參數(shù)帐偎。
定義一個(gè)異常非常簡單逐纬,如下所示:

def functionName( level ):
   if level < 1:
      raise "Invalid level!", level
      # The code below to this would not be executed
      # if we raise the exception

注意:為了能夠捕獲異常,"except"語句必須有用相同的異常來拋出類對(duì)象或者字符串削樊。

例如我們捕獲以上異常豁生,"except"語句如下所示:

try:
   Business Logic here...
except "Invalid level!":
   Exception handling here...else:
   Rest of the code here...

用戶自定義異常

通過創(chuàng)建一個(gè)新的異常類,程序可以命名它們自己的異常漫贞。異常應(yīng)該是典型的繼承自Exception類沛硅,通過直接或間接的方式。
以下為與RuntimeError相關(guān)的實(shí)例,實(shí)例中創(chuàng)建了一個(gè)類绕辖,基類為RuntimeError摇肌,用于在異常觸發(fā)時(shí)輸出更多的信息。
在try語句塊中仪际,用戶自定義的異常后執(zhí)行except塊語句围小,變量 e 是用于創(chuàng)建Networkerror類的實(shí)例。

class Networkerror(RuntimeError):
   def __init__(self, arg):
      self.args = arg

在你定義以上類后树碱,你可以觸發(fā)該異常肯适,如下所示:

try:
   raise Networkerror("Bad hostname")
except Networkerror,e:
   print e.args
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市成榜,隨后出現(xiàn)的幾起案子框舔,更是在濱河造成了極大的恐慌,老刑警劉巖赎婚,帶你破解...
    沈念sama閱讀 218,682評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件刘绣,死亡現(xiàn)場離奇詭異,居然都是意外死亡挣输,警方通過查閱死者的電腦和手機(jī)纬凤,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,277評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來撩嚼,“玉大人停士,你說我怎么就攤上這事⊥昀觯” “怎么了恋技?”我有些...
    開封第一講書人閱讀 165,083評(píng)論 0 355
  • 文/不壞的土叔 我叫張陵,是天一觀的道長逻族。 經(jīng)常有香客問我蜻底,道長,這世上最難降的妖魔是什么瓷耙? 我笑而不...
    開封第一講書人閱讀 58,763評(píng)論 1 295
  • 正文 為了忘掉前任朱躺,我火速辦了婚禮,結(jié)果婚禮上搁痛,老公的妹妹穿的比我還像新娘长搀。我一直安慰自己,他們只是感情好鸡典,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,785評(píng)論 6 392
  • 文/花漫 我一把揭開白布源请。 她就那樣靜靜地躺著,像睡著了一般彻况。 火紅的嫁衣襯著肌膚如雪谁尸。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,624評(píng)論 1 305
  • 那天纽甘,我揣著相機(jī)與錄音良蛮,去河邊找鬼。 笑死悍赢,一個(gè)胖子當(dāng)著我的面吹牛决瞳,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播左权,決...
    沈念sama閱讀 40,358評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼皮胡,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了赏迟?” 一聲冷哼從身側(cè)響起屡贺,我...
    開封第一講書人閱讀 39,261評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎锌杀,沒想到半個(gè)月后甩栈,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,722評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡糕再,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,900評(píng)論 3 336
  • 正文 我和宋清朗相戀三年谤职,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片亿鲜。...
    茶點(diǎn)故事閱讀 40,030評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡允蜈,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出蒿柳,到底是詐尸還是另有隱情饶套,我是刑警寧澤,帶...
    沈念sama閱讀 35,737評(píng)論 5 346
  • 正文 年R本政府宣布垒探,位于F島的核電站妓蛮,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏圾叼。R本人自食惡果不足惜蛤克,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,360評(píng)論 3 330
  • 文/蒙蒙 一捺癞、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧构挤,春花似錦髓介、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,941評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至矾飞,卻和暖如春一膨,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背洒沦。 一陣腳步聲響...
    開封第一講書人閱讀 33,057評(píng)論 1 270
  • 我被黑心中介騙來泰國打工豹绪, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人申眼。 一個(gè)月前我還...
    沈念sama閱讀 48,237評(píng)論 3 371
  • 正文 我出身青樓森篷,卻偏偏與公主長得像,于是被迫代替她去往敵國和親豺型。 傳聞我的和親對(duì)象是個(gè)殘疾皇子仲智,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,976評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • 前言 ||| 第二章 使用ArcPy編寫腳本 Python支持大部分在其他語言中出現(xiàn)的編程結(jié)構(gòu)。在本章內(nèi)容中姻氨,我們...
    muyan閱讀 90,048評(píng)論 10 55
  • Python進(jìn)階框架 希望大家喜歡钓辆,點(diǎn)贊哦首先感謝廖雪峰老師對(duì)于該課程的講解 一、函數(shù)式編程 1.1 函數(shù)式編程簡...
    Gaolex閱讀 5,499評(píng)論 6 53
  • 要點(diǎn): 函數(shù)式編程:注意不是“函數(shù)編程”肴焊,多了一個(gè)“式” 模塊:如何使用模塊 面向?qū)ο缶幊蹋好嫦驅(qū)ο蟮母拍钋傲傩浴?..
    victorsungo閱讀 1,510評(píng)論 0 6
  • W先生: 你好!時(shí)間過得真是快娶眷,這一轉(zhuǎn)眼已經(jīng)給你寫了一個(gè)星期的信了似嗤,我喜歡這樣的交流,我想我會(huì)堅(jiān)持下去届宠,期...
    趙卓然閱讀 272評(píng)論 1 1
  • 贊美柴靜的人很多豌注,貶低她的人也不少伤塌。 但翻開她的履歷,從湖南廣播電視總臺(tái)文藝廣播節(jié)目《夜色溫柔》的主持人轧铁,到去中國...
    滾滾的小石頭閱讀 333評(píng)論 0 0