Heim >Web-Frontend >js-Tutorial >js修改原型的属性使用介绍_javascript技巧

js修改原型的属性使用介绍_javascript技巧

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOriginal
2016-05-16 17:02:141284Durchsuche

在javascript中原型(prototype)定义了特定类型的所有实例都可以访问的属性和方法,很多些情况下需要重新对原型中的属性赋值,如果方法错误会导致一些意想不到的情况(仅仅是对像我这样的新手奋斗),下面通过测试对这部分知识做一个简单的总结。

基本类型定义如下:

复制代码 代码如下:

function Person(){}

Person.prototype={
constructor:Person,
name:"person",
age:100,
friends:["a","b"],
getName:function(){
return this.name;
}
};

定义两个Person的实例,修改实例中的name属性(该属性是在prototype中定义的),测试代码如下
复制代码 代码如下:

var p1=new Person();
var p2=new Person();

document.write(p1.name+"
"); //person
document.write(p2.name+"
"); //person

p1.name="p1";
document.write(p1.name+"
"); //p1
document.write(p2.name+"
"); //person

document.write(p1.hasOwnProperty("name")+"
"); //true 属于对象
document.write(p2.hasOwnProperty("name")+"
"); //false 属于原型

document.write(Object.keys(p1)+"
"); //name
document.write(Object.keys(p2)+"
"); // 空
document.write(Object.getOwnPropertyNames(Person.prototype)+"
"); //constructor,name,age,friends,getName
document.write(Person.prototype.name+"
"); //person

经过测试可以发现p1.name="p1"并不是修改了name的值而是在实例p1中新增加了一个name属性覆盖了prototype中的name属性,从后续的判断中就可以看出这时候p1的name属性已经是一个实例属性而不是原型属性,后面的Object.keys(p1)也可以看出p1这个实例中多了一个name属性而p2中没有。在js中所有的传递都是值传递,这个值可以是个指向引用类型的指针,所以等号并不意味着修改这个引用对象,而是切换了原来的引用关系,下面再通过代码说明这个问题
复制代码 代码如下:

var obj=new Object();
obj.name="obj";

function changeObj(o){
o.name="changed";
o=new Object();
o.name="newObj";
}
changeObj(obj);

document.write(obj.name); //changed

在changedObj方法中o=new Object()并没有修改参数o的值,而是切断了原来的引用关系,所以结果并不是newObj而是changed

接下来测试一下修改第一个例子中prototype中的friends属性,这个属性是一个引用类型
复制代码 代码如下:

p1.friends.push("c");
document.write(p1.friends+"
"); //a,b,c
document.write(p2.friends+"
"); //a,b,c

p1.friends=["x","y","z"];
document.write(p1.friends+"
"); //x,y,z
document.write(p2.friends+"
"); //a,b,c

document.write(p1.hasOwnProperty("friends")+"
"); //true 属于对象
document.write(p2.hasOwnProperty("friends")+"
"); //false 属于原型

document.write(Object.keys(p1)+"
"); //name,friend
document.write(Object.keys(p2)+"
"); //空
document.write(Object.getOwnPropertyNames(Person.prototype)+"
"); //constructor,name,age,friends,getName
document.write(Person.prototype.friends+"
"); //a,b,c

这次测试结果与第一次的测试基本相同,当通过等号修改时,这时候会切断原来的引用并为实例创建一个新的属性且覆盖了prototype中的同名属性

基于这两个测试结果发现在实例中不能直接修改prototype中的值类型属性(当然这种值类型不应该定义在prototype,这里的代码实例仅仅说明这个知识点,并无实际意义)
Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn