[TOC]
### 綁定方法
正常情況下,當我們定義了一個`class`,創建了一個`class`的實例后,我們可以給該實例綁定任何屬性和方法,這就是動態語言的靈活性。先定義`class`:
~~~
class Student(object):
pass
~~~
然后,嘗試給實例綁定一個屬性:
~~~
>>> s = Student()
>>> s.name = 'Michael' # 動態給實例綁定一個屬性
>>> print(s.name)
Michael
~~~
還可以嘗試給實例綁定一個方法:
~~~
>>> def set_age(self, age): # 定義一個函數作為實例方法
... self.age = age
...
>>> from types import MethodType
>>> s.set_age = MethodType(set_age, s) # 給實例綁定一個方法
>>> s.set_age(25) # 調用實例方法
>>> s.age # 測試結果
25
~~~
但是,給一個實例綁定的方法,對另一個實例是不起作用的:
~~~
>>> s2 = Student() # 創建新的實例
>>> s2.set_age(25) # 嘗試調用方法
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: 'Student' object has no attribute 'set_age'
~~~
為了給所有實例都綁定方法,可以給class綁定方法:
~~~
>>> def set_score(self, score):
... self.score = score
...
>>> Student.set_score = set_score
~~~
給class綁定方法后,所有實例均可調用:
~~~
>>> s.set_score(100)
>>> s.score
100
>>> s2.set_score(99)
>>> s2.score
99
~~~
通常情況下,上面的`set_score`方法可以直接定義在`class`中,但動態綁定允許我們在程序運行的過程中動態`給class`加上功能,這在靜態語言中很難實現。
### 使用__slots__
但是,如果我們想要限制實例的屬性怎么辦?比如,只允許對`Student`實例添加`name`和`age`屬性。
為了達到限制的目的,Python允許在定義`class`的時候,定義一個特殊的`__slots__`變量,來限制該`class`實例能添加的屬性:
~~~
class Student(object):
__slots__ = ('name', 'age') # 用tuple定義允許綁定的屬性名稱
~~~
然后,我們試試:
~~~
>>> s = Student() # 創建新的實例
>>> s.name = 'Michael' # 綁定屬性'name'
>>> s.age = 25 # 綁定屬性'age'
>>> s.score = 99 # 綁定屬性'score'
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: 'Student' object has no attribute 'score'
~~~
由于`'score'`沒有被放到`__slots__`中,所以不能綁定`score`屬性,試圖綁定`score`將得到`AttributeError`的錯誤。
#### 注意
使用`__slots__`要注意,`__slots__`定義的屬性僅對當前類實例起作用,對繼承的子類是不起作用的:
~~~
>>> class GraduateStudent(Student):
... pass
...
>>> g = GraduateStudent()
>>> g.score = 9999
~~~
除非在子類中也定義`__slots__`,這樣,子類實例允許定義的屬性就是自身的`__slots__`加上父類的`__slots__`。
- Python教程
- Python簡介
- 安裝Python
- Python解釋器
- 第一個 Python 程序
- 使用文本編輯器
- Python代碼運行助手
- 輸入和輸出
- 源碼
- learning.py
- Python基礎
- 數據類型和變量
- 字符串和編碼
- 使用list和tuple
- 條件判斷
- 循環
- 使用dict和set
- 函數
- 調用函數
- 定義函數
- 函數的參數
- 遞歸函數
- 高級特性
- 切片
- 迭代
- 列表生成式
- 生成器
- 迭代器
- 函數式編程
- 高階函數
- map/reduce
- filter
- sorted
- 返回函數
- 匿名函數
- 裝飾器
- 偏函數
- Python函數式編程——偏函數(來自博客)
- 模塊
- 使用模塊
- 安裝第三方模塊
- 面向對象編程
- 類和實例
- 訪問限制
- 繼承和多態
- 獲取對象信息
- 實例屬性和類屬性
- 面向對象高級編程
- 使用__slots__
- 使用@property
- 多重繼承
- 定制類
- 使用枚舉類
- 使用元類
- 錯誤、調試和測試
- 錯誤處理
- 調試
- 單元測試
- 文檔測試
- IO編程
- 文件讀寫
- StringIO和BytesIO
- 操作文件和目錄
- 序列化
- 進程和線程
- 多進程
- 多線程
- ThreadLocal
- 進程 vs. 線程
- 分布式進程
- 正則表達式
- 常用內建模塊
- datetime
- collections
- base64
- struct
- hashlib
- itertools
- contextlib
- XML
- HTMLParser
- urllib
- 常用第三方模塊
- PIL
- virtualenv
- 圖形界面
- 網絡編程
- TCP/IP簡介
- TCP編程
- UDP編程
- 電子郵件
- SMTP發送郵件
- POP3收取郵件
- 訪問數據庫
- 使用SQLite
- 使用MySQL
- 使用SQLAlchemy
- Web開發
- HTTP協議簡介
- HTML簡介
- WSGI接口
- 使用Web框架
- 使用模板
- 異步IO
- 協程
- asyncio
- async/await
- aiohttp
- 實戰
- Day 1 - 搭建開發環境
- Day 2 - 編寫Web App骨架
- Day 3 - 編寫ORM
- Day 4 - 編寫Model
- Day 5 - 編寫Web框架
- Day 6 - 編寫配置文件
- Day 7 - 編寫MVC
- Day 8 - 構建前端
- Day 9 - 編寫API
- Day 10 - 用戶注冊和登錄
- Day 11 - 編寫日志創建頁
- Day 12 - 編寫日志列表頁
- Day 13 - 提升開發效率
- Day 14 - 完成Web App
- Day 15 - 部署Web App
- Day 16 - 編寫移動App
- FAQ
- 期末總結