4

原创设计

这是我最初的Models设置方式:

class UserData(db.Model):
    user = db.UserProperty()
    favorites = db.ListProperty(db.Key) # list of story keys
    # ...

class Story(db.Model):
    title = db.StringProperty()
    # ...

在显示故事的每个页面上,我都会查询当前用户的 UserData:

user_data = UserData.all().filter('user =' users.get_current_user()).get()
story_is_favorited = (story in user_data.favorites)

新设计

看完这个演讲后:Google I/O 2009 - Scalable, Complex Apps on App Engine,我想知道我是否可以更有效地进行设置。

class FavoriteIndex(db.Model):
    favorited_by = db.StringListProperty()

是一样的Story Model,但我摆脱了UserData Model. new 的每个实例FavoriteIndex Model都有一个Story作为父实例的实例。每个都在其属性FavoriteIndex中存储一个用户 ID 列表。favorited_by

如果我想查找某个用户收藏的所有故事:

index_keys = FavoriteIndex.all(keys_only=True).filter('favorited_by =', users.get_current_user().user_id())
story_keys = [k.parent() for k in index_keys]
stories = db.get(story_keys)

这种方法避免了与 ListProperty 关联的序列化/反序列化。

效率与简单

我不确定新设计的效率如何,尤其是在用户决定收藏 300 个故事之后,但这就是我喜欢它的原因:

  1. 收藏的故事与用户相关联,而不是与她相关联user data

  2. 在我显示 a 的页面上story,很容易询问story它是否已被收藏(无需调用填充用户数据的单独实体)。

    fav_index = FavoriteIndex.all().ancestor(story).get()
    fav_of_current_user = users.get_current_user().user_id() in fav_index.favorited_by
    
  3. 也很容易获得所有收藏故事的用户的列表(使用#2中的方法)

有没有更简单的方法?

请帮忙。这种事情通常是怎么做的?

4

4 回答 4

2

你所描述的是一个很好的解决方案。但是,您可以进一步优化它:对于每个收藏夹,创建一个“UserFavorite”实体作为相关故事条目的子实体(或等效地,作为 UserInfo 条目的子实体),并将键名设置为用户的唯一ID。这样,您可以通过简单的获取来确定用户是否收藏了某个故事:

UserFavorite.get_by_name(user_id, parent=a_story)

get 操作比查询快 3 到 5 倍,因此这是一个实质性的改进。

于 2009-10-13T20:04:12.340 回答
1

我不想解决您的实际问题,但这里有一个非常小的提示:您可以替换此代码:

if story in user_data.favorites:
    story_is_favorited = True
else:
    story_is_favorited = False

用这一行:

story_is_favorited = (story in user_data.favorites)

story in user_data.favorites如果你不想的话,你甚至不需要在括号周围加上括号;我只是认为这更具可读性。

于 2009-10-13T19:20:41.927 回答
1

您可以将最喜欢的索引设置为两个模型的连接

class FavoriteIndex(db.Model):
    user = db.UserProperty()
    story = db.ReferenceProperty()

或者

class FavoriteIndex(db.Model):
    user = db.UserProperty()
    story = db.StringListProperty()

然后,您对用户的查询为用户收藏的每个故事返回一个 FavoriteIndex 对象

您还可以按故事查询,看看有多少用户收藏了它。

您不想扫描任何东西,除非您知道它仅限于小尺寸

于 2009-10-13T20:08:28.290 回答
1

使用您的新设计,您可以查找用户是否通过查询收藏了某个故事。
您不需要 UserFavorite 类实体。
它是一个 keys_only 查询,因此不如 get(key) 快,但比普通查询快。
FavoriteIndex 类都具有相同的 key_name='favs'。
您可以根据 __key__ 进行过滤。

a_story = ......
a_user_id  = users.get_current_user().user_id()
favIndexKey = db.Key.from_path('Story', a_story.key.id_or_name(), 'FavoriteIndex', 'favs')
doesFavStory = FavoriteIndex.all(keys_only=True).filter('__key__ =', favIndexKey).filter('favorited_by =', a_user_id).get()

如果您使用多个 FavoriteIndex 作为 Story 的子项,则可以使用祖先过滤器

doesFavStory = FavoriteIndex.all(keys_only=True).ancestor(a_story).filter('favorited_by =', a_user_id).get()
于 2009-10-19T15:27:48.780 回答