我想要一个显示标题的页面以及用户朋友最近10篇帖子的博客链接。我还希望能够继续回溯旧条目。
所以在sql land中它会是这样的:
select * from blog_post where user_id in (select friend_id from user_friend where user_id = :userId) order by date
我不相信这些解决方案中的任何一个都会扩展。
我确定其他人已经遇到了这个问题,但我已经搜索过,观看谷歌io视频,阅读其他代码......我错过了什么?
答案 0 :(得分:13)
如果你看看你提供的SQL解决方案将如何执行,它将基本上像这样:
您可以在App Engine中自己执行完全相同的过程,方法是将Query实例用作迭代器并对它们进行合并连接。
你是对的,这对于大量的朋友来说不会很好地扩展,但是它会遇到与SQL实现完全相同的问题,它也不会伪装它们:获取最新的20个(例如)条目大致花费O(n log n)工作,其中n是朋友的数量。
答案 1 :(得分:7)
Google io talk中介绍了此主题: http://code.google.com/events/io/sessions/BuildingScalableComplexApps.html
Google团队建议使用列表属性以及他们称之为关系索引实体的内容,可以在此处找到示例应用程序:http://pubsub-test.appspot.com/
答案 2 :(得分:1)
“加载用户,遍历好友列表并加载他们最新的博客帖子。”
这就是所有连接 - 嵌套循环。某些类型的连接是带有查找的循环。大多数查找只是循环;有些是哈希。
“最后合并所有博客帖子以查找最新的10个博客条目”
这是一个有限制的订单。这就是数据库为你做的事情。
我不确定这个不可扩展的内容;无论如何,这都是数据库的作用。
答案 3 :(得分:0)
以下是来自http://pubsub-test.appspot.com/:
的python示例任何人都有一个java?感谢。
from google.appengine.ext import webapp
from google.appengine.ext import db
class Message(db.Model):
body = db.TextProperty(required=True)
sender = db.StringProperty(required=True)
receiver_id = db.ListProperty(int)
class SlimMessage(db.Model):
body = db.TextProperty(required=True)
sender = db.StringProperty(required=True)
class MessageIndex(db.Model):
receiver_id = db.ListProperty(int)
class MainHandler(webapp.RequestHandler):
def get(self):
receiver_id = int(self.request.get('receiver_id', '1'))
key_only = self.request.get('key_only').lower() == 'on'
if receiver_id:
if key_only:
keys = db.GqlQuery(
'SELECT __key__ FROM MessageIndex WHERE receiver_id = :1',
receiver_id).fetch(10)
messages.extend(db.get([k.parent() for k in keys]))
else:
messages.extend(Message.gql('WHERE receiver_id = :1',
receiver_id).fetch(10))