类 Class

类的概念应该是面向对象语言的一个特色,但是JavaScript并不像Java,C++等高级语言那样拥有正式的类,而是多数通过构造器以及原型方式来仿造实现。在讨论构造器和原型方法前,我可以看看一种叫做“工厂模式”的仿造方法。

function start() {
alert("Bang!!");
} function createCar(color, title) {
var car = {};
car.color = color;
car.title = title;
car.start = start;
return car;
} var car1 = createCar("red", "BMW");
var car2 = createCar("yellow", "BYD");

这种方式显然可以实现class的功能,但是外形上怎么也无法说它是个class以及class实例的创建过程。因此,出现了“构造函数模式”,它的关键在于构造器(Constructor)概念的引入。

构造器 Constructor

我们先来看“构造函数模式”的具体做法:

function start() {
alert("Bang!!");
} function Car(color, title) {
this.color = color;
this.title = title;
this.start = start;
} var car1 = new Car("red", "BMW");
var car2 = new Car("yellow", "BYD");

这个看起来有点类的样子了吧(先不提那个难看的外置function)?我们发现,那个constructor(Car)其实就是一个简单的function,它与“工厂方式”中的createCar()区别就在于:

  1、方法名开头字母大写(注:构造函数方法名开头字母大写是国际惯例);

  2、没有了空对象的创建和返回;

    3、使用this做引用。

那原来的那个空对象的创建以及返回的步骤去哪了呢?这两个步骤,现在都由创建实例时的“new”实现了。“new”这个操作符负责创建一个空对象,然后将那个叫做构造器的function添加到实例对象中并触发它,这样这个function实际上就是这个对象的一个method,function中的this指向的便是这个对象,最后将这个对象返回。根据如上分析,我们可以把这个过程简单分解为如下代码:

var obj = {};
obj.constructor = Car;
obj.constructor("red", "BMW");
return obj;

“构造函数模式”方式虽然与高级面向对象语言中的类创建方式已经很接近(使用”new“创建),但是貌似那个游离在类之外的function start()其实却是个相当有碍观瞻的瑕疵。我们应该想一种办法让这个方法与类挂钩,让它成为类的一个属性,不是全局的。于是,这就产生了“构造函数+原型法”的类构造方法。

原型 prototype

在”构造函数+原型法“中,我们对于类的method期待得到的效果是:

  1、仅是类的method而不是全局的

  2、只在类被定义时创建一个method实例,然后被所有类的实例共用。

由这两个目标,我们很容易想到高级面向对象语言Java的private static变量的特点。JavaScript没有为我们提供这么简单的符号来实现这个复杂功能,但是却有一个属性可以帮我们仿造出这种效果:prototype。我们先来看几段prototype的使用代码。

function Car() {
} Car.prototype.material = "steel";
var car1 = new Car();
var car2 = new Car();
console.log(car1.material);//steel
console.log(car2.material);//steel car1.material = "iron";
console.log(car1.material);//iron
console.log(car2.material);//steel
console.log(Car.prototype.material); //steel Car.prototype.material = "wood";
var car2 = new Car();
console.log(car1.material); //iron
console.log(car2.material); //wood
console.log(Car.prototype.material); //wood

分析该段代码前,需要明确两个概念:对象的直属属性和继承属性。直接在构造函数中通过this.someproperty = xxx这种形式定义的someproperty属性叫做对象的直属属性,而通过如行代码那样Car.prototype.material = "steel";这种形式定义的material属性叫做继承属性。由上面这段代码,我们可以总结出prototype属性的如下特点:

prototype属性的特点
    1、prototype是function下的属性(其实任意object都拥有该属性,function是对象的一种);
    2、prototype属性的值是一个对象,因此可任意添加子属性;
    3、类的实例可以直接通过”.”来直接获取prototype下的任意子属性;
    4、所有以此function作为构造函数创建的类实例共用prototype中的属性及值;
    5、类的实例没有prototype属性;
    6、可以直接通过 “实例.属性 = xxx” 的方式修改继承属性,修改后的值将覆盖继承自prototype的属性,但此修改不影响prototype本身,也不影响其它类实例;
    7、继承属性修改后,该属性就成为类实例的直属属性;
    8、可以直接修改prototype的属性值,此改变将作用于此类下的所有实例,但无法改变直属属性值。

【注:对象实例在读取某属性时,如果在本身的直属属性中没有查找到该属性,那么就会去查找function下的prototype的属性。所以类的实例可以直接通过”.”来直接获取prototype下的任意子属性】

好好理解下prototype的这些特点,我们不难看出,在prototype中定义的属性与Java类中的static属性特点极为相近,适合定义那些所有类实例都可共用的一些属性的值,比如汽车的构造材料这一类。

了解了prototype的这些特点,我们可以言归正传,把它应用到类方法的仿造和实现上了。其实做法很简单,就是把value换成function,先看下面的代码:

function Car() {
} Car.prototype.start = function() {
alert("Bang!!");
}; var car1 = new Car();
var car2 = new Car();
car1.start();
car2.start();

得益于prototype的强大特性,我们知道这个method确实是可以这么被绑定到类上的。但是,最有用的是,那个method(start)只会在执行Car.prototype.start =这个定义的时候创建一个实例,之后通过new创建的对象实例里面使用的这个method都是它的引用,因此不会有内存浪费。

好了,method的问题终于解决了,那么整理一下,下面写出使用“构造函数模式+原型模式”构造一个类的完整代码:

function Car(color, title) {
this.color = color;
this.title = title;
} Car.prototype.material = "steel";
Car.prototype.start = function () {
alert("Bang!!!");
};
Car.prototype.repaint = function (color) {
this.color = color;
}; var car1 = new Car("red", "BMW");
var car2 = new Car("yellow", "VW"); car1.repaint("pink");
console.log(car1.color); //pink
console.log(car2.color); //yellow

怎么样,这个跟高级面向对象语言中的class的样子更~加类似了吧?Perfect ?NO!!!上述写法只是在“语义”上达到了对类属性和方法的封装,很多面向对象思想的完美主义者希望在“视觉”上也达到封装,因此就产生了“动态原型法”,请看下面的代码:

function Car(color, title) {
this.color = color;
this.title = title; if (typeof Car._initialized == "undefined") {
Car.prototype.start = function() {
alert("Bang!!!");
};
Car.prototype.material = "steel";
Car._initializde = true;
}
}

我们看,其实Car.prototype的属性定义是可以被放进Car function的定义之中的,这样就达到了“视觉”封装。但是我们没有单纯的move,我们需要加一个条件,让这些赋值操作只执行一次,而不是每次创建对象实例的时候都执行,造成内存空间的浪费。添加_initialized 属性的目的就在于此。

注意:

对于类的属性值,我们上面只考虑了primitive type(原始类型)和function的类型情况,如果属性值是对象,比如Array,那情况就又复杂了,思考下面的一段代码。

function Car() {
} Car.prorotype.drivers = ["Kitten", "Bear"]; var car1 = new Car();
var car2 = new Car(); car1.drivers.push("Point"); console.log(car1.drivers); //Kitten,Bear,Point
console.log(car2.drivers); //Kitten,Bear,Point

可以看到,我们只是想对car1drivers做改变,但是car2的居然也改了,这显然是我们不希望出现的结果。为什么会这样呢?因为Car.prototype.drivers只是引用而已。在使用new关键字创建类实例时,附加给类实例的属性的属性值也是跟它指向相同值的引用。如果它所指向的数据是primitive type(原始类型)或function自然没有问题,因为没有什么操作可以直接修改这些值,我们只能修改引用本身,让它们指向新的值,但这种改变根本不会影响其它指向这个值的引用。但如果指向的是数组就不一样了,在任何一个指向它的引用上实施.push操作,都会影响这个Array的值,并且这种改变会对所有引用都生效。因此,像Array这种复杂数据类型是不应该放在prototype里面做属性值的,这样,上述代码就应该更改为:

function Car() {
this.drivers = ["Kitten", "Bear"];
} var car1 = new Car();
var car2 = new Car(); car1.drivers.push("Point"); console.log(car1.drivers); //Kitten,Bear,Point
console.log(car2.drivers); //Kitten,Bear

下面我们来创建一个完成的类:

function Car() {
this.drivers = ["Kitten", "Bear"];
if (typeof Car._initialized == "undefined") {
Car.prototype.getDiversNumber = function () {
alert(this.drivers.length);
};
Car._initialized = true;
}
} var car1 = new Car();
var car2 = new Car(); car1.drivers.push("Point"); console.log(car1.drivers); //Kitten,Bear,Point
console.log(car2.drivers); //Kitten,Bear
car1.getDiversNumber(); //
car2.getDiversNumber(); //

总结出来一句话就是:用构造函数模式定义对象的所有非函数属性,用原型模式定义对象的函数属性。

