一、javascript模拟重载

java中根据参数类型和数量的区别来实现重载,javascript弱类型,没有直接的机制实现重载,javascript中参数类型不确定和参数个数任意,通过判断实际传入的参数的个数来实现重载。

<script>
function Person() {
var args = arguments;
if (typeof args[0] === 'object' && args[0]) { //判断第一个参数是否为对象,进行相应操作。传参为空时typeof null也是object,所以要同时判断args[0]不为空
if (args[0].name) {
this.name = args[0].name;
}
if (args[0].age) {
this.age = args[0].age;
} } else {
if (args[0]) {
this.name = args[0];
}
if (args[1]) {
this.age = args[1];
}
}
}
//toString继承自Object,一般的对象toString会显示"[object Object]",没有什么意义,所以可以重新一下,方便输出
Person.prototype.toString = function() {
return 'name=' + this.name + ',age=' + this.age;
} var bosn = new Person('Bosn', 27);
bosn.toString(); //name=Bosn,age=27 var lxy = new Person({
name: 'lxy',
age: 25
});
lxy.toString(); //name=lxy,age=25
</script>

二、调用基类方法

1、调用基类的方法进行初始化

基类人Person

学生Student

初始化Student的时候,初始化了className后需要初始化Person部分,可以通过

Person.call(this,name);调用基类的方法和属性。

<script>
function Person(name){
this.name=name;
}
function Student(name,className){
this.className=className;
Person.call(this,name); //调用基类的构造器
} var bosn=new Student('Bosn','Network064');
console.log(bosn);
</script>

输出时能够正确的被初始化。

2、子类覆盖了基类的方法

Student的init覆盖了Person的init,初始化Student时需要初始化Person,通过Person.prototype.init.apply(this,arguments);

<script>
Person.prototype.init=function(){ }; Student.prototype.init=function(){
//do sth
Person.prototype.init.apply(this,arguments);
}
</script>

三、链式调用

类似jquery选择器链式调用。

关键要在ClassManager.prototype.addClass操作完成后return this

<script>
function ClassManager(){//构造器
} ClassManager.prototype.addClass=function(str){ //在构造器上挂载一个方法,表示addClass
console.log('Class:'+str+' added.');
return this;// this总是指向ClassManager的实例,所以return就实现了链式调用
} var manager=new ClassManager();
manager.addClass('classA').addClass('classB').addClass('classC');
//Class:classA added.
//Class:classB added.
//Class:classC added. </script>

四、抽象类

没有机制,可以进行模拟。主要是用throw new Error()

<script>
function DetectorBase(){
throw new Error('Abstract class can not be invoked directly!');
//在构造器里抛一个异常,防止抽象类被直接调用
}
DetectorBase.detect=function(){console.log('Detection starting...');}
DetectorBase.stop=function(){console.log('Detector stopped.');}
DetectorBase.init=function(){
throw new Error('Error');
//想要子类覆盖而不想被直接调用的方法,抛出异常
} function LinkDetector(){}
//通过Object.create实现继承
LinkDetector.prototype=Object.create(DetectorBase.prototype);
LinkDetector.prototype.constructor=LinkDetector;
</script>

五、definePrototype(ES5)

<script>
//ES5里面对于属性有一些getter setter方法,有一些标签
//configurable:能否delete
//enumberable:能否for in遍历
//writable:能否读写属性
//value:记录属性值,默认为undefine
//控制属性的访问情况 function Person(name){
//name可遍历,不可读写,不可删除
Object.defineProperty(this,'name',{value:name,enumerable:true});
}
//ARMS_NUM相当于定义了一个常量,可枚举,不可写,不可删除
Object.defineProperty(Person,'ARMS_NUM',{value:2,enumerable:true});
Object.seal(Person.prototype); //Person在初始化之后不想被扩展,不想被配置了,用seal方法
Object.seal(Person); function Student(name,className){
this.className=className;
Person.call(this,name);
}
Student.prototype=Object.create(Person.prototype);
Student.prototype.constructor=Student;
</script>

六、模块化

模块化工具或者说类库有有seajs,requirejs等。

