提升统计页面获取数据效率的实现思路

ROR项目中经常会有统计页面的需求。统计页面由于数据量和运算量较大,因此不同的数据获取计算方法,可能对页面加载的速度有极大的影响(可能加载时间会差10倍以上)。下面分享一下我最近用过的一种方法(其实应该说是思路)

应用场景——比如我需要统计全国各个省份下,在指定时间范围内,所有订单的状况(未付款、已付款、已收货、已评价)

解决思路——
早期我可能会用比较笨的方法,先将31个省份用each迭代输出,然后每行再根据省份,去数据库里查询省份==当前省份,状态等于指定状态的订单的总数。这么算下来,先对省份表进行了一次查询,然后31个省份,每个省份针对四种不同的状态,各查询了四次,总计对数据库进行了125次查询。这还是省份数量和状态数量不多的情况。这样做,代码写起来虽然很简单,但付出的代价是非常大的,当数据量大到一定程度时,可能会出现页面加载超时而报错的结果。

现在,我说一下我的解决思路:对于这类的统计,推荐用sql语句一次性从数据库里取出所有需要用到的结果,然后按照一定规则,将结果推入一个哈希中,最后再view里合理调用这个哈希进行输出。拿上面的例子来说,具体的实现方法是这样的:

1.取数据

sql = "select o.province_id, o.status, count(o.id) as total_num from orders o where o.created_at >= 'xxxx-xx-xx' and o.created_at <= 'xxxx-xx-xx' group by o.province_id, o.status"
search  = Order.find_by_sql([sql])

2.创建一个存结果的hash

@result = {}
seach.each do |s|
  @result["#{s.province}_#{s.status}"] = s.total_num
end
上面这些代码我大致解释一下:先通过数据库的group by方法,把需要的统计结果,按照省份和订单状态的维度进行切割,并取出。然后通过这两个参数来往@result这个空的哈希里推送数据,理论上,每个省份会有四条数据(因为可能为空)
然后我们再一次性从province表中取出需要用到的省份信息
@provinces = Province.all.map{|p| [p.id, p.name] }

3.前台view里输出

假设order表的status字段用了枚举: Status = {unpay: 1, payed: 2, received: 3, commented: 4},我们用Oreder::Status.values可以直接获得一个[1, 2, 3, 4]数组
%table
  %tr 
    %th 省份 
    %th 未付款
    %th 已付款
    %th 已收货
    %th 已评价
  @provinces.each do |p|
    %tr
      %td= p[1]#这个就是省份的名称,因为@provinces是一个数组,数组的每个元素又是一个数组
      Order::Status.values.each do |s|
        %td= @result["#{p[0]}_#{s}"] || 0
#上面这一句是view最重要的方法,根据在controller里推送进@result哈希里的数据的命名规则,调出这些数据,如果没有被调用到,说明该时间段内,该省份在当前状态下,没有订单,因此如果为空,则补一个0即可。
#至此整个统计页面都完成了,可以看到,首先,代码量非常少,简洁易懂。其次,整个统计只对数据库进行了两次查询,并且在view里也没有逐项调用子方法去计算结果,而是简单的利用.each来输出哈希里的值,因此整个页面对系统的产生的负担非常小,页面加载速度也非常快,几乎都是秒开。

FROM:http://www.jianshu.com/p/fe46fe52a317

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值