当列表长度不同时,迭代Python中的多个列表

时间:2016-07-13 02:29:05

标签: python mysql for-loop flask jinja2

我有这个数据表,我从MySQL数据库中提取数据并尝试匹配行,但我无法弄清楚最后一部分。我使用Flask使用' izip_longest'传递三个词典列表。到Jinja模板然后在Jinja模板中的for循环来遍历每行的变量。问题是我迭代的三个列表具有可变长度并且不会总是具有相同的长度。

我已经如下所示,但是我需要在整个行中匹配日期。在这种情况下,S-Date / 07-11应该是整行的07-11,或者在其他列中是空白。

S-Date | S-Place | G-Date | G-Place | F-Date | F-Place
------------------------------------------------------
 07-11     7       07-12     7        07-11      7
 07-12     7       07-13     7        
 07-13     7       07-14     7
 07-14     7

我觉得我已经接近搞清楚但是我一直在盯着这个并且不能完全理解它。

这是我从中获取数据的SQL表:

location | date | status
------------------------
 001      07-10    success
 002      07-10    success
 123      07-11    fail
 222      07-11    fail
 333      07-11    fail
 232      07-11    fail
 444      07-12    pending
 555      07-13    pending

这是获取每个特定日期失败次数的查询:

SELECT 'date', COUNT(`location`) as 'location' FROM mytable 
  WHERE YEARWEEK(`date`, 1) = YEARWEEK(CURDATE(), 1) 
   AND `status` LIKE '%fail%' GROUP BY DATE(`date`)

这是关于每天的成功次数:

SELECT `date`, COUNT(`location`) as 'location' FROM mytable 
   WHERE YEARWEEK(`date`, 1) = YEARWEEK(CURDATE(), 1) 
     AND `status` NOT LIKE '%fail%' GROUP BY DATE(`date`)

我需要将失败和成功与当天分组相匹配,这样我就可以生成一个堆积条形图来显示单个日期/条的每个数字。

1 个答案:

答案 0 :(得分:1)

不确定所有这些S-DateF-Date业务是什么,但我认为您想知道的是每个唯一日期的成功和失败总数。

我在Python中有一个非常简单的解决方案:

from collections import defaultdict
from collections import Counter

data = defaultdict(list)

for row in database_table:
    data[row[1]].append(row[2])

results = {}

for date, attempts in data.iteritems():
   stats = Counter(attempts) 
   results[date] = {'total': len(attempts)}
   for stat, count in stats.most_common():
      results[date][stat] = count

现在结果是一个字典,键是日期,值是另一个包含所有统计数据的字典。在您的模板中,您只需:

<table>
   <thead>
      <tr>
        <th>Date</th>
        <th>Total Places</th>
        <th>Success</th>
        <th>Failed</th>
        <th>Pending</th>
      </tr>
   </thead>
   <tbody>
     {% for date, stats in results.iteritems() %}
         <tr>
            <td>{{ date }}</td>
            <td>{{ stats['total'] }}</td>
            <td>{{ stats['success'] }}</td>
            <td>{{ stats['fail'] }}</td>
            <td>{{ stats['pending'] }}</td>
         </tr>
     {% endfor %}
   </tbody>
</table>
  

我遇到麻烦的一件事是日期没有出现在   正确的顺序。他们从07-14到07-11,而不是07-11到   07-14。我知道Python字典没有订购,所以现在我需要   调查如何正确订购日期。

要对日期进行排序,您需要先将它们转换为datetime对象;这样排序就可以了。

import datetime

# rest of the code here

fmt = '%m-%d'
results = [] # a list, so its sortable
for date, attempts in data.iteritems():
   stats = Counter(attempts) 
   record = {'total': len(attempts),
             'date': datetime.datetime.strptime(date, fmt)}
   for stat, count in stats.most_common():
      record[stat] = count

   results.append(record)

results = sorted(results, key=lamdba x: x['date'])

然后,只需调整您的模板:

{% for record in results %}
  <tr>
    <td>{{ record['date'] }}</td>
    <td>{{ record['total'] }}</td>
    <td>{{ record['success'] }}</td>
    <td>{{ record['fail'] }}</td>
    <td>{{ record['pending'] }}</td>
  </tr>
{% endfor %}