【问题标题】:Correct method to initialize a database table with web2py DAL.define_table()使用 web2py DAL.define_table() 初始化数据库表的正确方法
【发布时间】:2011-12-28 00:18:17
【问题描述】:

我正在尝试使用完全相同的 SQL 定义动态构建名为 db.blogdb.code 的表。在我定义它们之后,我想用 10 行随机数据填充它们,并且不再执行该初始化代码。

我的问题是,当我查看db.codedb.blog 的newblog appadmin 界面时,每次在浏览器上点击刷新时都会执行初始化代码:https://172.25.1.1/newblog/appadmin/select/db?query=db.code.id>0

我在newblog/models/newblog.py中初始化db.blogdb.code

from gluon import *
from gluon.contrib.populate import populate

## initialize db.blog and db.code: 
##     At runtime, build TAGGED_TABLES (once)
TAGGED_TABLES = set(['blog', 'code'])
for tt in TAGGED_TABLES:
    if not db.get(tt, False):
        db.define_table(tt,
            Field('name', length=32, notnull=True),
            Field('value', length=65535, notnull=True),
            Field('tags', type='list:reference tag', unique=False, notnull=False),
            )

        populate(db.get(tt), 10)

        ## cross-reference db.tagged_tables to this one
        db.tagged_tables.insert(name=tt,
            database_pointer='reference %s' % tt)
        db.commit()

不知何故,if not db.get(tt, False): 允许多次执行它下面的例程。我不明白为什么...如果表已经创建,那么not db.get(tt, False) 应该是False。但是,web2py 从不跳过初始化代码,这意味着 db.blogdb.code 在每次重新加载时增加 10 个条目。

问题:为什么if not db.get(tt, False): 不能防止多次执行?

我在 Debian 6.0 / sqlite 3.7.3 / Cherokee 1.2.101 / uWSGI 0.9.9.3 上运行 web2py 1.99.4

解决方案

根据 Interrobang 的回答,正确的写法是:

from gluon import *
from gluon.contrib.populate import populate


TAGGED_TABLES = set(['blog', 'code'])
for tt in TAGGED_TABLES:
    # db.define_table() must be called on **every page**
    #    this sets things up in memory...
    db.define_table(tt,
        Field('name', length=32, notnull=True),
        Field('value', length=65535, notnull=True),
        Field('tags', type='list:reference tag', unique=False, notnull=False),
        )

    ## initialize db.blog and db.code: 
    ##     At runtime, populate tables named in TAGGED_TABLES (once)
    if not (db(db.get(tt).id>0).select()):
        populate(db.get(tt), 10)
        ## cross-reference db.tagged_tables to this table (named in var tt)
        db.tagged_tables.insert(name=tt,
            database_pointer='reference %s' % tt)
        db.commit()

现在db.blogdb.code 的大小保持不变。

总结

每次页面渲染都必须调用db.define_tables();我的理解(它只需要运行一次就可以将表定义写入磁盘)是不正确的。

【问题讨论】:

  • +1 以获得很好的解决方案总结。
  • @Interrobang,谢谢。我也投了赞成票。您的理解是正确完成这项工作的关键

标签: python database web2py


【解决方案1】:

您可以添加一个fixtures文件,该文件基本上包含在创建表时要插入一次的默认数据。

这里给出一个例子:http://thadeusb.com/weblog/2010/4/21/using_fixtures_in_web2py

【讨论】:

  • 这是很好的信息,我将来可能会走这条路;但是,我仍然想知道为什么我的代码被破坏了。
  • 虽然我不确定db.get() 语法,但db.define_table 并不是您只运行一次即可创建表。它还定义了您的模型,并且应该在每次页面加载时运行(尽管 web2py 足够聪明,不会重新创建您的表,如果它存在,它会尝试在模型更改时更新表)。所以 db.get() 没有什么可得到的,因为你还没有定义模型。
  • db.get(<tablename>, False) 是我尝试检查表是否已在数据库中初始化。因为db 的行为就像一个python 字典,所以看起来是正确的咒语;但是,根据您的意见,我需要做些其他事情。
猜你喜欢
  • 2021-01-08
  • 1970-01-01
  • 2023-03-12
  • 2015-03-16
  • 1970-01-01
  • 2022-08-11
  • 2019-10-24
  • 2019-10-03
  • 1970-01-01
相关资源
最近更新 更多