转 关于GAE你可能不知道的十件事

# 程序版本号并不是number类型而是string
尽管在多数的例子中都将app.yaml和appengine-web.xml中的'version'字段作为number类型,实际上这只是人们的一种习惯。程序版本可以是任何URL中允许的string类型。例如,你可以将你的版本设置成"live"和"dev",这样你就可以通过"live.latest.yourapp.appspot.com"和"dev.latest.yourapp.appspot.com"这样的地址来访问了。
# 你可以在程序中同时使用多个版本
这里有一点是很NB的,不同的版本可以使用不同的运行环境。换言之,你可以为你的程序开发使用Java语言的版本和Python语言的版本。
# Java运行时支持任何编译成Java bytecode的语言
这就是为什么人们需要Java runtime的原因,一旦这家伙被支持了就等于你可以在GAE上使用什么JRuby,Groovy,Scala,Rhino(JavaScript编译器),Quercus(PHP编译器),甚至Jython!
# 使用'IN'和'!='运算符可以生成多重数据查询
例如,查询"SELECT * FROM People WHERE name IN ('Bob', 'Jane')"将会被编译成两个查询等价于使用"SELECT * FROM People WHERE name = 'Bob'"和"SELECT * FROM People WHERE name = 'Jane'"并合并查询的结果。
查 询"SELECT * FROM People WHERE name IN ('Bob', 'Jane') AND age != 25"将会为每种可能生成生成查询,(age小于或者等于25, 和name是否为'Bob'或者'Jane'),然后将每种可能合并成结果。
# 你可以使用批处理执行put,get和delete操作来提高效率。
每当你做出一个数据请求,比如一个查询或者get() 操作,你的程序将会对数据库发出请求,而数据库将会执行这些操作并返还一个response。请求到响应的循环过程将会花费一定时间,所以如果你执行的请求过多的话将会导致用户等待结果的时间过长(万恶的datastore timeout: operation took too long异常)。
很幸运的是,GAE提供一个简单的方法来减少数据查询中的这种循环:比处理操作。db.put(),db.get(),和 db.delete()函数都可以接受可选的list参数来执行操作。当一个传递一个list以后这些操作将会并行的遍历list的每一个元素,这样就大大节省了操作的时间。例如:

for entity in MyModel.all().filter("color =",
    old_favorite).fetch(100):
  entity.color = new_favorite
  entity.put()

更新操作需要一个数据库往返的查询,还需要加上一次操作来更新数据库中实体的属性————也就是说一共需要101次的往返!再看下面的例子:

updated = []
for entity in MyModel.all().filter("color =",
    old_favorite).fetch(100):
  entity.color = new_favorite
  updated.append(entity)
db.put(updated)

这个例子中只需要2次数据库往返,也就是说我们将101次往返减少到了2次。
# 数据库的性能并不取决于其中实体的数量
# 建立索引的时间开销并不全部取决于它的大小
# 'Stored Data'的值是按日更新的
# app.yaml,web.xml和appengine-web.xml中指明了项目的处理顺序
人们经常犯的一个错误时忘记配置程序处理文件的顺序是自上而下的。比如,当使用remote_api时候很多人这样做:

handlers:
- url: /.*
  script: request.py

- url: /remote_api
  script: $PYTHON_LIB/google/appengine/ext/remote_api/handler.py
  login: admin

上面的例子看起来没有什么问题,但是由于handler是按照顺序处理的,request.py是首先出现的,而所有请求包括对 remote_api的请求都是使用request.py处理的。由于request.py加载后并没有出现remote_api,所以就会造成a 404 Not Found error的错误。解决的方法是将request.py的handler放置到remote_api的handler下面。在Java环境的配置文件中也是一样。
# 你不需要手动构造GQL字符
下面是一个反例:

q = db.GqlQuery("SELECT * FROM People "
    "WHERE first_name = '" + first_name
    + "' AND last_name = '" + last_name + "'")

很幸运,GAE支持在GqlQuery语句中使用参数替换,上面例子可以改写成:

q = db.GqlQuery("SELECT * FROM People "
    "WHERE first_name = :1 "
    "AND last_name = :2", first_name, last_name)

GqlQuery还支持使用name来代替数字参数:

q = db.GqlQuery("SELECT * FROM People "
    "WHERE first_name = :first_name "
    "AND last_name = :last_name",
    first_name=first_name, last_name=last_name)

这样做出了能使代码看起来更加简洁以外,还可以进行一些优化。如果你需要使用不同的值来进行多次同样的查询,你可以使用bind()函数来为每个查询重新绑定参数。这比起每次都要构造新的查询更加快捷,因为查询只需要分析一次:

q = db.GqlQuery("SELECT * FROM People "
    "WHERE first_name = :first_name "
    "AND last_name = :last_name")
for first, last in people:
  q.bind(first, last)
  person = q.get()
  print person

你可能感兴趣的:(scala,python,groovy,GAE,jython)