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来输出哈希里的值,因此整个页面对系统的产生的负担非常小,页面加载速度也非常快,几乎都是秒开。
以上便是我对此类页面的一些心得,希望内给各位读者一些启发。若是有大神觉得有更简洁高效的方法,希望能不吝赐教。