我正在尝试将 JSON 加载回对象。 “loads”方法似乎没有错误,但对象似乎没有我期望的属性。
我怎样才能检查/检查我拥有的对象(这是基于网络的代码)。
results = {"Subscriber": {"firstname": "Neal", "lastname": "Walters"}}
subscriber = json.loads(results)
for item in inspect.getmembers(subscriber):
self.response.out.write("<BR>Item")
for subitem in item:
self.response.out.write("<BR> SubItem=" + subitem)
上面的尝试返回了这个:
Item
SubItem=__class__
我认为这不重要,但对于上下文: JSON 实际上来自 Google App Engine 中的 urlfetch 到 使用此实用程序创建的休息 Web 服务: http://code.google.com/p/appengine-rest-server . 正在使用以下定义从数据存储中检索数据:
class Subscriber(db.Model):
firstname = db.StringProperty()
lastname = db.StringProperty()
谢谢, 尼尔
更新 #1:基本上我正在尝试将 JSON 反序列化回一个对象。 理论上它是从一个对象序列化的,我现在想把它放回一个对象中。 也许更好的问题是如何做到这一点?
更新 #2:我试图将一个复杂的程序抽象为几行代码,因此我在“伪编码”中犯了一些错误,以便在此处发布。
这是一个更好的代码示例,现在从我可以在 PC 上运行的网站中取出。
results = '{"Subscriber": {"firstname": "Neal", "lastname": "Walters"}}'
subscriber = json.loads(results)
for key, value in subscriber.items():
print " %s: %s" %(key, value)
上面的运行,它显示的内容看起来并不比 JSON 字符串本身更有结构。它显示这个: 订户:{u'lastname':u'Walters',u'firstname':u'Neal'}
我有更多的 Microsoft 背景,所以当我听到序列化/反序列化时,我想从一个对象到一个字符串,然后从一个字符串回到一个对象。因此,如果我序列化为 JSON,然后反序列化,我会得到什么,字典、列表还是对象?实际上,我是从 REST 网络方法获取 JSON,这是代表我为我序列化我的对象。
理想情况下,我想要一个与上面的订阅者类匹配的订阅者对象,并且理想情况下,我不想编写一次性自定义代码(即特定于“订阅者”的代码),因为我想这样做其他几十个类(class)也是如此。如果我必须编写一些自定义代码,我将需要通用地执行它,以便它适用于任何类。
更新 #3:这是为了更多地解释为什么我认为这是一个必需的工具。我正在编写一个巨大的应用程序,可能是在 Google App Engine (GAE) 上。我们倾向于 REST 架构有几个原因,但其中一个原因是我们的 Web GUI 应该通过 REST Web 层访问数据存储。 (我更习惯于 SOAP,因此切换到 REST 本身就是一个小挑战)。因此,获取和更新数据的经典方法之一是通过业务或数据层。通过使用上面提到的 REST 实用程序,我可以选择 XML 或 JSON。我希望在我们开发大型应用程序之前为两者做一个小型工作原型(prototype))。然后,假设我们有一个成功的应用程序,并且 GAE 将其价格翻倍。然后我们可以只重写数据层,并采用我们的 Python/Django 用户层(网络代码),并在亚马逊或其他地方运行它。
如果我要做所有这些,为什么我希望所有内容都是字典对象。我不想要成熟的类(class)结构的力量吗?接下来的技巧之一是某种对象关系映射 (ORM),这样我们就不必公开我们的确切数据表,而是更多地公开一个逻辑层。
我们还想向可能使用任何语言的付费用户公开一个 RESTful API。对于他们来说,他们可以使用 XML 或 JSON,他们不会使用这里讨论的序列化例程。
最佳答案
json 仅对字符串、 float 、整数、javascript 对象(python 字典)和列表进行编码。
您必须创建一个函数来将返回的字典转换为一个类,然后使用 object_hook
关键字参数和 json 字符串将其传递给 json.loads
.下面是一些充实它的代码:
import json
class Subscriber(object):
firstname = None
lastname = None
class Post(object):
author = None
title = None
def decode_from_dict(cls,vals):
obj = cls()
for key, val in vals.items():
setattr(obj, key, val)
return obj
SERIALIZABLE_CLASSES = {'Subscriber': Subscriber,
'Post': Post}
def decode_object(d):
for field in d:
if field in SERIALIZABLE_CLASSES:
cls = SERIALIZABLE_CLASSES[field]
return decode_from_dict(cls, d[field])
return d
results = '''[{"Subscriber": {"firstname": "Neal", "lastname": "Walters"}},
{"Post": {"author": {"Subscriber": {"firstname": "Neal",
"lastname": "Walters"}}},
"title": "Decoding JSON Objects"}]'''
result = json.loads(results, object_hook=decode_object)
print result
print result[1].author
这将处理任何可以在没有构造函数参数的情况下实例化并且 setattr
将起作用的类。
此外,这使用 json
。我没有使用 simplejson
的经验,所以 YMMV 但我听说它们是相同的。
请注意,虽然两个订阅者对象的值相同,但生成的对象却不同。这可以通过内存 decode_from_dict
类来解决。
关于python - 如何在 Python 中检查神秘的反序列化对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3706208/