欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

pydantic進階用法示例詳解

 更新時間:2023年03月15日 14:24:24   作者:it_miclon  
這篇文章主要為大家介紹了pydantic進階用法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

正文

pydantic是一個Python的數(shù)據(jù)驗證和轉(zhuǎn)換庫,它的特點是輕量、快速、可擴展、可配置。筆者常用的用于數(shù)據(jù)接口schema定義與檢查。

具體的基本用法本文不再做過多的介紹,可以參考pydantic官方文檔。本文主要是結(jié)合實際項目開發(fā)中遇到的問題和解題思路,介紹一些pydantic的高階玩法。

當前現(xiàn)狀

在項目中,pydantic的定義是在數(shù)據(jù)的出口進行規(guī)范化,從而使得下游接受方能更快地去解析和清洗這些數(shù)據(jù)。

from pydantic import BaseModel, Field

# 定義數(shù)據(jù)模型
class Project(BaseModel):
    url: str = Field(...)
    title: str = Field(...)
    content: str = Field(...)
    company: List[Dict] = Field(default=[])
    industry: str = Field(...)

以上是簡單的一個數(shù)據(jù)模型定義,代碼僅為示例,隱去了一些字段和配置。也就是我們必須傳輸給Project模型對應(yīng)的數(shù)據(jù)才可以通過它的數(shù)據(jù)校驗,否則就無法繼續(xù)向下(可能是發(fā)往下游)

這么做一直以來沒什么問題,直到本次項目中的接口返回出現(xiàn)了大更新,使得之前的所有代碼層做的數(shù)據(jù)字段映射必須重新對應(yīng)匹配。

比如之前title字段對應(yīng)的是title,現(xiàn)在變成了detail-article-title。

這使得我們必須在代碼層做諸如:

# project_data均為接口返回的數(shù)據(jù),加數(shù)據(jù)演示

# 之前的代碼
project_data = {
    "url": "https://www.baidu.com",
    "title": "百度一下,你就知道",
}
project = Project(
    **project_data
)

# 現(xiàn)在的代碼
project_data = {
  "detail": {
      "url": "xxx"
      "article": {
          "title": "項目標題",
      }
  }
}
project = Project(
    url=project_data["detail"]["url"],
    title=project_data["detail"]["article"]["title"],
)

以上代碼取值變得復(fù)雜,這還沒考慮到數(shù)據(jù)可能存在出錯的問題,比如detail字段不存在,這樣就會導致KeyError異常。

而且這并不是夸張的舉例(因為事實情況更復(fù)雜)。

我怎么能容忍這種情況呢?

解決方案

我當然不是想摒棄掉pydantic,而是想找到一種結(jié)合它更優(yōu)雅的方式來解決這個問題。

于是我第一時間想到了jmespath模塊,因為它是一個JSON查詢語言,可以用來在JSON數(shù)據(jù)中查找和提取數(shù)據(jù)。

from jmespath import search
project_data = {
    "detail": {
        "article": {
            "title": "項目標題",
        }
    }
}

title = search("detail.article.title", project_data)
assert title == "項目標題"  # True

# 即使是path不存在,也不會異常,而是返回None
assert search("detail.article.title1", project_data) is None  # True

所以我打算做一個結(jié)合pydanticjmespath的方式來解決這個問題。

class Project(BaseModel):
    url: str = Field(...)
    title: str = Field(...)
    content: str = Field(...)
    company: List[Dict] = Field(default=[])
    industry: str = Field(...)

    @root_validator(pre=True, skip_on_failure=True)
    def data_converter(cls, v):
        return {
            "url": search("detail.id", v),
            "title": search("detail.article.title", v),
            "content": search("detail.article.content", v),
            "company": search("company[*].name", v),
            "industry": search("industry", v)
        }
    
    @validator("url")
    def url_validator(cls, v):
        # 由于這里的v是拿到的ID,需要組合成url
        return f"https://xxxxx/{v}"

從代碼中可以知道,我是在root_validator中提前做了數(shù)據(jù)的轉(zhuǎn)換,將jmespath的查詢結(jié)果賦值給對應(yīng)的字段。

但是做完之后我越看越變扭,我為了做這個事情,先要申明所有字段,還要對這些字段一一映射。

于是,我想到了pydanticConfig類,它可以用來配置pydantic的一些行為。而且通過查看源碼,我認為我可以通過Field類中輸入一個path變量,告訴未來的處理器,這個path是用來做數(shù)據(jù)提取的。

class Project(BaseModel):
    url: str = Field(..., path="temporaryLibrary.id")
    company_names: str = Field(..., path="company[0].enterprise.name")
    versions: List[str] = Field(..., path="versionList[*].id")

當然現(xiàn)在代碼是沒有任何意義的,因為path是我們自定義的,pydantic并不知道如何處理它。

所以下一步我們要做的是,如何更好的讓pydantic知道如何處理path。

在多次翻閱它源代碼,并結(jié)合官方文檔中對Model類的介紹,我找到了一個可行的方案。

Pydantic models can be created from arbitrary class instances to support models that map to ORM objects.

也就是說,我可以將原始數(shù)據(jù)通過from_orm傳遞給pydantic的模型,然后通過Data binding的方式,將數(shù)據(jù)綁定到模型中。Data binding允許我們自定義數(shù)據(jù)的取值來源。

