//集合类Set( ES6标准才有的类,目前兼容性较差)
//自定义集合类:
extend = function (o,p){ //定义一个复制对象属性的类函数
for(var x in p){ o[x] = p[x]; }
return o;
};
function Set(){
this.values = {};
this.n = 0;
this.add.apply(this,arguments); //把所有参数都添加进这个集合
}
Set.prototype.add = function(){ //定义添加方法
for(var i = 0;i<arguments.length;i++){
var val = arguments[i];
var str = Set._v2s(val);
if(!this.values.hasOwnProperty(str)){ //判断是否在集合中
this.values[str] = val; //按字符串和值对添加进集合
this.n++; //集合中的值+1
}
}
return this; //支持链式调用
};
Set.prototype.remove = function(){ //定义删除方法
for(var i = 0;i<arguments.length;i++){
var str = Set._v2s(arguments[i]);
if(this.values.hasOwnProperty(str)){
delete this.values[str];
this.n--;
}
}return this;
};
Set.prototype.contains = function(value){
return this.values.hasOwnProperty(Set._v2s(value));
};
Set.prototype.size = function(){ return this.n;};
//遍历集合中的所有元素,在指定的上下文context中对每个元素执行函数f
Set.prototype.foreach = function(f,context){
for(var s in this.values){
if(this.values.hasOwnProperty(s)){
f.call(context,this.values[s]);
}
}
}; //作用:对任意对象都返回一个字符串,针对不同对象返回不同的字符串,对于同一对象返回相同字符串
Set._v2s = function(val){
switch (val){
case undefined:return 'u';
case null: return 'n';
case true: return 't';
case false: return 'f';
default :switch (typeof val){
case 'number': return '#'+val;
case 'string': return '"'+val;
default :return '@'+objectId(val);
}
}
//给对象o创建一个不可枚举且只读的属性,用来存放id
function objectId(o){
var prop = '|**objectid**|';
if(!o.hasOwnProperty(prop)) o[prop] = Set._v2s.next++;
return o[prop];
}
};
Set._v2s.next = 100;
//==类工厂,子类,父类===
function fromSubclass(superclass,f){
var c = function(){
superclass.apply(this,arguments); //调用父类的构造函数
};
var proto = c.prototype =Object.create(superclass.prototype);
proto.constructor = c;
proto.add = function(){
for(var i=0;i<arguments.length;i++){
var v=arguments[i];
if(!f(v)) throw ('value'+v+'!');
}
superclass.prototype.add.apply(this,arguments); //调用父类的原型方法
};
return c;
}
var f1 = function(x){return typeof x === 'string';};
var f2 = function(x){return typeof x === 'function';};
var StringSet = fromSubclass(Set,f1); //可以很方便的根据不同的父类来创建子类,只需改一下参数即可
var MySet = fromSubclass(Set,f2);
//===定义并立即调用
var NonNullSet = (function(){
var superclass = Set; //只需改此值,就可以继承自不同的父类
return extend(
function(){ superclass.apply(this,arguments)}, //构造函数指代类本身
{
add:function () {
for (var i = 0; i < arguments.length; i++) {
if (arguments[i] == null) {
throw new Error('can not add null or undefined');
}
}
return superclass.prototype.add.apply(this, arguments);
}
}
);
}());
//======='组合优于继承': 使用组合代替继承========================================
//只需创建一个单独的ChildClass子类,利用这个类的实例来创建继承自不同的superClass和执行不同的func方法
var ChildClass = extend(
function ChildClass(superClass,func){
this.set = superClass;
this.filter = func;
},
{
add:function(){
if(this.filter){
for(var i = 0;i<arguments.length;i++){
var v = arguments[i];
if(!this.filter(v)){
throw new Error('ChildClass:value '+v+'null');
}
}
this.set.add.apply((this.set,arguments));
return this;
}
},
remove:function(){
this.set.remove.apply(this.set,arguments);
return this;
},
contains:function(v){return this.set.contains(v);},
size:function(){return this.set.size();},
foreach:function(f,c){this.set.foreach(f,c);}
}
);
//==
var s = new ChildClass(new Set(),function(x){return x!== null;});
var t = new ChildClass(s,function(x){return !(x instanceof Set);});
//=======抽象类和非抽象类的层次结构========================================
// 抽象方法
function abstractmethod() { throw new Error("abstract method"); }
//抽象类
function AbstractSet() { throw new Error("Can't instantiate abstract classes");}
AbstractSet.prototype.contains = abstractmethod; //抽象方法contains var AbstractWritableSet = extend(
function() { throw new Error("Can't instantiate abstract classes"); },
{
add: abstractmethod,
remove: abstractmethod,
union: function(that) {
var self = this;
that.foreach(function(v) { self.add(v); });
return this;
},
intersection: function(that) {
var self = this;
this.foreach(function(v) { if (!that.contains(v)) self.remove(v);});
return this;
},
difference: function(that) {
var self = this;
that.foreach(function(v) { self.remove(v); });
return this;
}
});

