JS对大量数据进行多重过滤的方法


Posted in Javascript onNovember 04, 2016

前言

主要的需求是前端通过 Ajax 从后端取得了大量的数据,需要根据一些条件过滤,首先过滤的方法是这样的:

class Filter {
 filterA(s) {
 let data = this.filterData || this.data;
 this.filterData = data.filter(m => m.a === s);
 }
 
 filterB(s) {
 let data = this.filterData || this.data;
 this.filterData = data.filter(m => m.b === s);
 }
}

现在迷糊了,觉得这样处理数据不对,但是又不知道该怎么处理。

发现问题

问题就在过滤上,这样固然可以实现多重过滤(先调用 filterA() 再调用 filterB() 就可以实现),但是这个过滤是不可逆的。

假如过滤过程是这样:

f.filterA("a1");
f.filterB("b1");
f.filterA("a2");

本来是希望按 "a1" 和 "b1" 过滤了数据之后,再修改第一个条件为 "a2",但结果却成了空集。

解决问题

发现了问题,就针对性的解决。这个问题既然是因为过滤过程不可逆造成的,那每次都直接从 this.data 开始过滤,而不是从 this.filterData 开始过滤,就能解决问题。如果要这样做,就需要将选择的过滤条件先记录下来。

记录过滤条件

用一个列表记录过滤条件当然是可行的,但是注意对同一个条件的两次过滤是互斥的,只能保留最后一个,所以应该用 HashMap 更为合适。

class Filter {
 constructor() {
 this.filters = {};
 }

 set(key, filter) {
 this.filters[key] = filter;
 }

 getFilters() {
 return Object.keys(this.filters).map(key => this.filters[key]);
 }
}

这种情况下,像上面的过程表示为

f.set("A", m => m.a === "a1");
f.set("B", m => m.b === "b1");
f.set("A", m => m.a === "a1");

let filters = f.getFilters(); // length === 2;

上面第 3 句设置的 filter 覆盖了第 1 句设置的那个。现在再用最后取得的 filters 依次来过滤原数据 this.data,就能得到正确的结果。

有人会觉得 getFilters() 返回的列表不是按 set 的顺序的——的确,这是 HashMap 的特点,无序。不过对于简单条件的判断,不管谁先谁后,结果是一样的。但是对于一些复合条件判断,就可能会有影响。

确实需要的话,可以通过 array 代替 map 来解决一下顺序的问题,但这样查找效率会降低(线性查找)。如果还想解决查找效率的问题,可以用 array + map 来处理。这里就不多说了。

过滤

实际上在使用的时候,每次都 getFilter() 再用一个循环来处理确实比较慢。既然 data 都封装成 Filter 中,可以考虑直接给一个 filter() 方法来送货过滤接口。

class Filter {
 filter() {
 let data = this.data;
 for (let f of this.getFilters()) {
  data = data.filter(f);
 }
 return data;
 }
}

不过这样我觉得效率不太好,尤其是对大量数据的时候。不妨利用一下 lodash 的延迟处理过程。

利用 lodash 的延迟处理

filter() {
 let chain = _(this.data);
 for (let f of this.getFilters()) {
 chain = chain.filter(f);
 }
 return chain.value();
}

lodash 在数据大于 200 的时候会启用延迟处理过程,也就是说,它会处理成一个循环中依次调用每一个 filter,而不是对每一个 filter 进行一次循环。

延迟处理和非延迟处理通过下图可以看出来区别。非延迟处理总共会进行 n(这里 n = 3) 次大循环,产生 n - 1 个中间结果。而延迟处理只会进行一次大循环,没有中间结果产生。

JS对大量数据进行多重过滤的方法

不过说实在的,我不太喜欢为了一点小事多加载一个库,所以干脆自己做个简单的实现

自己实现延迟处理

filter() {
 const filters = this.getFilters();
 return data.filter(m => {
 for (let f of filters) {
  // 如果某个 filter 已经把它过滤掉了,也不用再用后面的 filter 来判断了
  if (!f(m)) {
  return false;
  }
 }
 return true;
 });
}

里面的 for 循环还可以用 Array.prototype.every 来简化:

filter() {
 const filters = this.getFilters();
 return data.filter(m => {
 return filters.every(f => f(m));
 });
}

数据过滤其实并不是多复杂的事情,只要把思路理清楚,搞明白什么数据是需要保留的,什么数据是临时(中间过程)的,什么数据是最终结果……利用 Array.prototype 中的相关方法,或者诸如 lodash 之类的工具,很容易就处理出来了。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能有所帮助,如果有疑问大家可以留言交流。

