日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不

當前位置:首頁 > 科技  > 軟件

再探泛型 API,感受 Python 對象的設計哲學

來源: 責編: 時間:2024-05-27 08:53:35 161觀看
導讀之前我們提到了泛型 API,這類 API 的特點是可以處理任意類型的對象,舉個例子。// 返回對象的長度PyObject_Size// 返回對象的某個屬性PyObject_GetAttr// 返回對象的哈希值PyObject_Hash// 將對象轉成字符串返回PyObjec

之前我們提到了泛型 API,這類 API 的特點是可以處理任意類型的對象,舉個例子。eKU28資訊網——每日最新資訊28at.com

// 返回對象的長度PyObject_Size// 返回對象的某個屬性PyObject_GetAttr// 返回對象的哈希值PyObject_Hash// 將對象轉成字符串返回PyObject_Str

對應到 Python 代碼中,就是下面這個樣子。eKU28資訊網——每日最新資訊28at.com

# PyObject_Sizeprint(len("古明地覺"))print(len([1, 2, 3]))"""43"""# PyObject_GetAttrprint(getattr("古明地覺", "lower"))print(getattr([1, 2, 3], "append"))print(getattr({}, "update"))"""<built-in method lower of str object at 0x7f081aa7e920><built-in method append of list object at 0x7f081adc1100><built-in method update of dict object at 0x7f081aa8fd80>"""# PyObject_Hashprint(hash("古明地覺"))print(hash(2.71))print(hash(123))"""81525063933782332031637148536541722626123"""# PyObject_Strprint(str("古明地覺"))print(str(object()))"""古明地覺<object object at 0x7fdfa0209d10>"""

這些 API 能處理任意類型的對象,這究竟是怎么辦到的?要想搞清楚這一點,還是要從 PyObject 入手。eKU28資訊網——每日最新資訊28at.com

我們知道對象在 C 看來就是一個結構體實例,并且結構體嵌套了 PyObject。eKU28資訊網——每日最新資訊28at.com

# 創建一個列表,讓變量 var 指向它var = [1, 2, 3]# 創建一個浮點數,讓變量 var 指向它var = 2.71

列表對應的結構體是 PyListObject,浮點數對應的結構體是 PyFloatObject,變量 var 是指向對象的指針。那么問題來了,憑啥一個變量可以指向不同類型的對象呢?或者說變量和容器里面為什么可以保存不同對象的指針呢?eKU28資訊網——每日最新資訊28at.com

原因在前面的文章中解釋的很詳細了,因為對象的指針會統一轉成 PyObject * 之后再交給變量保存,以創建列表為例。eKU28資訊網——每日最新資訊28at.com

圖片圖片eKU28資訊網——每日最新資訊28at.com

當然創建浮點數也是同理,因此變量和容器里的元素本質上就是一個泛型指針 PyObject *。而對象的指針在交給變量保存的時候,也都會先轉成 PyObject *,因為不管什么對象,它底層的結構體都嵌套了 PyObject。正是因為這個設計,變量才能指向任意的對象。eKU28資訊網——每日最新資訊28at.com

圖片圖片eKU28資訊網——每日最新資訊28at.com

所以 Python 變量相當于一個便利貼,可以貼在任意對象上。eKU28資訊網——每日最新資訊28at.com

不過問題來了,由于對象的指針會統一轉成 PyObject * 之后再交給變量保存,那么變量怎么知道自己指向的是哪種類型的對象呢?相信你肯定知道答案:通過 ob_type 字段。eKU28資訊網——每日最新資訊28at.com

圖片圖片eKU28資訊網——每日最新資訊28at.com

對象對應的結構體可以有很多個字段,比如 PyListObject,但變量能看到的只有前兩個字段。至于之后的字段是什么,則取決于對象的類型,總之對變量來說是不可見的,因為它是 PyObject *。eKU28資訊網——每日最新資訊28at.com

