如何通过逐行求和(SQL Alchemy)来创建布尔掩码?
Posted
技术标签:
【中文标题】如何通过逐行求和(SQL Alchemy)来创建布尔掩码?【英文标题】:How can I create a boolean mask by summing row-wise (SQL Alchemy)? 【发布时间】:2021-08-10 03:41:14 【问题描述】:给定:
一个数字 SQL Alchemy 表 (<class 'sqlalchemy.sql.schema.Table'>
)
以及 SQL Alchemy 列 (<class 'sqlalchemy.sql.elements.ColumnClause'>
) 的列表(长度可能不同)
我需要在列的逐行总和等于数字的记录上构建一个布尔掩码 True
。
例子:
数字 = 6 column_list = [sa.column(col_1), sa.column(col_2), sa.column(col_3)] 表:col_1 | col_2 | col_3 | col_4 |
---|---|---|---|
1 | 2 | 3 | 3 |
2 | 0 | 2 | 8 |
6 | 0 | 0 | 1 |
返回:[真,假,真]
警告:必须在执行查询之前执行计算(例如,在where
子句、case
语句等中),而不是在执行 fetchall 或类似操作之后。
我什至不知道从哪里开始。我知道我可以选择列:
sa.select(column_list).select_from(table)
另外,可能还需要一个案例陈述:
sa.case([(ROW_WISE_SUM == number, True)], else=False)
编辑:我想我越来越近了,但我仍然不知道如何处理列子句列表:
sa.case((sa.func.sum(column_list) == number, True), else_=False)
psycopg2.ProgrammingError: can't adapt type 'ColumnClause'
.
.
.
[SQL: CASE WHEN (sum(%(sum_1)s) = %(sum_2)s) THEN %(param_1)s ELSE %(param_2)s END]
[parameters: 'sum_1': [<sqlalchemy.sql.elements.ColumnClause at 0x7fe62bf075d0; col_1>, <sqlalchemy.sql.elements.ColumnClause at 0x7fe62bf6e090; col_2>, <sqlalchemy.sql.elements.ColumnClause at 0x7fe62b0fabd0; col_3>], 'sum_2': 6, 'param_1': True, 'param_2': False]
(Background on this error at: http://sqlalche.me/e/14/f405)
【问题讨论】:
【参考方案1】:一种方法是使用子查询:
# create test table
tbl = sa.Table(
"so68720845",
sa.MetaData(),
sa.Column("id", sa.Integer, primary_key=True, autoincrement=False),
sa.Column("col_1", sa.Integer),
sa.Column("col_2", sa.Integer),
sa.Column("col_3", sa.Integer),
sa.Column("col_4", sa.Integer),
)
tbl.drop(engine, checkfirst=True)
tbl.create(engine)
# insert test data
with engine.begin() as conn:
conn.execute(
tbl.insert(),
[
"id": 1, "col_1": 1, "col_2": 2, "col_3": 3, "col_4": 3,
"id": 2, "col_1": 2, "col_2": 0, "col_3": 2, "col_4": 8,
"id": 3, "col_1": 6, "col_2": 0, "col_3": 0, "col_4": 1,
],
)
# do the test
target_value = 6
col_list = [tbl.c.col_1, tbl.c.col_2, tbl.c.col_3]
col_sum = sum(col_list)
with engine.begin() as conn:
subq = sa.select(tbl.c.id, col_sum.label("col_sum")).subquery()
results = conn.execute(
sa.select(subq.c.id, sa.case((subq.c.col_sum == 6, 1), else_=0))
).fetchall()
print(results) # [(1, 1), (2, 0), (3, 1)]
【讨论】:
以上是关于如何通过逐行求和(SQL Alchemy)来创建布尔掩码?的主要内容,如果未能解决你的问题,请参考以下文章
如何在 Sql Alchemy Python 中为 MS SQL 后端指定事务隔离级别
如何在 SQL Alchemy 中仅将 Python 表达式用作混合属性
python 如何使用SQL Alchemy编译INSERT ... ON DUPLICATE KEY UPDATE并支持批量插入。