python - SQLAlchemy 中的 GroupBy 和 Sum?

标签 python sql flask sqlalchemy flask-sqlalchemy

我试图将表格中的几个字段分组,然后对这些组求和,但它们被重复计算了。

我的模型如下:

class CostCenter(db.Model):
     __tablename__ = 'costcenter'
     id = db.Column(db.Integer, primary_key=True, autoincrement=True)
     name = db.Column(db.String)
     number = db.Column(db.Integer)

class Expense(db.Model):

    __tablename__ = 'expense'
    id = db.Column(db.Integer, primary_key=True, autoincrement=True)
    glitem_id = db.Column(db.Integer, db.ForeignKey('glitem.id'))
    glitem = db.relationship('GlItem')
    costcenter_id = db.Column(db.Integer, db.ForeignKey('costcenter.id'))
    costcenter = db.relationship('CostCenter')
    value = db.Column(db.Float)
    date = db.Column(db.Date)

我一直在使用:

expenses=db.session.query(Expense,func.sum(Expense.value)).group_by(Expense.date).filter(CostCenter.id.in_([1,2,3]))

当我打印费用时,它会显示后面的 SQL 语句。它看起来对我来说是正确的,但我对 SQL 并不熟悉。问题是它输出为 sum_1 的值被多次计算。如果我在“in statement”中有 [1] 项,它将对所有三个项求和。如果我有 [1,2],它会将所有三个加起来然后加倍,如果我有 [1,2,3],它会将所有三个加起来并加三倍。我不确定为什么要多次计数。我该如何解决?

SELECT expense.id AS expense_id, expense.glitem_id AS expense_glitem_id, expense.costcenter_id AS         expense_costcenter_id, expense.value AS expense_value, expense.date AS expense_date, sum(expense.value) AS sum_1 
FROM expense, costcenter 
WHERE costcenter.id IN (:id_1, :id_2, :id_3) GROUP BY expense.date

谢谢!

最佳答案

这里有几个问题;您似乎没有查询正确的事情。按 Expense.date 分组时选择 Expense 对象是没有意义的。 CostCenter 和 Expense 之间需要有一些连接条件,否则行将重复,每个成本中心都计数,但两者之间没有关系。

您的查询应如下所示:

session.query(
    Expense.date,
    func.sum(Expense.value).label('total')
).join(Expense.cost_center
).filter(CostCenter.id.in_([2, 3])
).group_by(Expense.date
).all()

生成此 sql:

SELECT expense.date AS expense_date, sum(expense.value) AS total 
FROM expense JOIN cost_center ON cost_center.id = expense.cost_center_id 
WHERE cost_center.id IN (?, ?) GROUP BY expense.date

这是一个简单的可运行示例:

from datetime import datetime
from sqlalchemy import create_engine, Column, Integer, ForeignKey, Numeric, DateTime, func
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import Session, relationship

engine = create_engine('sqlite://', echo=True)
session = Session(bind=engine)
Base = declarative_base(bind=engine)


class CostCenter(Base):
    __tablename__ = 'cost_center'

    id = Column(Integer, primary_key=True)


class Expense(Base):
    __tablename__ = 'expense'

    id = Column(Integer, primary_key=True)
    cost_center_id = Column(Integer, ForeignKey(CostCenter.id), nullable=False)
    value = Column(Numeric(8, 2), nullable=False, default=0)
    date = Column(DateTime, nullable=False)

    cost_center = relationship(CostCenter, backref='expenses')


Base.metadata.create_all()

session.add_all([
    CostCenter(expenses=[
        Expense(value=10, date=datetime(2014, 8, 1)),
        Expense(value=20, date=datetime(2014, 8, 1)),
        Expense(value=15, date=datetime(2014, 9, 1)),
    ]),
    CostCenter(expenses=[
        Expense(value=45, date=datetime(2014, 8, 1)),
        Expense(value=40, date=datetime(2014, 9, 1)),
        Expense(value=40, date=datetime(2014, 9, 1)),
    ]),
    CostCenter(expenses=[
        Expense(value=42, date=datetime(2014, 7, 1)),
    ]),
])
session.commit()

base_query = session.query(
    Expense.date,
    func.sum(Expense.value).label('total')
).join(Expense.cost_center
).group_by(Expense.date)

# first query considers center 1, output:
# 2014-08-01: 30.00
# 2014-09-01: 15.00
for row in base_query.filter(CostCenter.id.in_([1])).all():
    print('{}: {}'.format(row.date.date(), row.total))

# second query considers centers 1, 2, and 3, output:
# 2014-07-01: 42.00
# 2014-08-01: 75.00
# 2014-09-01: 95.00
for row in base_query.filter(CostCenter.id.in_([1, 2, 3])).all():
    print('{}: {}'.format(row.date.date(), row.total))

关于python - SQLAlchemy 中的 GroupBy 和 Sum?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25558582/

相关文章:

python - 使用 Django 处理动态静态文件路径

SQL:如果在 postgresql 中重新参加考试,如何从学生表中选择通过行?

mysql - 获取多个子查询结果的 SUM

python - 在 Flask 中设置静态文件夹路径

python - 如何在 uWSGI 中正确加载 Flask 应用程序模块?

python - 如何在具有时区感知时间戳列的数据帧上附加?

python - 如何检测 Sprite 是否已在pygame中被点击

python - python如何将任意对象转换为列表?

c# - LINQ Group By 项目成非匿名类型?

python - Flask:创建在多个请求中保留的对象