模块化的一般形式是:一个定义了私有变量和函数的函数;利用闭包创建可以访问私有变量和函数的特权函数;最后返回这个特权函数,或者把它们保存到一个可访问到的地方。

1、通过返回一个对象实现简单模块化

<script>
var moduleA;//只需要把最核心的东西moduleA放到全局作用域
moduleA=function(){
var prop=1;
function func(){}
return{ //func和prop不会被泄露到全局作用域
func:func,
prop:prop
}
}() //moduleA用立即执行的匿名函数
</script>

2、通过返回this的方法实现简单模块化

通过new function(),通过设置this的值,我们知道通过new调用最后会返回this,除非你return的是一个对象。

<script>
//通过new默认返回this的方法
va moduleA;
moduleA=new function(){
var prop=1;
function func(){}
this.func=func;
this.prop=prop;
}
</script>

3、模块化应用【update20170307】

模块化可以摒弃全局变量的使用。促进了信息隐藏和其他优秀的设计实践。对于应用程序的封装,或者构造其他单例对象,模块模式非常有效。

模块模式也可以用来产生安全的对象。

例:构造一个用来产生序列号的对象。

<script>
var serial_maker=function(){
/*返回一个用来产生唯一字符串的对象。
唯一字符串由两部分组成:前缀+序列号。
该对象包含一个设置前缀的方法,一个设置序列号的方法和一个产生唯一字符串的gensym方法。
*/
var prefix='';
var seq=0;
return{
set_prefix:function(p){
prefix=String(p);
},
set_seq:function(s){
seq=s;
},
gensym:function(){
var result=prefix+seq;
seq+=1;
return result;
}
}
};
var seqer=serial_maker();
seqer.set_prefix('Q');
seqer.set_seq(1000);
var unique;
unique=seqer.gensym();
console.log(unique);//unique是"Q1000"
unique=seqer.gensym();
console.log(unique);//unique是"Q1001" </script>

seqer()的3个方法都没有用到this或者that,因此没有办法损害seqer。除非调用对应的方法,否则没法改变prefix或seq的值。

seqer对象是可变的,所以它的方法可能会被替换掉,但替换后的方法依然不能访问私有变量。

比如:

seqer.gensym=function(){
var result=prefix+seq;
seq+=3;
console.log(seq);
return result;
}

就会报错,因为替换后的方法不能访问私有变量。

seqer就是一组函数的集合,而且那些函数被授予特权,拥有使用或修改私有状态的能力。

如果我们把seqer.gensym作为一个值传递给第三方函数,那个函数能用它产生唯一字符串,却不能通过修改它来改变prefix或seq的值。

参考:

js模块化历程

http://www.cnblogs.com/lvdabao/p/js-modules-develop.html

写了十年JS却不知道模块化为何物?

https://blog.wilddog.com/?p=587

深入浅出ES6(十三):类 Class

http://www.infoq.com/cn/articles/es6-in-depth-classes/

本文作者starof,因知识本身在变化,作者也在不断学习成长,文章内容也不定时更新,为避免误导读者,方便追根溯源,请诸位转载注明出处:http://www.cnblogs.com/starof/p/4904929.html有问题欢迎与我讨论,共同进步。

