如何将这个复杂的SQL转换为Django模型查询?

Saq*_*Ali 5 python mysql django

我正在编写一个Python/Django应用程序来进行一些库存分析.

我有两个非常简单的模型,如下所示:

class Stock(models.Model):
    symbol = models.CharField(db_index=True, max_length=5, null=False, editable=False, unique=True)

class StockHistory(models.Model):
    stock = models.ForeignKey(Stock, related_name='StockHistory_stock', editable=False)
    trading_date = models.DateField(db_index=True, null=False, editable=False)
    close = models.DecimalField(max_digits=12, db_index=True, decimal_places=5, null=False, editable=False)

    class Meta:
        unique_together = ('stock', 'trading_date')
Run Code Online (Sandbox Code Playgroud)

这是我填充的虚拟数据:

import datetime
a = Stock.objects.create(symbol='A')
b = Stock.objects.create(symbol='B')
c = Stock.objects.create(symbol='C')
d = Stock.objects.create(symbol='D')

StockHistory.objects.create(trading_date=datetime.date(2018,1,1), close=200, stock=a)
StockHistory.objects.create(trading_date=datetime.date(2018,1,2), close=150, stock=a)
StockHistory.objects.create(trading_date=datetime.date(2018,1,3), close=120, stock=a)
StockHistory.objects.create(trading_date=datetime.date(2018,4,28), close=105, stock=a)
StockHistory.objects.create(trading_date=datetime.date(2018,5,3), close=105, stock=a)

StockHistory.objects.create(trading_date=datetime.date(2017,5,2), close=400, stock=b)
StockHistory.objects.create(trading_date=datetime.date(2017,11,11), close=200, stock=b)
StockHistory.objects.create(trading_date=datetime.date(2017,11,12), close=300, stock=b)
StockHistory.objects.create(trading_date=datetime.date(2017,11,13), close=400, stock=b)
StockHistory.objects.create(trading_date=datetime.date(2017,11,14), close=500, stock=b)

StockHistory.objects.create(trading_date=datetime.date(2018,4,28), close=105, stock=c)
StockHistory.objects.create(trading_date=datetime.date(2018,4,29), close=106, stock=c)
StockHistory.objects.create(trading_date=datetime.date(2018,4,30), close=107, stock=c)
StockHistory.objects.create(trading_date=datetime.date(2018,5,1), close=108, stock=c)
StockHistory.objects.create(trading_date=datetime.date(2018,5,2), close=109, stock=c)
StockHistory.objects.create(trading_date=datetime.date(2018,5,3), close=110, stock=c)
StockHistory.objects.create(trading_date=datetime.date(2018,5,4), close=90, stock=c)
Run Code Online (Sandbox Code Playgroud)

我想找到过去一周内年度低点的所有股票.

但是为了使这个问题更简单,只要假设我想找到自那时起'2017-05-04'或之后发生的最低点的所有股票'2018-04-30'.下面是我写的SQL来找到它.有用.

但是我需要帮助找出要写的Django Query以获得与此SQL相同的结果.我该怎么做?

mysql> select
    ->     s.symbol,
    ->     sh.trading_date,
    ->     low_table.low
    -> from
    ->     (
    ->         select
    ->             stock_id,
    ->             min(close) as low
    ->         from
    ->             stocks_stockhistory
    ->         where
    ->             trading_date >= '2017-05-04'
    ->         group by
    ->             stock_id
    ->     ) as low_table,
    ->     stocks_stockhistory as sh,
    ->     stocks_stock as s
    -> where
    ->     sh.stock_id = low_table.stock_id
    ->     and sh.stock_id = s.id
    ->     and sh.close = low_table.low
    ->     and sh.trading_date >= '2018-04-30'
    -> order by
    ->     s.symbol asc;
+--------+--------------+-----------+
| symbol | trading_date | low       |
+--------+--------------+-----------+
| A      | 2018-05-03   | 105.00000 |
| C      | 2018-05-04   |  90.00000 |
+--------+--------------+-----------+
2 rows in set (0.02 sec)
Run Code Online (Sandbox Code Playgroud)

Joh*_*fis 7

编辑:我设法使用Django子查询改革解决方案.

我们可以使用Django将查询翻译成Django ORM aggregates with SubQuery expressions:

  1. 创建子查询以检索close每个的最低值symbol:

    from django.db.models import OuterRef, Subquery, Min     
    
    lows = StockHistory.objects.filter(
        stock=OuterRef('stock'), 
        trading_date__gte='2017-05-04'
    ).values('stock__symbol')
    .annotate(low=Min('close'))
    .filter(trading_date__gte='2018-04-30')
    
    Run Code Online (Sandbox Code Playgroud)
    • 分解:

      • filter查询集只获取股票trading_date >= '2017-05-04'.
      • "GROUP BY" stock__symbol(在Djnago中的分组示例:GROUP BY ... MIN/MAX,GROUP BY ... COUNT/SUM).
      • annotatelow每个元素的最低价格().
      • filter再次查询集以仅获取具有low字段的对象trading_date >= '2018-04-30'.
    • 中级结果:

      虽然我们无法在此阶段获得结果,但子查询将如下所示:

      [
          {'stock__symbol': 'A', 'low': Decimal('105.00000')},            
          {'stock__symbol': 'C', 'low': Decimal('90.00000')}
      ]
      
      Run Code Online (Sandbox Code Playgroud)

      我们错过了trading_date.

  2. 利用子查询来检索特定StockHistory对象:

    StockHistory.objects.filter(
        stock__symbol=Subquery(lows.values('stock__symbol')),
        close=Subquery(lows.values('low')),
        trading_date__gte='2018-04-30'
    ).values('stock__symbol', 'trading_date', 'close')
    .order_by('stock__symbol')
    
    Run Code Online (Sandbox Code Playgroud)
    • 分解:

      • lows.values('stock__symbol') 和lows.values('low')从子查询中检索相应的值.
      • filter针对lows子查询值的查询集.也是filter在指定日期close之前,以消除在该日期之前发生的低价格.
      • 获取指定的values.
      • stock__symbol(默认情况下ascending)排序结果.
    • 结果:

      [
          {
              'close': Decimal('105.00000'), 
              'trading_date': datetime.date(2018, 5, 3), 
              'stock__symbol': 'A'
          }, 
          {
              'close': Decimal('90.00000'), 
              'trading_date': datetime.date(2018, 5, 4), 
              'stock__symbol': 'C'
          }
      ]
      
      Run Code Online (Sandbox Code Playgroud)

  • 是的@JohnMoutafis,你是对的.现在你提供的Django给出了与我的SQL相同的结果.我有一个后续问题.但我会在此期间接受你的回答并单独发布. (2认同)
  • 以下是我的后续问题:/sf/ask/3520024021/ (2认同)

use*_*741 6

对于较新版本的Django(1.11,2.0):

from django.db.models import Min
low_stocks_qs = StockHistory.objects.filter(trading_date__gt='2017-05-04').annotate(low=Min('close')).filter(trading_date__gte='2018-04-30').order_by('stock__symbol')
Run Code Online (Sandbox Code Playgroud)

您可以遍历查询集以获取low和stock.symbol的单个值,可能是这样的:

low_stocks_dict = {}
for inst in low_stocks_qs:
    low_stocks_dict[inst.stock.Symbol] = inst.low
Run Code Online (Sandbox Code Playgroud)