javascript对象的创建--相对java 怎样去创建了"类"i以及实例化对象

  由于javascript没有java那么多基本类型,同时也没有提供class这个东西,那么我们想实现javascript的对象创建应该怎么办呢,我简单地从w3c提供的课件中提取了一下几种方法:

一.工厂模式

  说白了就是类似于c语言中的定义一个函数,然后给一个struct的各个属性复制,再返回这个struct。代码示例:

    function person(name,age,sex){

      var o=new Object();

      o.name=name;

      o.age=age;

      o.sex=sex;

      o.sayName=function(){

        alert(this.name);

      }

      return o;

    }

    var person1=new person("wangke",21,"Male");

    var person2=new person("gaoyuanyuan",34,"Female");

    person1.sayName();//wangke

    person2.sayName();//gaoyuanyuan

这种办法的确可以创建多个类似的对象,但是个人觉得类似c++中struct与class的区别,它没有解决对象的识别问题。

二.构造函数模式:

  其实在javascript中,构造函数和普通函数没什么区别,起码从格式上看来都是一样的,所谓的构造函数无非就是在调用的时候前面加了个new,比如上节的var person1=new person(...)。在这种模式中,通过使用this,将属性和方法赋值给this对象,这样就不用return了,同时一般构造函数的首字母会大写的(这也算是个认为加上去的区别吧)。示例代码:

  function Person(name,age,sex){

    this.name=name;

    this.age=age;

    this.sex=sex;

    this.sayName=function(){...};

  }

  var person1=new Person("wangke",21,"Male");

  var person2=new Person("gaoyuanyuan",34,"Female");

这种模式就可以进行类型检测了,通常用instanceof方法:

  alert(person1 instanceof Object);//true

  alert(person1 instanceof Person);//true

缺陷:每次实例化一个对象,里面的方法都会被重新创建一遍,浪费资源,所以我们可以想着把它放到全局变量中(但是只被个别方法调用,作用域为全局显得名不副实)。修改如下:

  

 function Person(name,age,sex){

    this.name=name;

    this.age=age;

    this.sex=sex;

    this.sayName=sayName;

  }

  function sayName(){

    alert(this.name);

  }

使用这种方法的话,会出现太多的全局函数了,所以我们会在下一种这种方法中解决这个问题。

三.原型模式:

  了解什么是prototype(原型):w3c给的概念有点绕,我通俗点说:prototype就是“一个给类的对象添加方法的方法”,假如有你把一个对象a赋值为一个对象b,假如对对象a进行拓展的话,那么对象b也能访问a新定义的属性或方法。可能这两句话还是不能理解,那就看代码以及w3c上提供的图(看了图就会恍然大悟了)。

  function Person(){

  }

  Person.prototype={

    name:"Nicholas",

    age:29,

    job:"Soft Engineer",

    sayName:function(){

      alert(this.name);

    }

  };

  var person1=new Person();

  var person2=new Person();

  alert(person1.sayName==person2.sayName();//true

 

备注:1.原型的动态性(开头也提到过,对a的拓展)即可以先创建实例再修改原型也可以:

  var person=new Person();

  Person.prototype.saySth(){

    alert("i want to sleep");

  };

  person.saySth();//i want to sleep

虽然修改可以立即在所有对象实例中反映过来,但是要重写所有的原型对象就会不一样了。

2.原生对象的原型--不光自定义类型方面原型很重要,所有原生的引用类型都是采用这种模式创建的,所以我们可以自己添加一些方法:

  String.prototype.startWith=function(text){

    return this.indexof(text)==0;

  };

  var msg="Hello,World";

  alert(msg.startWith("Hello");//true

3.原型模式的缺陷:共享的本性,特别是array。

  function Person(){

  }

  Person.prototype={

    name:"Nicholas",

    age:29,

    job:"Soft Engineer",

    friends:["wangke","yuanyuan"],

    sayName:function(){

      alert(this.name);

    }

  }

  var person1=new Person();

  var person2=new Person();

  person1.friends.push("***");

  alert(person1.friends);//wangke,yuanyuan,***

  alert(person2.friends);//wangke,yuanyuan,***

四.组合

  现在只要解决了原型模式中共享造成的缺陷,那么就比较完美了,我们注意到第二条的构造函数模式了吗?下面就是最常见的创建自定义类型的方式:

  function Person(name,age,sex){

    this.name=name;

    this.age=age;

    this.sex=sex;

    this.friends=["obama","kobe"];

  }

  Person.prototype={

    constructor:Person,

    sayName:function(){}

  }

实例属性都是在构造函数中定义,而共享的属性和方法都是在原型模式中定义。

五.其他构造方式:

  偶尔会用到,大家可以自己查一下:动态原型模式,寄生构造方式 ,稳妥构造函数模式

以上只是个人的一点总结,欢迎大家阅读并交流!

 

 

posted @ 2013-12-16 11:37  wangkeustc  阅读(227)  评论(0编辑  收藏  举报