资讯专栏INFORMATION COLUMN

学习JavaScript数据结构与算法(三):集合

BDEEFE / 1488人阅读

摘要:至于这三个的具体概念,可以看图中集合的实现首先,创建一个构造函数。前端路漫漫,且行且歌的前端乐园原文链接寒假前端学习学习数据结构与算法三集合

</>复制代码

  1. 本系列的第一篇文章: 学习JavaScript数据结构与算法(一),栈与队列
    第二篇文章:学习JavaScript数据结构与算法(二):链表
    第三篇文章:学习JavaScript数据结构与算法(三):集合
    第四篇文章:学习JavaScript数据结构与算法(四):二叉搜索树

集合(Set)

说起集合,就想起刚进高中时,数学第一课讲的就是集合。因此在学习集合这种数据结构时,倍感亲切。
集合的基本性质有一条: 集合中元素是不重复的。因为这种性质,所以我们选用了对象来作为集合的容器,而非数组。
虽然数组也能做到所有不重复,但终究过于繁琐,不如集合。

集合的操作

集合的基本操作有交集、并集、差集等。这儿我们介绍JavaScipt集合中交集、并集、差集的实现。至于这三个的具体概念,可以看图:

JavaScipt中集合的实现

首先,创建一个构造函数。

</>复制代码

  1. /**
  2. * 集合的构造函数
  3. */
  4. function Set() {
  5. /**
  6. * 集合元素的容器,以对象来表示
  7. * @type {Object}
  8. */
  9. var items = {};
  10. }

集合需要有如下方法:

has(value): 检测集合内是否有某个元素

add(value): 给集合内添加某个元素

remove(value): 移除集合中某个元素

clear(value): 清空集合

size(): 返回集合长度

values(): 返回集合转换的数组

union(otherSet): 返回两个集合的并集

intersection(otherSet): 返回两个集合的交集

difference(otherSet): 返回两个集合的差集

subset(otherSet): 判断该集合是否为传入集合的子集

has方法:

说明:集合中元素是不重复的。所以在其它任何操作前,必须用has方法确认集合是否有某个元素。这儿使用了hasOwnProperty方法来检测。
实现:

</>复制代码

  1. /**
  2. * 检测集合内是否有某个元素
  3. * @param {Any} value 要检测的元素
  4. * @return {Boolean} 如果有,返回true
  5. */
  6. this.has = function(value) {
  7. // hasOwnProperty的问题在于
  8. // 它是一个方法,所以可能会被覆写
  9. return items.hasOwnProperty(value)
  10. };
add方法:

说明: 给集合内添加某个元素。
实现:

</>复制代码

  1. /**
  2. * 给集合内添加某个元素
  3. * @param {Any} value 要被添加的元素
  4. * @return {Boolean} 添加成功返回True
  5. */
  6. this.add = function(value) {
  7. //先检测元素是否存在。
  8. if (!this.has(value)) {
  9. items[value] = value;
  10. return true;
  11. }
  12. //如果元素已存在则返回false
  13. return false;
  14. };
remove方法:

说明: 移除集合中某个元素
实现:

</>复制代码

  1. /**
  2. * 移除集合中某个元素
  3. * @param {Any} value 要移除的元素
  4. * @return {Boolean} 移除成功返回True
  5. */
  6. this.remove = function(value) {
  7. //先检测元素是否存在。
  8. if (this.has(value)) {
  9. delete items[value];
  10. return true;
  11. }
  12. //如果元素不存在,则删除失败返回false
  13. return false;
  14. };
clear方法:

说明: 清空集合
实现:

</>复制代码

  1. /**
  2. * 清空集合
  3. */
  4. this.clear = function() {
  5. this.items = {};
  6. };
size方法

说明: 返回集合长度,这儿有两种方法。第一种方法使用了Object.keys这个Api,但只支持IE9及以上。第二种则适用于所有浏览器。
实现:

</>复制代码

  1. /**
  2. * 返回集合长度,只可用于IE9及以上
  3. * @return {Number} 集合长度
  4. */
  5. this.size = function() {
  6. // Object.keys方法能将对象转化为数组
  7. // 只可用于IE9及以上,但很方便
  8. return Object.keys(items).length;
  9. }
  10. /**
  11. * 返回集合长度,可用于所有浏览器
  12. * @return {Number} 集合长度
  13. */
  14. this.sizeLegacy = function() {
  15. var count = 0;
  16. for (var prop in items) {
  17. if (items.hasOwnProperty(prop)) {
  18. ++count;
  19. }
  20. }
  21. return count;
  22. }