类(class)、构造函数(constructor)、原型(prototype)的更多相关文章

  1. JavaScript 类、构造函数、原型

    类.构造函数.原型  :本质均为函数 利用的原理是:词法作用域,调用对象及作用域链  闭包  属性查找方式    设计和new运算符一起使用的函数叫做构造函数. 构造函数的工作:初始化一个新创建的对象 ...

  2. 关于构造函数和原型prototype对象的理解

    构造函数     1.什么是构造函数 构造函数,主要用于对象创建的初始化,和new运算符一起用于创建对象,一个类可以有多个构造函数,因为函数名相同,所以只能通过参数的个数和类型不同进行区分,即构造函数 ...

  3. JavaScript中的原型prototype和__proto__的区别及原型链概念

    问题 初学js的同学,总是搞不清楚js中的原型是什么东西,看着控制台打印出来的一串串__proto__,迷惑不已. 例如我定义一个Person,创建一个实例p,并打印实例. function Pers ...

  4. Javascript面向对象——创建对象、构造函数的原型

    Javascript面向对象--创建对象.构造函数的原型 其实:JavaScript中的创建类,就是创建一个构造函数,在创建对象时用到new这个关键字, 一.创建对象 1.函数创建对象 functio ...

  5. JavaScript中的类(class)、构造函数(constructor)、原型(prototype)

    类 Class 类的概念应该是面向对象语言的一个特色,但是JavaScript并不像Java,C++等高级语言那样拥有正式的类,而是多数通过构造器以及原型方式来仿造实现.在讨论构造器和原型方法前,我可 ...

  6. JS中关于构造函数、原型链、prototype、constructor、instanceof、__proto__属性

    在Javascript不存在类(Class)的概念,javascript中不是基于类的,而是通过构造函数(constructor)和原型链(prototype chains)实现的.但是在ES6中引入 ...

  7. Javascript函数、构造函数、原型、类和对象

    函数 函数是JavaScript中特殊的对象,对函数执行typeof运算会返回字符串"function",因为函数也是对象,他们可以拥有属性和方法. 静态方法 函数在JS中定义了类 ...

  8. <JavaScript>constructor、prototype、__proto__和原型链

    在看了网上很多相关的文章,很多都是懵逼看完,并不是说各位前辈们写得不好,而是说实在不容易在一两次阅读中理解透.我在阅读了一些文章后,自己整理总结和绘制了一些相关的图,个人认为会更容易接受和理解,所以分 ...

  9. js最好的继承机制:用对象冒充继承构造函数的属性,用原型prototype继承对象的方法。

    js最好的继承机制:用对象冒充继承构造函数的属性,用原型prototype继承对象的方法. function ClassA(sColor) { this.color = sColor; } Class ...

随机推荐

  1. 关于正则表达式处理textarea里的换行

    将textarea里的内容存入数据库时,会自动将回车换行符过滤成空格,也会将多个空格转换成一个空格,即:将\n等换成 “  ”存入数据库 因此为了将内容从数据库中按照原来格式读出写入到html 就必须 ...

  2. 100 open source Big Data architecture papers for data professionals

    zhuan :https://www.linkedin.com/pulse/100-open-source-big-data-architecture-papers-anil-madan Big Da ...

  3. Android Studio导入Project的方法

    Android Studio到现在已经发展到0.8+的版本了,最近也在试着使用它,原因是多方面的,一个毕竟是未来的趋势,二则是github上越来越多的大牛开源项目都是基于Android Studio的 ...

  4. AndroidStudio导入项目常见问题

    问题一: 解决:少了依赖的路径 问题二: 解决:把runProguard 改成minfyEnable 来源:http://blog.csdn.net/pengkv/article/details/44 ...

  5. 二十四种设计模式:中介者模式(Mediator Pattern)

    中介者模式(Mediator Pattern) 介绍用一个中介对象来封装一系列的对象交互.中介者使各对象不需要显式地相互引用,从而使其耦合松散,而且可以独立地改变它们之间的交互. 示例有一个Messa ...

  6. OpenJudge计算概论-Tomorrow never knows【输入日期计算下一天的日期】

    /*====================================================================== Tomorrow never knows? 总时间限制 ...

  7. Codeigniter MongoDB类库

    安装方法:1.将mongodb.php 放到config目录2.将Mongo_db.php放到library目录 使用方法: $this->mongo_db->where_gte('age ...

  8. php 开启COM组件

    1.先到PHP.INI中打开COM选项,com.allow_dcom = true 2.我这里的环境是PHP5.4.7,PHP 5.4.5后,com/dotnet 模块已经成了单独的扩展,所以需要在P ...

  9. Amazon关键词抓取

    亚马逊的网址构造很简单,几乎算是静态的网页,花费3小时完美收工,不要在意细节! 在python3下利用xpath就可以完美解决 xpath的使用方法请见: python之lxml(xpath) 入口图 ...

  10. BI案例:BI在连锁零售业应用(ZT)

    第一部分:连锁零售企业上BI的必要性. 目前国内的连锁零售行业的发展趋势,呈现出产业规模化,经营业态多样化,管理精细化的特点.所谓管理精细化就是"精耕细作搞管理,领先一步订系统". ...