所以變量會先通過 ob_type 字段獲取對象的類型,如果 ob_type 字段的值為 &PyList_Type,那么變量指向的就是 PyListObject。如果 ob_type 字段的值為 &PyFloat_Type,那么變量指向的就是 PyFloatObject,其它類型同理。eKU28資訊網——每日最新資訊28at.com

當得到了對象的類型,那么再轉成相應的指針即可,假設 ob_type 是 &PyList_Type,那么變量會再轉成 PyListObject *,這樣就可以操作列表的其它字段了。eKU28資訊網——每日最新資訊28at.com

所以我們再總結一下:eKU28資訊網——每日最新資訊28at.com

圖片圖片eKU28資訊網——每日最新資訊28at.com

變量和容器里的元素只能保存相同的指針類型,而不同類型的對象,其底層的結構體是不同的。但這些結構體無一例外都嵌套了 PyObject,因此它們的指針會統一轉成 PyObject * 之后再交給變量保存。eKU28資訊網——每日最新資訊28at.com

然后變量在操作對象時,會先通過 ob_type 判斷對象的類型,假如是 &PyList_Type,那么會再轉成 PyListObject *,其它類型同理。eKU28資訊網——每日最新資訊28at.com

圖片圖片eKU28資訊網——每日最新資訊28at.com

相信你已經知道為什么泛型 API 可以處理任意類型的對象了,以 PyObject_GetAttr 為例,它內部會調用類型對象的 tp_getattro。eKU28資訊網——每日最新資訊28at.com

// 等價于 getattr(v, name)PyObject *PyObject_GetAttr(PyObject *v, PyObject *name){       // 獲取對象 v 的類型對象    PyTypeObject *tp = Py_TYPE(v);    PyObject* result = NULL;    // 如果類型對象實現了 tp_getattro,那么進行調用    // 等價于 Python 中的 type(v).__getattr__(v, name)    if (tp->tp_getattro != NULL) {        result = (*tp->tp_getattro)(v, name);    }    // 否則會退化為 tp_getattr,它要求屬性名稱必須是 C 字符串    // 不過 tp_getattr 已經廢棄,應該使用 tp_getattro    else if (tp->tp_getattr != NULL) {        const char *name_str = PyUnicode_AsUTF8(name);        if (name_str == NULL) {            return NULL;        }        result = (*tp->tp_getattr)(v, (char *)name_str);    }    // 否則說明對象 v 沒有指定屬性    else {        PyErr_Format(PyExc_AttributeError,                    "'%.100s' object has no attribute '%U'",                    tp->tp_name, name);    }    return result;}

函數先通過 ob_type 找到對象的類型,然后通過類型對象的 tp_getattro 調用對應的屬性查找函數。所以 PyObject_GetAttr 會根據對象的類型,調用不同的屬性查找函數,因此這就是泛型 API 能處理任意對象的秘密。eKU28資訊網——每日最新資訊28at.com

我們再以 Python 為例:eKU28資訊網——每日最新資訊28at.com

class A:    def __getattr__(self, item):        return f"class:A,item:{item}"class B:    def __getattr__(self, item):        return f"class:B,item:{item}"a = A()b = B()print(getattr(a, "some_attr"))print(getattr(b, "some_attr"))"""class:A,item:some_attrclass:B,item:some_attr"""# 以上等價于print(type(a).__getattr__(a, "some_attr"))print(type(b).__getattr__(b, "some_attr"))"""class:A,item:some_attrclass:B,item:some_attr"""

在 Python 里的表現和源碼是一致的,我們再舉個 iter 的例子:eKU28資訊網——每日最新資訊28at.com

data = [1, 2, 3]print(iter(data))print(type(data).__iter__(data))"""<list_iterator object at 0x7fb8200f29a0><list_iterator object at 0x7fb8200f29a0>"""

如果一個對象支持迭代器操作,那么它的類型對象一定實現了 __iter__,通過 type(data) 可以獲取到類型對象,然后將 data 作為參數調用 __iter__ 即可。eKU28資訊網——每日最新資訊28at.com