values方法

说明: 返回集合转换的数组,这儿也有两种方法。理由同上。使用了Object.keys,只能支持IE9及以上。
实现:

</>复制代码

  1. /**
  2. * 返回集合转换的数组,只可用于IE9及以上
  3. * @return {Array} 转换后的数组
  4. */
  5. this.values = function() {
  6. return Object.keys(items);
  7. };
  8. /**
  9. * 返回集合转换的数组,可用于所有浏览器
  10. * @return {Array} 转换后的数组
  11. */
  12. this.valuesLegacy = function() {
  13. var keys = [];
  14. for (var key in items) {
  15. keys.push(key)
  16. };
  17. return keys;
  18. };
union方法

说明: 返回两个集合的并集
实现:

</>复制代码

  1. /**
  2. * 返回两个集合的并集
  3. * @param {Set} otherSet 要进行并集操作的集合
  4. * @return {Set} 两个集合的并集
  5. */
  6. this.union = function(otherSet) {
  7. //初始化一个新集合,用于表示并集。
  8. var unionSet = new Set();
  9. //将当前集合转换为数组,并依次添加进unionSet
  10. var values = this.values();
  11. for (var i = 0; i < values.length; i++) {
  12. unionSet.add(values[i]);
  13. }
  14. //将其它集合转换为数组,依次添加进unionSet。
  15. //循环中的add方法保证了不会有重复元素的出现
  16. values = otherSet.values();
  17. for (var i = 0; i < values.length; i++) {
  18. unionSet.add(values[i]);
  19. }
  20. return unionSet;
  21. };
intersection方法

说明: 返回两个集合的交集
实现:

</>复制代码

  1. /**
  2. * 返回两个集合的交集
  3. * @param {Set} otherSet 要进行交集操作的集合
  4. * @return {Set} 两个集合的交集
  5. */
  6. this.intersection = function(otherSet) {
  7. //初始化一个新集合,用于表示交集。
  8. var interSectionSet = new Set();
  9. //将当前集合转换为数组
  10. var values = this.values();
  11. //遍历数组,如果另外一个集合也有该元素,则interSectionSet加入该元素。
  12. for (var i = 0; i < values.length; i++) {
  13. if (otherSet.has(values[i])) {
  14. interSectionSet.add(values[i])
  15. }
  16. }
  17. return interSectionSet;
  18. };
difference方法

说明: 返回两个集合的差集
实现:

</>复制代码

  1. /**
  2. * 返回两个集合的差集
  3. * @param {Set} otherSet 要进行差集操作的集合
  4. * @return {Set} 两个集合的差集
  5. */
  6. this.difference = function(otherSet) {
  7. //初始化一个新集合,用于表示差集。
  8. var differenceSet = new Set();
  9. //将当前集合转换为数组
  10. var values = this.values();
  11. //遍历数组,如果另外一个集合没有该元素,则differenceSet加入该元素。
  12. for (var i = 0; i < values.length; i++) {
  13. if (!otherSet.has(values[i])) {
  14. differenceSet.add(values[i])
  15. }
  16. }
  17. return differenceSet;
  18. };
subset方法

说明: 判断该集合是否为传入集合的子集。这段代码在我自己写完后与书上一比对,觉得自己超级low。我写的要遍历数组三次,书上的只需要一次,算法复杂度远远低于我的。
实现:

</>复制代码

  1. /**
  2. * 判断该集合是否为传入集合的子集
  3. * @param {Set} otherSet 传入的集合
  4. * @return {Boolean} 是则返回True
  5. */
  6. this.subset = function(otherSet) {
  7. // 第一个判定,如果该集合长度大于otherSet的长度
  8. // 则直接返回false
  9. if (this.size() > otherSet.size()) {
  10. return false;
  11. } else {
  12. // 将当前集合转换为数组
  13. var values = this.values();
  14. for (var i = 0; i < values.length; i++) {
  15. if (!otherSet.has(values[i])) {
  16. // 第二个判定。只要有一个元素不在otherSet中
  17. // 那么则可以直接判定不是子集,返回false
  18. return false;
  19. }
  20. }
  21. return true;
  22. }
  23. };
