首页 > 编程笔记 > JavaScript笔记 > JS面向对象编程 阅读:62

JS模块化编程简介

模块就是提供一个接口,却隐藏状态与实现的函数或对象。一般在开发中使用闭包函数来构建模块,摒弃全局变量的滥用,规避 JavaScript 缺陷。

全局变量是 JavaScript 最糟糕的特性之一,在一个大型 Web 应用中,全局变量简直就是一个魔鬼,可能带来无穷的灾难。

示例1

本示例为 String 扩展一个 toHTML 原型方法,该方法能够把字符串中的 HTML 转义字符替换为对应的字符串。
//为Function增加method原型方法
Function.prototype.method = typeof Function.prototype.method === "function" ?
    Function.prototype.method :  //先检测是否已经存在该方法,否则定义函数
    function (name, func) {
        if (!this.prototype[name]){  //检测当前类型中是否存在指定名称的原型
            this.prototype[name] = func;  //绑定原型方法
        }
        return this;  //返回类型
    };
String.method('toHTML', function () {  //为String增加toHTML原型方法
    var entity = {  //过滤的转义字符实体
        quot : '""',
        lt : '<',
        gt :'>'
    };
    return function () {  //返回方法的函数体
        return this.replace(/&([^&;]+);/g, function (a, b) {  //匹配字符串中HTML转义字符
            var r = entity[b];  //映射转义字符实体
            return typeof r === 'string' ? r : a;  //替换并返回
        });
    };
}());  //生成闭包体
在上面代码中,为 String 类型扩展了一个 toHTML 原型方法,它调用 String 对象的 replace 方法来查找以&开头和以;结束的字符串。如果这些字符可以在转义字符实体表 entity 中找到,那么就将该字符实体替换为映射表中的值。toHTML 方法用到了一个正则表达式。 return this.replace(/&([^&;]+);/g, function (a, b) { var r = entity[b]; return typeof r === 'string' ? r : a; }); 在最后一行使用()运算符立刻调用刚刚构造出来的函数。这个调用所创建并返回的函数才是 toHTML 方法。
console.log('&lt;quot;&gt;');  //&lt;quot;&gt;
console.log('&lt;quot;&gt;'.toHTML());  //<''>
模块利用函数作用域和闭包来创建绑定对象与私有属性的关联。在这个示例中,只有 toHTML 方法才有权访问字符实体表 entity 这个数据对象。

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

使用模块可以避免全局变量的滥用,从而保护信息的安全性,实现优秀的设计实践。使用这种模式也可以实现应用程序的封装,或者构建其它框架。

模块模式通常结合实例模式使用。JavaScript 的实例就是对象字面量表示法创建的,对象的属性值可以是数值或函数,并且属性值在该对象的生命周期中不会发生变化。模块通常作为工具为程序其他部分提供功能支持。通过这种方式能够构建比较安全的对象。

示例2

下面示例设计一个能够自动生成序列号的对象。toSerial() 函数返回一个能够产生唯一序列字符串的对象。这个字符串由两部分组成:字符前缀+序列号。这两部分可以分别使用 setPrefix 和 setSerial 方法进行设置,然后调用实例对象的 get 方法来读取这个字符串。没执行该方法,都会自动产生唯一一个序列字符串。
var toSerial = function () {  //包装函数
    var prefix = '';  //私有变量,前缀字符,默认为空字符
    var serial = 0;  //私有变量,序列号,默认为0
    return {  //返回一个对象直接量
        setPrefix : function (p) {  //设置前缀字符
            prefix = String (p);  //强制转换为字符串
        },
        setSerial : function (s) {  //设置序列号
            serial = typeof s == "number" ? s : 0;  //如果参数不是数字,则设置为0
        },
        get : function () {  //读取自动生成的序列号
            var result = prefix + serial;
            serial += 1;  //递加序列号
            return result;  //返回结果
        }
    };
};
var serial = toSerial ();  //获取生成序列号对象
serial.setPrefix ('NO.');  //设置前缀字符串
serial.setSerial (100);  //设置起始序号
console.log(serial.get());  //“No.100”
console.log(serial.get());  //“No.101”
console.log(serial.get());  //“No.102”
serial 对象包含的方法都没有使用 this 或 that,因此没有办法损害 serial,除非调用对应的方法,否则不能改变 prefix 或 serial 的值。serial 对象是可变的,所以它的方法可能会被替换掉,但是替换后的方法依然不能访问私有成员。如果把 serial.get 作为一个值传递给第三方函数,那么这个函数只能通过它产生唯一字符串,不能通过它来改变 prefix 或 serial 的值。

所有教程

优秀文章