所以通過 ob_type 字段,這些泛型 API 實現了類似多態的效果,一個函數,多種實現。eKU28資訊網——每日最新資訊28at.com

本文鏈接:http://www.www897cc.com/showinfo-26-90852-0.html再探泛型 API,感受 Python 對象的設計哲學

聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。郵件:2376512515@qq.com

上一篇: Vue3 新玩法!我能操控計算屬性 Computed!

下一篇: Python 類型注解與檢查:讓代碼“開口說話”的八個妙招

標簽:
  • 熱門焦點
  • 鴻蒙OS 4.0公測機型公布:甚至連nova6都支持

    華為全新的HarmonyOS 4.0操作系統將于今天下午正式登場,官方在發布會之前也已經正式給出了可升級的機型產品,這意味著這些機型會率先支持升級享用。這次的HarmonyOS 4.0支持
  • 28個SpringBoot項目中常用注解,日常開發、求職面試不再懵圈

    前言在使用SpringBoot開發中或者在求職面試中都會使用到很多注解或者問到注解相關的知識。本文主要對一些常用的注解進行了總結,同時也會舉出具體例子,供大家學習和參考。注解
  • 如何使用JavaScript創建一只圖像放大鏡?

    譯者 | 布加迪審校 | 重樓如果您曾經瀏覽過購物網站,可能遇到過圖像放大功能。它可以讓您放大圖像的特定區域,以便瀏覽。結合這個小小的重要功能可以大大改善您網站的用戶體驗
  • 使用LLM插件從命令行訪問Llama 2

    最近的一個大新聞是Meta AI推出了新的開源授權的大型語言模型Llama 2。這是一項非常重要的進展:Llama 2可免費用于研究和商業用途。(幾小時前,swyy發現它已從LLaMA 2更名為Lla
  • 騰訊蓋樓,字節拆墻

    來源 | 光子星球撰文 | 吳坤諺編輯 | 吳先之&ldquo;想重溫暴刷深淵、30+技能搭配暴搓到爽的游戲體驗嗎?一起上晶核,即刻暴打!&rdquo;曾憑借直播騰訊旗下代理格斗游戲《DNF》一
  • 新電商三兄弟,“抖快紅”成團!

    來源:價值研究所作 者:Hernanderz 隨著內容電商的概念興起,抖音、快手、小紅書組成的&ldquo;新電商三兄弟&rdquo;成為業內一股不可忽視的勢力,給阿里、京東、拼多多帶去了巨大壓
  • 年輕人的“職場羞恥感”,無處不在

    作者:馮曉亭 陶 淘 李 欣 張 琳 馬舒葉來源:燃次元&ldquo;人在職場,應該選擇什么樣的著裝?&rdquo;近日,在網絡上,一個與著裝相關的帖子引發關注,在該帖子里,一位在高級寫字樓亞洲金
  • 華為舉行春季智慧辦公新品發布會 首次推出電子墨水屏平板

    北京時間2月27日晚,華為在巴塞羅那舉行春季智慧辦公新品發布會,在海外市場推出之前已經在中國市場上市的筆記本、平板、激光打印機等辦公產品,并首次推出搭載
  • 上海舉辦人工智能大會活動,建設人工智能新高地

    人工智能大會在上海浦江兩岸隆重拉開帷幕,人工智能新技術、新產品、新應用、新理念集中亮相。8月30日晚,作為大會的特色活動之一的上海人工智能發展盛典人工
Top 主站蜘蛛池模板: 南郑县| 陆川县| 濉溪县| 收藏| 图们市| 云龙县| 鄢陵县| 马边| 沛县| 菏泽市| 达尔| 建瓯市| 焉耆| 乃东县| 扶沟县| 瓮安县| 买车| 万荣县| 盐亭县| 平陆县| 福鼎市| 江安县| 都昌县| 商水县| 漳平市| 邯郸市| 定边县| 孝昌县| 巴林左旗| 南开区| 镇雄县| 南陵县| 抚顺县| 水富县| 东安县| 新闻| 上高县| 永靖县| 盱眙县| 始兴县| 松原市|