源代码

源代码在此~

</>复制代码

  1. 集合-源代码

ES6中的集合

ES6也提供了集合,但之前看ES6的集合操作一直迷迷糊糊的。实现一遍后再去看,感觉概念清晰了很多。
具体的我掌握的不是很好,还在学习中,就不写出来啦~推荐看阮一峰老师的《ECMAScript 6入门》中对ES6 Set的介绍。

</>复制代码

  1. 《ECMAScript 6入门》-- SetMap数据结构

感想

到了这儿,已经掌握了一些基本的数据结构。剩下的都是难啃的骨头了(对我而言)。

字典的散列表、图、树、排序算法。算是四大金刚,所以近期关于数据结构与算法系列的文章,可能会更新的很慢。对我来说,也算是一个坎。希望这个寒假,能跨过这个坎。

前端路漫漫,且行且歌~

</>复制代码

  1. Lxxyx的前端乐园
    原文链接:寒假前端学习(5)——学习JavaScript数据结构与算法(三):集合

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/91590.html

相关文章

  • CSS技巧

    摘要:技巧使你的更加专业这是上关于技巧的一篇译文,另外你也可以在本项目看到原文。列举了一些很实用的技巧,比如给空内容的标签添加内容,逗号分隔列表等等。排序算法看源码,把它背下来吧排序算法的封装。主要帮助初学者更好的掌握排序算法的实现。 成为专业程序员路上用到的各种优秀资料、神器及框架 成为一名专业程序员的道路上,需要坚持练习、学习与积累,技术方面既要有一定的广度,更要有自己的深度。 Java...

    DangoSky 评论0 收藏0
  • CSS技巧

    摘要:技巧使你的更加专业这是上关于技巧的一篇译文,另外你也可以在本项目看到原文。列举了一些很实用的技巧,比如给空内容的标签添加内容,逗号分隔列表等等。排序算法看源码,把它背下来吧排序算法的封装。主要帮助初学者更好的掌握排序算法的实现。 成为专业程序员路上用到的各种优秀资料、神器及框架 成为一名专业程序员的道路上,需要坚持练习、学习与积累,技术方面既要有一定的广度,更要有自己的深度。 Java...

    zgbgx 评论0 收藏0
  • CSS技巧 - 收藏集 - 掘金

    摘要:笔者作为一位,将工作以来用到的各种优秀资料神器及框架整理在此,毕竟好记性不如烂键盘,此前端知识点大百科全书前端掘金,,不定期更新技巧前端掘金技巧,偶尔更新。计算数组的极值技巧使你的更加专业前端掘金一个帮你提升技巧的收藏集。 CSS 样式画各种图形 - 前端 - 掘金下面是一些我在 CSS 中经常用到的图案,还有一些是在css-tricks看到的。记录一下,以后会用到。会持续更新… 一、...

    Jonathan Shieber 评论0 收藏0
  • CSS技巧 - 收藏集 - 掘金

    摘要:笔者作为一位,将工作以来用到的各种优秀资料神器及框架整理在此,毕竟好记性不如烂键盘,此前端知识点大百科全书前端掘金,,不定期更新技巧前端掘金技巧,偶尔更新。计算数组的极值技巧使你的更加专业前端掘金一个帮你提升技巧的收藏集。 CSS 样式画各种图形 - 前端 - 掘金下面是一些我在 CSS 中经常用到的图案,还有一些是在css-tricks看到的。记录一下,以后会用到。会持续更新… 一、...

    SHERlocked93 评论0 收藏0
  • 学习数据结构算法集合

    本系列所有文章:第一篇文章:学习数据结构与算法之栈与队列第二篇文章:学习数据结构与算法之链表第三篇文章:学习数据结构与算法之集合第四篇文章:学习数据结构与算法之字典和散列表第五篇文章:学习数据结构与算法之二叉搜索树 集合简介 记得高一数学第一节课学的就是集合,现在快大四了再看到它有种见了老朋友的感觉。哈哈,闲话不多扯,进入正题。 集合是由一组无序且不重复的项组成的数据结构。这里集合的概念和高中数学...

    fai1017 评论0 收藏0

发表评论

0条评论

BDEEFE

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<