Javascript 相关文章推荐
jQuery bxCarousel实现图片滚动切换效果示例代码
May 15 Javascript
jQuery中:not选择器用法实例
Dec 30 Javascript
jQuery+html5实现div弹出层并遮罩背景
Apr 15 Javascript
IE10中flexigrid无法显示数据的解决方法
Jul 26 Javascript
JavaScript中的数据类型转换方法小结
Oct 26 Javascript
项目实践一图片上传之form表单还是base64前端图片压缩(前端图片压缩)
Jul 28 Javascript
jQuery 获取select选中值及清除选中状态
Dec 13 Javascript
Vuejs仿网易云音乐实现听歌及搜索功能
Mar 30 Javascript
基于JavaScript实现活动倒计时效果
Apr 20 Javascript
Vue中 v-if 和v-else-if页面加载出现闪现的问题及解决方法
Oct 12 Javascript
微信小程序防止多次点击跳转(函数节流)
Sep 19 Javascript
JS实现音乐导航特效
Jan 06 Javascript
AngularJS模板加载用法详解
Nov 04 #Javascript
jQuery 遍历map()方法详解
Nov 04 #Javascript
Sortable.js拖拽排序使用方法解析
Nov 04 #Javascript
jQuery图片加载显示loading效果
Nov 04 #Javascript
JavaScript 深层克隆对象详解及实例
Nov 03 #Javascript
jQuery.ajax实现根据不同的Content-Type做出不同的响应
Nov 03 #Javascript
简单学习vue指令directive
Nov 03 #Javascript
You might like
PHP4(windows版本)中的COM函数
2006/10/09 PHP
如何在PHP中进行身份认证
2006/10/09 PHP
PHP的时间戳与具体时间转化的简单实现
2016/06/13 PHP
PHP之图片上传类实例代码(加了缩略图)
2016/06/30 PHP
javascript语句中的CDATA标签的意义
2007/05/09 Javascript
XRegExp 0.2: Now With Named Capture
2007/11/30 Javascript
JSON 和 JavaScript eval使用说明
2010/06/13 Javascript
解决ExtJS在chrome或火狐中正常显示在ie中不显示的浏览器兼容问题
2013/01/11 Javascript
javascript实现通过表格绘制颜色填充矩形的方法
2015/04/21 Javascript
13个PHP函数超实用
2015/10/21 Javascript
javascript正则表达式定义(语法)总结
2016/01/08 Javascript
仅9张思维导图帮你轻松学习Javascript 就这么简单
2016/06/01 Javascript
完美实现八种js焦点轮播图(下篇)
2020/04/20 Javascript
bootstrap日期控件问题(双日期、清空等问题解决)
2017/04/19 Javascript
Angular2学习教程之TemplateRef和ViewContainerRef详解
2017/05/25 Javascript
Node.js使用orm2进行update操作时关联字段无法修改的解决方法
2017/06/13 Javascript
JavaScript函数绑定用法实例分析
2017/11/14 Javascript
Vue-路由导航菜单栏的高亮设置方法
2018/03/17 Javascript
vue2.x 对象劫持的原理实现
2020/04/19 Javascript
轻松实现python搭建微信公众平台
2016/02/16 Python
python中urlparse模块介绍与使用示例
2017/11/19 Python
Php多进程实现代码
2018/05/07 Python
Python实现动态添加属性和方法操作示例
2018/07/25 Python
如何使用python爬虫爬取要登陆的网站
2019/07/12 Python
谈一谈数组拼接tf.concat()和np.concatenate()的区别
2020/02/07 Python
找Python安装目录,设置环境路径以及在命令行运行python脚本实例
2020/03/09 Python
Python基于pyecharts实现关联图绘制
2020/03/27 Python
PyTorch在Windows环境搭建的方法步骤
2020/05/12 Python
Python爬虫教程之利用正则表达式匹配网页内容
2020/12/08 Python
CAT鞋加拿大官网:CAT Footwear加拿大
2020/08/05 全球购物
在线实验室测试:HealthLabs.com
2020/05/03 全球购物
小区的门卫岗位职责
2014/10/01 职场文书
单位租房协议书范本
2014/12/04 职场文书
2016年公务员六五普法心得体会
2016/01/21 职场文书
2019年新郎保证书3篇
2019/10/17 职场文书
解决spring.thymeleaf.cache=false不起作用的问题
2022/06/10 Java/Android