javascript OOP(下)(九)的更多相关文章

  1. Javascript多线程引擎(九)

    Javascript多线程引擎(九)--垃圾回收 垃圾回收这个话题对Programer来说是非常老旧的话题, 从手动的malloc/free 到半自动的 引用计数 再到全自动的 mark-sweep ...

  2. 使用JavaScript OOP特性搭建Web应用

    最近,我面试了一个有五年 Web 应用程序开发经验的软件开发人员.四年半来她一直在从事 JavaScript 相关的工作,她自认为 JavaScript 技能非常好,但在不久之后我就发现实际上她对 J ...

  3. Arcgis for Javascript API下类似于百度搜索A、B、C、D marker的实现方式

    原文:Arcgis for Javascript API下类似于百度搜索A.B.C.D marker的实现方式 多说无益,首先贴两张图让大家看看具体的效果: 图1.百度地图搜索结果 图2.Arcgis ...

  4. (JavaScript插件——下拉菜单)

    前言 阅读之前您也可以到Bootstrap3.0入门学习系列导航中进行查看http://www.cnblogs.com/aehyok/p/3404867.html 本文主要来学习一下JavaScrip ...

  5. JavaScript OOP 之 this指向

    今天给大家分享一个JavaScript OOP中关于分辨this指向对象的小技巧,很实用呦! 我们先来看一段代码: 大家能判断出func();和obj.func();这两句的this指向吗? 首先,我 ...

  6. JavaScript基础--内部类(九)

    js的内部类javascript 中本身提供一些,可以直接使用的类,这种类就是内部类,主要有:ObjectArrayMathBooleanStringRegExpDateNumber 1.内部类分类从 ...

  7. JavaScript OOP(一)之构造函数与new命令

    面向对象编程:Object Oriented Programming,简称OOP. 典型的oop语言,如hava.c++,存在着类的概念,类就是对象的模板 (类可以类比为人类:而实例化类后变为对象,对 ...

  8. JavaScript OOP 之「创建对象」

    工厂模式 工厂模式是软件工程领域一种广为人知的设计模式,这种模式抽象了创建具体对象的过程.工厂模式虽然解决了创建多个相似对象的问题,但却没有解决对象识别的问题. function createPers ...

  9. 玩转JavaScript OOP[4]——实现继承的12种套路

    概述 在之前的文章中,我们借助构造函数实现了"类",然后结合原型对象实现了"继承",并了解了JavaScript中原型链的概念. 理解这些内容,有助于我们更深入 ...

随机推荐

  1. C++ 读取字符串中的数字

    今天真是试了各种方法,笨方法聪明方法都有了 方法1:一个字符一个字符的读取 方法2:借助strtok实现split 适用于char 方法3:借助istringstream实现split 适用于stri ...

  2. Python os.access() 方法

    概述 os.access() 方法使用当前的uid/gid尝试访问路径.大部分操作使用有效的 uid/gid, 因此运行环境可以在 suid/sgid 环境尝试. 语法 access()方法语法格式如 ...

  3. 【转】C++对成员访问运算符->的重载

    运算符->的重载比较特别,它只能是非静态的成员函数形式,而且没有参数. 1.如果返回值是一个原始指针,那么就将运算符的右操作数当作这个原始指针所指向类型的成员进行访问: 2.如果返回值是另一个类 ...

  4. nginx中Geoip_module模块的使用

    nginx中Geoip_module模块的使用 .安装模块,nginx也是通过yum安装 yum install nginx-module-geoip -y # 可以看到模块的链接库文件 [root@ ...

  5. Linux内存使用调整

    前段时间在做播放器的时候,遇到个问题,花了很长时间,做个记录,希望对有需要的人有所帮助: 播放器的播视频的时候,无论是手动切换视频还是到视频播放完成,自动切换视频,一定次数后均出现黑屏现象,偶尔有声音 ...

  6. Find Lines

    (Uva 6955可以直接随机,湖大OJ 13348 要优化) 题意:给出 n个点的坐标, 一个 百分数p, 求是否有一条直线上有 n * p /100个点… 随机化算法,但也要优化下……(TLE, ...

  7. 洛谷P3676 小清新数据结构题 [动态点分治]

    传送门 思路 这思路好妙啊! 首先很多人都会想到推式子之后树链剖分+线段树,但这样不够优美,不喜欢. 脑洞大开想到这样一个式子: \[ \sum_{x} sum_x(All-sum_x) \] 其中\ ...

  8. ACM-ICPC 2018 焦作赛区网络预赛 I Save the Room

    Bob is a sorcerer. He lives in a cuboid room which has a length of AAA, a width of BBB and a height ...

  9. 一个完整Java Web项目背后的密码

    前言 最近自己做了几个Java Web项目,有公司的商业项目,也有个人做着玩的小项目,写篇文章记录总结一下收获,列举出在做项目的整个过程中,所需要用到的技能和知识点,带给还没有真正接触过完整Java ...

  10. RefineDet训练自己的数据

    https://github.com/sfzhang15/RefineDet 1.编译安装 cp Makefile.config.example Makefile.config make all -j ...