- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
peewee 允许通过 insert_many()
进行批量插入和 insert_from()
,但是 insert_many()
允许插入数据列表,但不允许从数据库的其他部分计算数据。 insert_from()
确实允许从数据库的其他部分计算数据,但不允许从 python 发送任何数据。
假设模型结构如下:
class BaseModel(Model):
class Meta:
database = db
class Person(BaseModel):
name = CharField(max_length=100, unique=True)
class StatusUpdate(BaseModel):
person = ForeignKeyField(Person, related_name='statuses')
status = TextField()
timestamp = DateTimeField(constraints=[SQL('DEFAULT CURRENT_TIMESTAMP')], index=True)
以及一些初始数据:
Person.insert_many(rows=[{'name': 'Frank'}, {'name': 'Joe'}, {'name': 'Arnold'}]).execute()
print ('Person.select().count():',Person.select().count())
输出:
Person.select().count(): 3
假设我们要添加一堆新的状态更新,例如此列表中的更新:
new_status_updates = [ ('Frank', 'wat')
, ('Frank', 'nooo')
, ('Joe', 'noooo')
, ('Arnold', 'nooooo')]
我们可能会尝试像这样使用insert_many()
:
StatusUpdate.insert_many( rows=[{'person': 'Frank', 'status': 'wat'}
, {'person': 'Frank', 'status': 'nooo'}
, {'person': 'Joe', 'status': 'noooo'}
, {'person': 'Arnold', 'status': 'nooooo'}]).execute()
但这会失败:person
字段需要一个 Person
模型或 Person.id
,我们必须额外创建一个查询以从名称中检索这些内容。
我们也许可以使用 insert_from()
来避免这种情况,它允许我们进行子查询,但 insert_from()
无法处理我们的列表或字典。该怎么办?
最佳答案
一个想法是使用 SQL VALUES
子句作为 SELECT
语句的一部分。
如果您熟悉 SQL,您以前可能见过 VALUES
子句,它通常用作 INSERT
语句的一部分,如下所示:
INSERT INTO statusupdate (person_id,status)
VALUES (1, 'my status'), (1, 'another status'), (2, 'his status');
这告诉数据库将三行(又称元组)插入到表statusupdate
中。
插入内容的另一种方法是执行以下操作:
INSERT INTO statusupdate (person_id,status)
SELECT ..., ... FROM <elsewhere or subquery>;
这相当于 peewee 提供的 insert_from()
功能。
但是您还可以执行另一项不太常见的操作:您可以在any select 中使用VALUES
子句来提供文字值。示例:
SELECT *
FROM (VALUES (1,2,3), (4,5,6)) as my_literal_values;
这将返回两行/元组的结果集,每行/元组有 3 个值。
因此,如果您可以将“批量”插入转换为 SELECT/FROM/VALUES
语句,那么您就可以执行您需要执行的任何转换(即将 Person.name 值转换为相应的值) Person.id 值),然后将其与 peewee 'insert_from()` 功能结合起来。
让我们看看这会是什么样子。
首先让我们开始构建 VALUES
子句本身。我们想要正确转义的值,因此我们现在将使用问号而不是值,稍后再放入实际值。
#this is gonna look like '(?,?), (?,?), (?,?)'
# or '(%s,%s), (%s,%s), (%s,%s)' depending on the database type
values_question_marks = ','.join(['(%s, %s)' % (db.interpolation,db.interpolation)]*len(new_status_updates))
下一步是构造值子句。这是我们的第一次尝试:
--the %s here will be replaced by the question marks of the clause
--in postgres, you must have a name for every item in `FROM`
SELECT * FROM (VALUES %s) someanonymousname
好的,现在我们有一个结果集,如下所示:
name | status
-----|-------
... | ...
除了!没有列名称。这很快就会让我们有点心痛,所以我们必须找到一种方法来为结果集提供正确的列名称。
postgres 的方式是只改变 AS
子句:
SELECT * FROM (VALUES %s) someanonymousname(name,status)
sqlite3 不支持(grr)。
因此,我们陷入了困境。幸运的是 stackoverflow 提供了: Is it possible to select sql server data using column ordinal position ,我们可以构造这样的东西:
SELECT NULL as name, NULL as status WHERE 1=0
UNION ALL
SELECT * FROM (VALUES %s) someanonymousname
首先使用正确的列名创建一个空结果集,然后将 VALUES
子句中的结果集连接到它。这将产生一个具有正确列名的结果集,将在 sqlite3 和 postgres 中工作。
现在把这个带回 peewee:
values_query = """
(
--a trick to make an empty query result with two named columns, to more portably name the resulting
--VALUES clause columns (grr sqlite)
SELECT NULL as name, NULL as status WHERE 1=0
UNION ALL
SELECT * FROM (VALUES %s) someanonymousname
)
"""
values_query %= (values_question_marks,)
#unroll the parameters into one large list
#this is gonna look like ['Frank', 'wat', 'Frank', 'nooo', 'Joe', 'noooo' ...]
values_query_params = [value for values in new_status_updates for value in values]
#turn it into peewee SQL
values_query = SQL(values_query,*values_query_params)
data_query = (Person
.select(Person.id, SQL('values_list.status').alias('status'))
.from_(Person,values_query.alias('values_list'))
.where(SQL('values_list.name') == Person.name))
insert_query = StatusUpdate.insert_from([StatusUpdate.person, StatusUpdate.status], data_query)
print (insert_query)
insert_query.execute()
print ('StatusUpdate.select().count():',StatusUpdate.select().count())
输出:
StatusUpdate.select().count(): 4
关于python - 小便 orm : bulk insert using a subquery but is based on python-side-data,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37297768/
peewee 允许通过 insert_many() 进行批量插入和 insert_from() ,但是 insert_many() 允许插入数据列表,但不允许从数据库的其他部分计算数据。 insert
我是一名优秀的程序员,十分优秀!