class ProjectGetter(GetterDict):

    def get(self, key: str, default: Any) -> Any:  # noqa
        # 由于getter_dict所能拿到的“數(shù)據(jù)權(quán)限”相對較低,
        # 也就是它的權(quán)限僅僅是處理數(shù)據(jù),而不是處理模型,
        # 所以我們需要自己去拿到模型,然后再去拿到path
        model, data = self._obj['model'], self._obj['data']
        for name, field in model.__fields__.items():
            path = field.field_info.extra.get('path')
            if path and name == key:
                return search(path, data)
        return default


class Project(BaseModel):
    url: str = Field(..., path="detail.id")
    company_names: str = Field(..., path="company[0].enterprise.name")
    versions: List[str] = Field(..., path="versionList[*].id")

    @validator("url")
    def url_validator(cls, v):
        return f"https://www.baidu.com/{v}"

    class Config:
        # 通過orm_mode指定數(shù)據(jù)的來源
        orm_mode = True
        # 通過getter_dict指定數(shù)據(jù)的獲取方式
        getter_dict = ProjectGetter

project_data = {
    "detail": {
        "id": 1,
        "article": {
            "title": "項目標題",
        }
    },
    "company": [
      {
        "enterprise": {
          "name": "企業(yè)名稱1"
        }
      },
      {
        "enterprise": {
          "name": "企業(yè)名稱2"
        }
      }
    ],
    "versionList": [{"id": "1.0"}, {"id": "2.0"}]
}
project = Project.from_orm({"model": Project, "data": project_data})
print(project)
# url='https://www.baidu.com/1' company_names='企業(yè)名稱1' versions=['1.0', '2.0']

這樣我們在業(yè)務(wù)端,只需要對Field指定其對應(yīng)數(shù)據(jù)提取的path,而不需要再去寫一堆的validator或者是在數(shù)據(jù)進入前做一堆的數(shù)據(jù)轉(zhuǎn)換。

總結(jié)

通過這個小例子,我們可以看到,pydantic的靈活性是非常強的,它可以通過Config類來配置一些行為,而且它的Field類也可以通過extra參數(shù)來傳遞一些額外的信息。這大大的提高了我們的對數(shù)據(jù)的處理能力。筆者也會在后續(xù)的文章中,繼續(xù)分享pydantic的一些使用技巧。

以上就是pydantic進階用法示例詳解的詳細內(nèi)容,更多關(guān)于pydantic進階用法的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Python3創(chuàng)建Django項目的幾種方法(3種)

    Python3創(chuàng)建Django項目的幾種方法(3種)

    這篇文章主要介紹了Python3創(chuàng)建Django項目的幾種方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-06-06
  • Python實現(xiàn)語音識別和語音合成功能

    Python實現(xiàn)語音識別和語音合成功能

    聲音的本質(zhì)是震動,震動的本質(zhì)是位移關(guān)于時間的函數(shù),波形文件(.wav)中記錄了不同采樣時刻的位移。這篇文章主要介紹了Python實現(xiàn)語音識別和語音合成,需要的朋友可以參考下
    2019-09-09
  • python爬蟲實戰(zhàn)之爬取京東商城實例教程

    python爬蟲實戰(zhàn)之爬取京東商城實例教程

    這篇文章主要介紹了python爬取京東商城的相關(guān)資料,文中通過爬取一個實例頁面進行了講解,通過示例代碼和圖文介紹的非常詳細,相信對大家具有一定的參考價值,需要的朋友們下面來一起學習學習吧。
    2017-04-04
  • Python Print實現(xiàn)在輸出中插入變量的例子

    Python Print實現(xiàn)在輸出中插入變量的例子

    今天小編就為大家分享一篇Python Print實現(xiàn)在輸出中插入變量的例子,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-12-12
  • python3+PyQt5 自定義窗口部件--使用窗口部件樣式表的方法

    python3+PyQt5 自定義窗口部件--使用窗口部件樣式表的方法

    今天小編就為大家分享一篇python3+PyQt5 自定義窗口部件--使用窗口部件樣式表的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-06-06
  • python實現(xiàn)簡單爬蟲功能的示例

    python實現(xiàn)簡單爬蟲功能的示例

    本文主要是介紹python實現(xiàn)簡單爬蟲功能的示例,主要實現(xiàn)了把我們想要的圖片爬蟲到本地的一個示例,有需要的朋友可以了解一下。
    2016-10-10
  • python 轉(zhuǎn)換 Javascript %u 字符串為python unicode的代碼

    python 轉(zhuǎn)換 Javascript %u 字符串為python unicode的代碼

    這篇文章主要介紹了python 轉(zhuǎn)換 Javascript %u 字符串為python unicode的代碼,需要的朋友可以參考下
    2016-09-09
  • Python判斷空的五種方法

    Python判斷空的五種方法

    本文介紹了多種在Python中判斷變量是否為空的方法,包括if語句、len()函數(shù)、not關(guān)鍵字、bool()函數(shù)和try...except語句等,具有一檔的參考價值,感興趣的可以了解一下
    2024-12-12
  • 如何優(yōu)雅地改進Django中的模板碎片緩存詳解

    如何優(yōu)雅地改進Django中的模板碎片緩存詳解

    這篇文章主要給大家介紹了關(guān)于如何優(yōu)雅地改進Django中的模板碎片緩存的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2018-07-07
  • 用django-allauth實現(xiàn)第三方登錄的示例代碼

    用django-allauth實現(xiàn)第三方登錄的示例代碼

    這篇文章主要介紹了用django-allauth實現(xiàn)第三方登錄的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2019-06-06

最新評論