ran*_*ser 7 python google-app-engine app-engine-ndb
假设我最初创建一个ndb.Model并想要更改字段的ndb属性类型(例如IntegerProperty到StringProperty),但是想要转换存储在该字段中的当前数据,以便我不会丢失该数据.一种方法是简单地创建一个新的字段名称,然后使用脚本迁移数据,但还有其他更方便的方法来完成此操作吗?
例如,假设我有以下模型:
class Car(ndb.Model):
name = ndb.StringProperty()
production_year = ndb.IntegerProperty()
Run Code Online (Sandbox Code Playgroud)
我存储了一个实体的实例:
c = new Car()
c.name = "Porsche"
c.production_year = 2013
Run Code Online (Sandbox Code Playgroud)
并希望将production_year更改为ndb.StringProperty()而不会"丢失"我设置的值(它仍然存在,但不可检索).如果我只是将production_year更改为ndb.StringProperty()的实例,则字段值不报告有意义的值,因为类型不匹配.
所以,如果我将模型更改为:
class Car(ndb.Model):
name = ndb.StringProperty()
production_year = ndb.StringProperty()
Run Code Online (Sandbox Code Playgroud)
尝试使用点表示法检索字段将导致值为None.任何人遇到这种情况,你能解释一下你做了什么来解决它吗?谢谢.
Tim*_*man 10
你如何处理这将取决于你有多少实体.如果您在10000中使用相对较少数量的实体,我会使用remote_api并从数据存储中检索原始基础数据并直接操作数据然后将其写回,而不是使用模型.例如,这将获取原始实体,并且可以像字典一样访问属性.此代码几乎从较低级别的appengine SDK代码中解除.
from google.appengine.api import datastore
from google.appengine.api import datastore_errors
def get_entities(keys):
rpc = datastore.GetRpcFromKwargs({})
keys, multiple = datastore.NormalizeAndTypeCheckKeys(keys)
entities = None
try:
entities = datastore.Get(keys, rpc=rpc)
except datastore_errors.EntityNotFoundError:
assert not multiple
return entities
def put_entities(entities):
rpc = datastore.GetRpcFromKwargs({})
keys = datastore.Put(entities, rpc=rpc)
return keys
Run Code Online (Sandbox Code Playgroud)
你可以使用如下(我使用fetch来简化这个例子的代码)
x = Car.query(keys_only=True).fetch(100)
results = get_entities([i.to_old_key() for i in x])
for i in results:
i['production_year'] = unicode(i['production_year'])
put_entities(results)
Run Code Online (Sandbox Code Playgroud)
这是我的旧代码并datastore.NormalizeAndTypeCheckKeys采用旧的db样式键,我没有看到有ndb样式键的等效函数,但这确实有效.(刚试过它;-)
此方法允许您在不部署任何新代码的情况下迁移数据.
如果您有数百万个实体,那么您应该查看其他处理方法,即使用此代码并使用mapreduce.
| 归档时间: |
|
| 查看次数: |
1515 次 |
| 最近记录: |