[上一講](https://github.com/qiwsir/ITArticles/blob/master/BasicPython/217.md)中創建了類,并且重點講述了構造函數以及類實例,特別是對那個self,描述了不少。在講述構造函數的時候特別提到,**init**()是一個函數,只不過在類中有一點特殊的作用罷了,每個類,首先要運行它,它規定了類的基本結構。
## 數據流轉過程
除了在類中可以寫這種函數之外,在類中還可以寫別的函數,延續上一講的例子:
~~~
#!/usr/bin/env python
#coding:utf-8
class Person:
def __init__(self, name, lang="golang", website="www.google.com"):
self.name = name
self.lang = lang
self.website = website
self.email = "qiwsir@gmail.com"
def author(self):
return self.name
laoqi = Person("LaoQi")
info = Person("qiwsir",lang="python",website="qiwsir.github.io")
print "Author name from laoqi:",laoqi.author()
print "Author name from info:",info.author()
#運行結果
Author name from laoqi: LaoQi
Author name from info: qiwsir
~~~
看官可能已經注意了,這段代碼比上一講多了一個函數author(self),這個我們先不管,稍后會詳細分解。首先看看數據是如何在這個代碼中流轉的。為了能夠清楚,畫一張圖,所謂一圖勝千言萬語,有圖有真相。
[](https://github.com/qiwsir/ITArticles/blob/master/Pictures/21801.png)
定義類Person,然后創建實例laoqi=Person("LaoQi"),看官注意觀察圖上的箭頭方向。laoqi這個實例和Person類中的self對應,它們都是引用了實例對象(很多時候簡化說成是實例對象)。"LaoQi"是一個具體的數據,通過構造函數中的name參數,傳給實例的屬性self.name,在類Person中的另外一個方法author的參數列表中第一個就是self,表示要承接self對象,return self.name,就是在類內部通過self對象,把它的屬性self.name的數據傳導如author。
當運行laoqi.author()的時候,就是告訴上面的代碼,調用laoqi實例對象,并得到author()方法的結果,laoqi這個實例就自動被告訴了author()(注意,self參數在這里不用寫,這個告訴過程是python自動完成的,不用我們操心了),author方法就返回laoqi實例的屬性,因為前面已經完成了laoqi與self的對應過程,所以這時候author里面的self就是laoqi,自然self.name=laoqi.name。
看官可以跟隨我在做一個實驗,那就是在author中,return laoqi.name,看看什么效果。因為既然laoqi和self是同一個實例對象,直接寫成laoqi.name是不是也可以呢?
~~~
#!/usr/bin/env python
#coding:utf-8
class Person:
def __init__(self, name, lang="golang", website="www.google.com"):
self.name = name
self.lang = lang
self.website = website
self.email = "qiwsir@gmail.com"
def author(self):
#return self.name
return laoqi.name #返回
laoqi = Person("LaoQi")
info = Person("qiwsir",lang="python",website="qiwsir.github.io")
print "Author name from laoqi:",laoqi.author()
print "Author name from info:",info.author()
#輸出結果
Author name from laoqi: LaoQi #laoqi實例輸出結果
Author name from info: LaoQi #info實例輸出結果
~~~
從結果中可以看出,沒有報錯。但是,info這個實例輸出的結果和laoqi實例輸出的結果一樣。原來,當調用了info實例之后,運行到author(),返回的是laoqi.name。所以,這里一定要用self實例。在調用不同的實例時,self會自動的進行匹配,當然,匹配過程是python完成,仍然不用我們操心。
OK,數據流轉過程,看官是否理解了呢?下面進入方法編寫的環節
## 為什么要用到方法
在類里面,可以用def語句來編寫函數,但是,通常這個函數的樣子是這樣的:
~~~
class ClassName:
def __init__(self,*args):
...
def method(self,*args): #是一個在類里面的函數
...
~~~
在類ClassName里面,除了前面那個具有初始化功能的構造函數之外,還有一個函數method,這個函數和以前學習過的函數一樣,函數里面要寫什么,也沒有特別的規定。但是,這個函數的第一個參數必須是self,或者說,可以沒有別的參數,但是self是必須寫上并且是第一個。這個self參數的作用前面已經說過了。
這樣看來,類里面的這個函數還有點跟以前函數不同的地方。
類里面的這個函數,我們就稱之為**方法**。
之所以用方法,也是用類的原因,也是用函數的原因,都是為了減少代碼的冗余,提高代碼的重用性,這也是OOP的原因。
方法怎樣被重用呢?看本最開始的那段代碼,里面有一個author方法,不管是laoqi還是info實例,都用這個方法返回實例導入的名字。這就是體現了重用。
## 編寫和操作方法
編寫方法的過程和編寫一個函數的過程一樣,需要注意的就是要在參數列表中第一個寫上self,即使沒有其它的參數。
~~~
#!/usr/bin/env python
#coding:utf-8
class Person:
def __init__(self, name, lang="golang", website="www.google.com"):
self.name = name
self.lang = lang
self.website = website
self.email = "qiwsir@gmail.com"
def author(self, address):
#return self.name
return laoqi.name+" in "+address
laoqi = Person("LaoQi")
info = Person("qiwsir",lang="python",website="qiwsir.github.io")
print "Author name from laoqi:",laoqi.author("China")
print "Author name from info:",info.author("Suzhou")
#運行結果
Author name from laoqi: LaoQi in China
Author name from info: LaoQi in Suzhou
~~~
這段代碼中,對author方法增加了一個參數address,當調用這個方法的時候:laoqi.author("China"),要對這個參數賦值,看官特別注意,在類中,這個方法顯示是有兩個參數(self,address),但是在調用的時候,第一個參數是自動將實例laoqi與之對應起來,不需要顯化賦值,可以理解成是隱含完成的(其實,也可以將laoqi看做隱藏的主體,偷偷地跟self勾搭上了)。
通過上面的講述,看官可以試試類了。提醒,一定要對類的數據流通過程清晰。
- 第零部分 獨上高樓,望盡天涯路
- 嘮叨一些關于Python的事情
- 為什么要開設本欄目
- 第一部分 積小流,至江海
- Python環境安裝
- 集成開發環境(IDE)
- 數的類型和四則運算
- 啰嗦的除法
- 開始真正編程
- 初識永遠強大的函數
- 玩轉字符串(1):基本概念、字符轉義、字符串連接、變量與字符串關系
- 玩轉字符串(2)
- 玩轉字符串(3)
- 眼花繚亂的運算符
- 從if開始語句的征程
- 一個免費的實驗室
- 有容乃大的list(1)
- 有容乃大的list(2)
- 有容乃大的list(3)
- 有容乃大的list(4)
- list和str比較
- 畫圈還不簡單嗎
- 再深點,更懂list
- 字典,你還記得嗎?
- 字典的操作方法
- 有點簡約的元組
- 一二三,集合了
- 集合的關系
- Python數據類型總結
- 深入變量和引用對象
- 賦值,簡單也不簡單
- 坑爹的字符編碼
- 做一個小游戲
- 不要紅頭文件(1): open, write, close
- 不要紅頭文件(2): os.stat, closed, mode, read, readlines, readline
- 第二部分 窮千里目,上一層樓
- 正規地說一句話
- print能干的事情
- 從格式化表達式到方法
- 復習if語句
- 用while來循環
- 難以想象的for
- 關于循環的小伎倆
- 讓人歡喜讓人憂的迭代
- 大話題小函數(1)
- 大話題小函數(2)
- python文檔
- 重回函數
- 變量和參數
- 總結參數的傳遞
- 傳說中的函數條規
- 關于類的基本認識
- 編寫類之一創建實例
- 編寫類之二方法
- 編寫類之三子類
- 編寫類之四再論繼承
- 命名空間
- 類的細節
- Import 模塊
- 模塊的加載
- 私有和專有
- 折騰一下目錄: os.path.<attribute>
- 第三部分 昨夜西風,亭臺誰登
- 網站的結構:網站組成、MySQL數據庫的安裝和配置、MySQL的運行
- 通過Python連接數據庫:安裝python-MySQLdb,連接MySQL
- 用Pyton操作數據庫(1):建立連接和游標,并insert and commit
- 用Python操作數據庫(2)
- 用Python操作數據庫(3)
- python開發框架:框架介紹、Tornado安裝
- Hello,第一個網頁分析:tornado網站的基本結構剖析:improt模塊、RequestHandler, HTTPServer, Application, IOLoop
- 實例分析get和post:get()通過URL得到數據和post()通過get_argument()獲取數據
- 問候世界:利用GAE建立tornado框架網站
- 使用表單和模板:tornado模板self.render和模板變量傳遞
- 模板中的語法:tornado模板中的for,if,set等語法
- 靜態文件以及一個項目框架
- 模板轉義
- 第四部分 暮然回首,燈火闌珊處
- requests庫
- 比較json/dictionary的庫
- defaultdict 模塊和 namedtuple 模塊
- 第五部分 Python備忘錄
- 基本的(字面量)值
- 運算符
- 常用的內建函數
- 擴展閱讀(來自網絡文章)
- 人生苦短,我用Python