溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Python利用動態(tài)屬性處理JSON數(shù)據(jù)源的方法

發(fā)布時(shí)間:2020-08-07 13:44:40 來源:億速云 閱讀:193 作者:小新 欄目:編程語言

小編給大家分享一下Python利用動態(tài)屬性處理JSON數(shù)據(jù)源的方法,希望大家閱讀完這篇文章后大所收獲,下面讓我們一起去探討吧!

利用動態(tài)屬性處理JSON數(shù)據(jù)源

屬性:在Python中,數(shù)據(jù)的屬性和處理數(shù)據(jù)的方法統(tǒng)稱屬性。

元編程:用元類進(jìn)行編程,元類→類→對象,元類比類更抽象,生成類的類。

1、使用動態(tài)屬性訪問JSON類數(shù)據(jù)

第一版:利用json.load(fp)審查數(shù)據(jù)

from urllib.request import urlopen
import warnings
import os
import json
 
URL = 'http://www.oreilly.com/pub/sc/osconfeed'
JSON = 'data/osconfeed.json'
 
def load():
  if not os.path.exists(JSON):
    msg = 'downloading {} to {}'.format(URL, JSON)
    warnings.warn(msg) #如果需要下載就發(fā)出提醒。
    with urlopen(URL) as remote, open(JSON, 'wb') as local: #在with語句中使用兩個(gè)上下文管理器分別用于讀取和保存遠(yuǎn)程文件。
      local.write(remote.read())
  with open(JSON) as fp:
    return json.load(fp)#json.load函數(shù)解析JSON文件,返回Python原生對象。

第二版:使用動態(tài)屬性訪問JSON類數(shù)據(jù)

第一版查閱深層數(shù)據(jù)的格式比較冗長,例如feed'Schedule'40,我們希望在讀取屬性上采用feed.Schedule.events[40].name這類方式來改進(jìn)。并且第二版的類能遞歸,自動處理嵌套的映射和列表。

from collections import abc
 
class FronenJSON():
  def __init__(self,mapping):
    self.__data=dict(mapping)#創(chuàng)建副本,同時(shí)確保處理的是字典。
     
  def __getattr__(self, name):#僅當(dāng)沒有指定名稱的屬性才調(diào)用__getattr__方法。
    if hasattr(self,name):
      return getattr(self.__data,name)
    else:
      return FronenJSON.build(self.__data[name])
   
  @classmethod 
  def __build__(cls,obj):
    if isinstance(obj,abc.Mapping):#判斷obj是否是映射。
      return cls(obj)#創(chuàng)建FrozenJSON對象。
    elif isinstance(obj,abc.MutableSequence):
      return [cls.build(item) for item in obj]#遞歸調(diào)用.build()方法,構(gòu)建一個(gè)列表。
    else:#既不是字典也不是列表,則返回元素本身。
      return obj

分析: FronenJSON類的關(guān)鍵是__getattr__方法。僅當(dāng)無法使用常規(guī)的方式獲取屬性(即在實(shí)例、類或超類中找不到指定的屬性),解釋器才會調(diào)用特殊的__getattr__方法。

2、處理無效屬性名

在Python中,由于關(guān)鍵字被保留,名稱為關(guān)鍵字的屬性是無效的。因此需要對第二版中的__init__進(jìn)行改進(jìn):

def __init__(self,mapping):
  self.__data={}
  for key,value in mapping.items():
    if keyword.iskeyword(key):
      key+='_'#與Python關(guān)鍵字重復(fù)的key在尾部加上下劃線。
    self.__data[key]=value

3、使用特殊方法__new__

第三版:使用__new__構(gòu)造方法把一個(gè)類轉(zhuǎn)換成一個(gè)靈活的對象工廠函數(shù)。

from collections import abc
 
class FronenJSON():
  def __new__(cls, arg): # __new__是類方法,第一個(gè)參數(shù)是類本身cls。
    if isinstance(arg, abc.Mapping):
      return super().__new__(cls) #委托給超類object基類的__new__方法處理。
    elif isinstance(arg, abc.MutableSequence): # 余下方法與原先的build方法一致。
      return [cls(item) for item in arg]
    else:
      return arg
  
   def __init__(self,mapping):
    self.__data={}
    for key,value in mapping.items():
      if keyword.iskeyword(key):
        key+='_'
      self.__data[key]=value 
 
  def __getattr__(self, name):
    if hasattr(self,name):
      return getattr(self.__data,name)
    else:
      return FronenJSON(self.__data[name])

看完了這篇文章,相信你對Python利用動態(tài)屬性處理JSON數(shù)據(jù)源的方法有了一定的了解,想了解更多相關(guān)知識,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI