热前端Web前端之路

js进阶篇:对象和原型

2017-02-19  本文已影响19人  leoying

1.js中没有太明确的类的概念,每个function函数都可以看做是一个类,类中可以通过prototype属性来添加这个
类中的方法和字段。
2.一般情况把对象赋值或者添加到一个函数的prototype属性中,函数的prototype
对对象的值做出更改,对象本身也是会发生变化的,可以使用一种临时函数来打破这种关系链。
临时函数new F();

代码示例:
var obj = {x:0,y:1};
var F = function(){};
F.prototype = obj;
var f = new F();
console.log(f); 

//实例化之后 obj这个对象会挂在到f对象的原型中
f.x = "zs"; //这个时候其实是为f这个对象添加了一个自身属性。所有并没有对原型属性进行覆盖更改.
console.log(obj); //{x:0,y:1}; 所以这里的obj对象不会有任何变化,也就是说这里只是继承了它的原型属性。
如果浏览器支持ES5可以通过 Object.create();这个方法来直接使用,和临时构造器的函数等价

代码示例:
var f = Object.create({x:2,y:3}); 

3.一个函数使用prototype添加方法属性,和不使用prototype的区别:

function person(){}
person.name = function(){console.log("name")};  //相当于使用了静态方法  
person.prototype.age = function(){};     //通过new

由于使用了静态方法:调用的时候直接 person.name();即可 如果new了之后会出错
age属于对象的方式,需要new之后使用。
4.使用 hasOwnProperty 方法可以判断属性是否存在(用来检测是否含有自身属性)。
5.for...in 遍历对象时,只会遍历可枚举属性。
6.自身属性和原型属性

代码示例:
function func()
{
    this.name = "";    //自身属性
}
func.prototype.hello = "";   //原型属性

所谓自身属性通过构造器定义的属性,原型属性会挂载到prototype上。
通过 关键字 new之后会将属性挂载到这个func中,然后就可以通过for in关键字查找。

var f = new func();
for(var p in f)
{
   if(f.hasOwnProperty(b))  //判断是自身属性
      console.log(p);       //判断之后这里只会打印name这个属性
}

这里我们也是可以这样理解的:f.prototype == f.proto; //true

对象查询属性的方式,会首先在自身属性中去查找,如果查找不到的话会通过proto到对象的原型中去查找,如果
通过原型依然没有找到,会去原型对象的原型中再去查找,一直到找到或者属性为null的时候为止。这里也可以得出
一个结论:js的原型链的本质在于proto这个神秘的链接

7.对象可以通过 delete 删除指定对象的属性,当删除一个不存在的属性时也同样会返回true,用来删除变量和其他,可配置属性是无效的

示例:var a = "liu ying";
     delete a; //返回一个  false,严格模式下这样做的话会直接报错

8.Object.create()的使用
该方法接收两个参数,第一个参数是一个对象,第二个为可选参数。通过对象的属性进一步描述。传入一个对象后
返回值会继承传入对象的属性,通过对该返回值做修改,传入的远对象值不会随着变化。

代码示例://
var a = {name:"liu ying",age:23};  
var b = create.Object(a);
b.name = "hello";  //对返回值做修改
console.log(a);    //{name:"liu ying",age:23}   这个时候a对象的值的属性依然不会变化。
所以从这个属性可以看出,打破了原有的引用关系链。

9.内置构造函数的原型是只读的,Object.prototype = "liu ying",比如这样设置,不会报错,但是赋值是没有
生效的。
10.hasOwnProperty可以用来检测当前的属性是否是对象的自身属性,如果是继承的属性将会返回false
11.constructor属性:所有的对象都会从他的原型上去继承一个constructor属性,函数其实也是属于一种特殊的对象。

代码示例:
var a = function(){};
var b = new a();
b.constructor == Object;   //这里是返回一个true;
但是有些js的内置对象,不通过实例化的方法也可直接比较,也会返回true。
比如:
document.constructor == Document;
document.form3.constructor == Form;

12.proto属性:该属性是一个访问器,(一个getter函数和一个setter函数),它公开访问它的对象的内部prototype属性(对象或null)。
proto的使用是有争议的,尽量不要使用
13.Object.defineproperty()属性
该属性具备三个参数,第一个为目标对象,第二个为目标对象的key值,第三个参数为对象配置参数.详细说明一下第三个参数
第三个参数和设置一般对象一样,可以设置是否可枚举(enumerable),是否可配置(configrable),是否可写(writable);
这里默认都是false.
如果设置不可枚举,for.. in 的话是无法遍历出来内置的不可枚举属性的。通过Object.defineproperty可以创建对象
属性:

代码示例
var o = {x:0,y:1};
Object.defineproperty(o,"z",{
    configrable:true,
    enumerable:true,
    writable:true,
    value:5
});
console.log(o);    //{x:0,y:1,z:5}

除了添加可配置的属性,还有一个特别强大的功能,就是get和set方法,配置get和set方法以后可以检测当前属性的变化
如果属性值变化的话,会自动触发set方法.

代码示例:
var o = {x:0,y:1};
Object.defineproperty(o,"x",{
    configrable:true,
    enumerable:true,
    writable:true,
    get:function()
    {
        return this.x;
    },
    set:function(newVal)
    {
        this.x = newVal;
    }
});
//这里的get和set如果我手动修改了o.x的值的话,set方法就会自动触发,然后get方法返回值。
o.x = 2;   //这个时候set方法就会触发

14.Object.freeze()属性可以冻结一个对象,使该对象为只读。如果一个对象的属性也为对象,则冻结无效,如果
需要彻底的冻结的话就要使用递归来层层使用 freeze()属性来设置。
15.Object.getOwnPropertyDescriptor();该方法带两个参数:对象和key,通过key查找该对象的自身属性,然后会返回
当前对象的一些信息(value,是否枚举,是否可写)等等.
16.getOwnPropertyNames(),接收一个参数,需要是一个对象类型,根据传入的数据返回一个数组(当前的对象的key值);和Object.keys相比
个人觉得还是有一点相似之处的,不同的是该属性包括不可枚举的属性名也会一起返回。
17.Object.getPrototypeOf() 接收一个对象,能够返回该对象的原型。

代码示例:
var Vue = function(){};
Vue.prototype.hello = function(){};
var vue = new Vue();
//这里把vue这个对象传入 
var v = Object.getPrototypeOf(vue);
console.log(v);  //{hello:function(){}}  

18.Object.is()属性,接收两个值,判断两个值是否完全相等,返回一个布尔值。
19.Object.isFrozen() 方法判断一个对象是否被冻结(frozen)。 这里的冻结指的是所有的配置属性都是冻结状态,才会返回true
20.obj.propertyIsEnumerable(prop);可以判断该对象的当前属性是否可枚举。

上一篇 下一篇

猜你喜欢

热点阅读