利用动态属性处理JSON数据源
属性:在Python中,数据的属性和处理数据的方法统称为属性。
元编程:用元类编程,元类→类→对象,元类比类更抽象,生成类的类。
1、使用动态属性访问JSON类数据
第一版:利用json.load(fp)审查数据
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) #如果需要下载就发出提醒。
with urlopen(URL) as remote, open(JSON, 'wb') as local: #在with语句中使用两个上下文管理器分别用于读取和保存远程文件。
local.write(remote.read())
with open(JSON) as fp:
return json.load(fp)#json.load函数解析JSON文件,返回Python原生对象。
第二版:使用动态属性访问JSON类数据
第一版查看深度数据的格式比较冗长,比如feed'Schedule'40,我们希望使用feed.Schedule.events[40].name来提高阅读性能。该类的第二个版本是递归的,自动处理嵌套的映射和列表。
from collections import abc
class FronenJSON():
def __init__(self,mapping):
self.__data=dict(mapping)#创建副本,同时确保处理的是字典。
def __getattr__(self, name):#仅当没有指定名称的属性才调用__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)#创建FrozenJSON对象。
elif isinstance(obj,abc.MutableSequence):
return [cls.build(item) for item in obj]#递归调用.build()方法,构建一个列表。
else:#既不是字典也不是列表,则返回元素本身。
return obj
分析:FronenJSON类的关键是__getattr__方法。只有在无法通过正常方式获取属性(即在实例、类或超类中找不到指定的属性)时,解释器才会调用特殊的 __getattr__ 方法。
2、处理无效属性名
在 Python 中,以关键字命名的属性是无效的,因为关键字是保留的。因此需要在第二版中改进 __init__ :
def __init__(self,mapping):
self.__data={}
for key,value in mapping.items():
if keyword.iskeyword(key):
key+='_'#与Python关键字重复的key在尾部加上下划线。
self.__data[key]=value
3、使用特殊方法__new__
版本 3:使用 __new__ 构造函数将类转换为灵活的对象工厂函数。
from collections import abc
class FronenJSON():
def __new__(cls, arg): # __new__是类方法,第一个参数是类本身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如何处理excel表格10/10
- ♥ python防止栈溢出的解决方法12/12
- ♥ python中如何给变量赋值09/26
- ♥ python现在流行吗?11/11
- ♥ \t在python中代表什么08/18
- ♥ python中的return是什么意思?08/26
内容反馈