JavaScript学习笔记-自定义集合类的更多相关文章

  1. JavaScript学习笔记- 自定义滚动条插件

    此滚动条仅支持竖向(Y轴) 一.Css /*这里是让用户鼠标在里面不能选中文字,避免拖动的时候出错*/ body { -moz-user-select: none; /*火狐*/ -webkit-us ...

  2. JavaScript学习笔记-自定义滚动条

    这是一个基本实现思路,如果有新手和我一样没什么事,喜欢瞎研究话,可以参考下. 一.Html <div class="scroll_con"> <div class ...

  3. JavaScript:学习笔记(2)——基本概念与数据类型

    JavaScript:学习笔记(2)——基本概念与数据类型 语法 1.区分大小写.Test 和 test 是完全不同的两个变量. 2.语句最好以分号结束,也就是说不以分号结束也可以. 变量 1.JS的 ...

  4. Java程序猿的JavaScript学习笔记(汇总文件夹)

    最终完结了,历时半个月. 内容包含: JavaScript面向对象特性分析,JavaScript高手必经之路. jQuery源代码级解析. jQuery EasyUI源代码级解析. Java程序猿的J ...

  5. Java程序猿的JavaScript学习笔记(8——jQuery选择器)

    计划按例如以下顺序完毕这篇笔记: Java程序猿的JavaScript学习笔记(1--理念) Java程序猿的JavaScript学习笔记(2--属性复制和继承) Java程序猿的JavaScript ...

  6. Java程序猿JavaScript学习笔记(2——复制和继承财产)

    计划和完成在这个例子中,音符的以下序列: Java程序猿的JavaScript学习笔记(1--理念) Java程序猿的JavaScript学习笔记(2--属性复制和继承) Java程序猿的JavaSc ...

  7. Java程序猿的JavaScript学习笔记(3——this/call/apply)

    计划按例如以下顺序完毕这篇笔记: Java程序猿的JavaScript学习笔记(1--理念) Java程序猿的JavaScript学习笔记(2--属性复制和继承) Java程序猿的JavaScript ...

  8. Java程序猿JavaScript学习笔记(4——关闭/getter/setter)

    计划和完成这个例子中,音符的顺序如下: Java程序猿的JavaScript学习笔记(1--理念) Java程序猿的JavaScript学习笔记(2--属性复制和继承) Java程序猿的JavaScr ...

  9. Java程序猿JavaScript学习笔记(14——扩大jQuery UI)

    计划和完成这个例子中,音符的顺序如下: Java程序猿的JavaScript学习笔记(1--理念) Java程序猿的JavaScript学习笔记(2--属性复制和继承) Java程序猿的JavaScr ...

随机推荐

  1. java中Object.equals()简单用法

    /* equals()方法默认的比较两个对象的引用! */ class Child { int num; public Child(int x){ num = x; } //人文的抛出运行时异常的好处 ...

  2. (转)Shell函数

    Shell函数类似于Shell脚本,里面存放了一系列的指令,不过Shell的函数存在于内存,而不是硬盘文件,所以速度很快,另外,Shell还能对函数进行预处理,所以函数的启动比脚本更快. 1.  函数 ...

  3. 学习网页制作中如何在正确选取和使用 CSS 单位

    在 CSS 测量系统中,有好几种单位,如像素.百分比.英寸.厘米等等,Web 开发人员很难了解哪些单位在何处使用,如何使用.很多人习惯了总是使用同一种单位,但这一决定可能会严重限制你的设计的执行. 这 ...

  4. node.js + mongodb 做项目的详解(一)

    想写博客很长时间了,因为一直身患懒癌,所以一直拖到了现在.markdown的语法也是刚刚学,试验一下效果好了不说了,直接上干货了.----------------------------------- ...

  5. [python]计算机使用过程中,眼睛强制休息

    前言 现在的电脑族们,在使用电脑的过程中,常常忘记了时间的流逝,要么忙碌在电视剧的观看中,要么忙碌在工作中,要么忙碌在游戏中,往往忽视了对眼睛的正常保护,让眼睛能够在空闲的时候获得足够的休息时间. 我 ...

  6. Visual Studio远程调试监视器(MSVSMON.EXE)的32位版本不能用于调试64位进程或64位转储

    在VS2013中调试Silverlight项目时,提示:无法附加.Visual Studio远程调试监视器(MSVSMON.EXE)的32位版本不能用于调试64位进程或64位转储.请改用64位版本. ...

  7. c#实现查询程序运行线程数

    class Program { static void Main(string[] args) { List<Thread> list = new List<Thread>() ...

  8. SQL Server代理(6/12):作业里的工作流——深入作业步骤

    SQL Server代理是所有实时数据库的核心.代理有很多不明显的用法,因此系统的知识,对于开发人员还是DBA都是有用的.这系列文章会通俗介绍它的很多用法. 如我们在这里系列的前几篇文章所见,SQL ...

  9. laravel5如何创建service provider和facade

    laravel5如何创建service provider和facade laravel5创建一个facade,可以将某个service注册个门面,这样,使用的时候就不需要麻烦地use 了.文章用一个例 ...

  10. WinPhone学习笔记(五)——LongListSelector

    LongListSelector也是WinPhone的特色控件之一,最初不了解这个控件叫啥名,知道它会在"人脉"里面出现,在应用程序列表也是这个LongListSelector(如 ...