我有这样的结构:有章节的书籍(祖先=书)有页面(祖先=章节)
我很清楚,要通过 ID 搜索章节,我需要通过祖先查询来搜索书籍。今天我了解到,如果我拥有所有密钥,我可以直接检索实体,而无需先获取书籍,然后获取章节,然后获取页面,如下所示:
page_key = ndb.Key('Book', long(bookId), 'Chapter', long(chapterId), 'Page', long(pageId))
page = page_key.get()
我的疑问是:要通过例如页码获取页面,我必须首先获取章节吗?
例如 :
class Book(ndb.Model):
name = ndb.StringProperty(required=True)
# Search by id
@classmethod
def by_id(cls, id):
return Book.get_by_id(long(id))
class Chapter(ndb.Model):
name = ndb.StringProperty(required=True)
# Search by id
@classmethod
def by_id(cls, id, book):
return Chapter.get_by_id(long(id), parent=book.key)
class Page(ndb.Model):
pageNumber = ndb.IntegerProperty(default=1)
content = ndb.StringProperty(required=True)
# Search by page
@classmethod
def by_page(cls, number, chapter):
page = Page.query(ancestor=chapter.key)
page = page.filter(Page.pageNumber == int(number))
return page.get()
实际上,当我需要搜索页面以显示其内容时,我正在这样做:
getPage?bookId=5901353784180736&chapterId=5655612935372800&page=2
所以,在控制器中,我这样做:
def get(self):
# Get the id parameters
bookId = self.request.get('bookId')
chapterId = self.request.get('chapterId')
pageNumber = self.request.get('page')
if bookId and chapterId and pageNumber:
# Must be a digit
if bookId.isdigit() and chapterId.isdigit() and pageNumber.isdigit():
# Get the chapter
chapter_key = ndb.Key('Book', long(bookId), 'Chapter', long(chapterId))
chapter = chapter_key.get()
if chapter:
# Get the page
page = Page.by_number(pageNumber, chapter)
这是正确的方法还是我缺少更好的方法,我只能访问数据存储,而不是两次?
如果这是正确的,我认为使用 NDB 的这种工作方式不会对数据存储产生任何影响,因为对此页面的重复调用总是会命中同一章节和页面的 NDB 缓存(因为我正在使用get()方法,它不是一个fetch()命令)。我的猜测对吗?