一 js中的对象是什么?
- 是一个容器,封装了属性(property)和方法(method),由若干键值对(key-value)组成
- 是对实物的抽象,实物间关系转换成了对象间关系,模拟现实的情况,转换为针对对象的编程。
- 例如将动物抽象成一类对象,此类对象有着动物中特有的属性和动作,不同于植物,而动物中又分有猪,狗,鸡不同类型的事物对象,这些类型事物对像既有着动物类型的属性和动作,又有着自身独特类型的属性和动作,可以这样一直细分下去,直到每一种动物的个例,也就是对象的实例了,它不仅具备上层事物共有的属性和方法,也有着自己独特个性,是一个独特的个体。
1-1 js的对象如何创建?
-- 对象可通过对象字面量({}),构造函数(如new Object(),或自定义),Object.create等方式创建。
1-1-1 对象字面量,与new Object()创建对象方式的异同?
相同点:创建的对象在使用上是一致的,var a = {} // 是var a= new Object()的语法糖
不同点:初始化的过程有些区别,new Object()是通过构造函数实例化对象,{ }是直接创建JSON对象,且在初始化时可以直接赋值,相比{ }比较高效
1-1-2 访问及修改对象的属性
-- 对象是由键值对组成,对象的键名又称属性(property),都是字符串类型,键名符合标识名条件或是数字可省略" ",系统会默认自动转为对应的字符串,键值可以是任意类型,若键值是函数,则该属性又可被称为方法,像函数般调用。
- 使用 . 或 [ ] 访问或修改增添对象的属性
- delete删除属性
- in检查是否属于自身的属性
- Object.keys(obj)查看所有自身可枚举属性,Object.getOwnPropertyNames(obj)包括自身不可枚举的属性如数组中的"length",都返回属性名数组
- for...in 循环遍历对象本身属性
- Object.observe(obj,function(changes))观察对象属性变化
1-1-2eg1:对象字面量式创建
<script> //对象字面量式创建 var obj = { name: 'fermin', run: function () { alert('run'); } }; obj.age = 18; //添加属性 console.log(obj.age); //18 delete obj.age; //删除属性,注意:不能删除继承的属性,会返回false,删除不存在的属性不会报错,且返回true alert(obj.age); // undefined,访问不存在的属性返回undefined alert("age" in obj); //false, 注意:in对对象拥有的属性都返回ture,无论是否是继承的,无法识别继承属性 console.log(Object.keys(obj)); //["name","run"] //Object.keys()返回所有本身属性名 for (let i in obj) { console.log(i); //name run //for..in 遍历可enumerable的自身和继承的属性,可加obj.hasOwnProperty(i),过滤为自身属性 } obj.run(); //run //用点号访问属性和方法 obj['run'](); //run //用[]访问属性和方法 alert("toString" in obj); //true
</script>
构造函数:构造函数创建对象是借用Object.create(原型) 来实现。
1-1-2eg2:new Object() 创建 (访问和修改对象属性与上述相同)
<script> //new Object()创建 var obj = new Object(); //new可以省略 obj.name = 'fermin'; obj.age = 18; //添加属性 obj.run = function () { alert('run'); }; //添加属性 delete obj.age; //删除属性 //delete只能删除对象的属性如:var a=1; delete a //false alert(obj.age); // undefined,访问不存在的属性返回undefined alert("age" in obj);//false console.log(Object.keys(obj)); //["name","run"] for (let i in obj) { console.log(i); //name run } obj.run(); //用点号访问属性和方法 obj['run'](); //用[]访问属性和方法 alert("toString" in obj); //true
</script>
1-1-2eg3:自定义构造函数
<script> //自定义构造函数 function Person(name) { this.name = name; } var obj = new Person("fermin"); obj.age = 18; obj.run = function () { alert("run"); }; //添加属性 alert(Object.keys(obj)); //name,age,run for (let i in obj) { console.log(i); //name age run } obj.run(); // run obj['run'](); // run alert("toString" in obj); //true </script>
1-1-2eg4:Object.create(原型) // Object.create(Object.prototype) <==> new Object()
<script> //Object.create(原型); var obj = Object.create(null); //原型为null,不能继承Object.prototype中的属性 obj.name = 'fermin'; obj.age = 18; //添加属性 obj.run = function () { alert('run'); }; //添加属性 delete obj.age; //删除属性 alert(obj.age); // undefined,访问不存在的属性返回undefined alert("age" in obj);//false console.log(Object.keys(obj)); //["name","run"] for (let i in obj) { console.log(i); //name run } obj.run(); //用点号访问属性和方法 obj['run'](); //用[]访问属性和方法 alert("toString" in obj); //false //没有继承Object.prototype中的属性 </script>
1-1-3 new一个对象的过程
var obj = new Function();
- 创建一个空对象 // var obj = {};
- 将实例的__proto__属性指向构造函数的prototype原型 // obj__proto__ = Function.prototype
- 将构造函数的指针指向实例 // Function.call(obj)
1-1-3eg1:模拟new过程
<script>
function Person(name) { this.name = name; } var obj = Person("fermin"); //不用new obj = {}; obj.__proto__ = Person.prototype; Person.call(obj); obj.age = 18; obj.run = function () { alert("run"); }; //添加属性 alert(Object.keys(obj)); //name,age,run for (let i in obj) { console.log(i); //name age run } obj.run(); // run obj['run'](); // run alert("toString" in obj); //true </script>
1-1-4 函数中this的指向问题 (避免在函数中包含多层this,往往第一层代表调用的对象,第二层就表示window了)
-- this的指向在函数执行时才能确定,总是指向调用该函数的对象
- 由new调用,指向新建的对象 //在构造函数执行
- 由call,apply,bind调用,指向绑定的对象 //在强制绑定对象执行
- 由上下文调用,指向所属的上下文对象 //在对象属性执行
- 没有所属对象时,严格模式指向undefined或指向全局对象(window或global) //在普通函数执行
1-1-4-eg1:
<script> var obj1= { a:3, b:{ a:10, fn:function(){ console.log(this.a); //undefined console.log(this); //window } } }; var j = obj1.b.fn; j(); //将fn赋值给变量j时,obj调用fn没有执行,所以它最终指向的是window,而不是obj var obj2 = { a : 'A', fn: function () { console.log(this.a); } }; obj2.fn(); //"A" // this ===obj2 obj2.fn.call({a: 'AA'}); //"AA" // this === {name: 'AA'} var fn1 = obj2.fn; fn1(); //undefined //this === window </script>
二 object对象的方法
2-1 六个实例对象方法,继承Object.prototype
- valueOf():返回当前对象对应的值,默认返回对象本身 // var o1 = new Object (); o1.valueOf === o1 //true
- toString():将对象转换为字符串形式并返回 // var o2 = {a : 1}; o2.toString() // [object Object] //用Object.prototype.toString.call() 更准确的判断类型[object,对象类型11种/ Number/ String/ Boolean/ Object/ Array/ Function/ Null/ Undefined/ RegExp/ NaN/ Infinite]
- toLocalString():将对像转换为本地对应的字符串并返回
- hasOwnPropoty():判断某个属性是否是自身的非继承属性,是的返回true
- isPrototypeOf(): 判断当前对象是为另一对象原型,是返回true
- propertylsEnumerable():判断某个属性是否可枚举
2-2 对象属性的特征 attributes 对象用Object.getOwnPropertyDescriptor(obj,字符串属性名)读取。
- value:表示该属性的值,默认undefined
- writable:表示该属性的值是否可改,默认true
- enumerable:表示该属性名是否可枚举,默认true //改为false时,for..in 及Object.keys(),JSON.stringify()不遍历该属性,可成私密属性
- configurable:表述该对像是否可配置,默认true //var 声明的变量会为false,没声明为false,如a=1或this.a=1 //改为false时,writable可从true改为false,当writable为true可改value值,其他值不可修改,不能再用delete删除该属性
- get:表属性取值函数(getter)默认undefined //定义后,writable不能为true且定义value的值 // get,set为存取器的命令,常使用于某属性值需依赖对象内部数据场合。 // 利用存取器可以实现数据对象与DOM对象的双向绑定
- set:表属性存值函数(setter)默认underfined //定义后,writable不能为true且定义value的值
2-3 定义对象属性的attributes对象,下面两种方法定义后,属性中的writable,enumerable,configurable的值默认值又都会变为false
- Object.defineProperty(obj,字符串属性名,attributesObj)
- Object.defineProperties(obj,{字符串属性名: attributesObj,字符串属性名: attributesObj,...})
2-3-eg1:定义单一attribules属性
<script> var o = Object.defineProperty({}, "p", { value: 3, writable: false, enumerable: true, configurable: false }); alert(o.p); //3 o.p = 33; alert(o.p); //3 //writable为false,修改不了该属性的值 </script>
2-3-eg2:定义多个attributes属性
<script> var o = Object.defineProperties({}, { p1: { value: 3, enumerable: true, writable: true, //configurable值没设定,默认为false }, p2: { get: function () { return this.p1 + 6; }, //有get,就不能直接定义其value值 enumerable: true, configurable: true } }); alert(o.p1); //3 o.p1 = 4; //writable,值为true后,可直接用.或[]修改该属性值, alert(o.p1); //4 alert(o.p2); //10 Object.defineProperty(o, "p1",{value:33}); //若再修改p1其它属性值会报错,因为一开始configurable为false alert(o.p1); //33 //PS:当writable值为false,configurable值为true,只能在defineProperty()中修改 </script>
2-4 对象的拷贝
--将一对象所有属性拷贝到另一个对象上,有重复的属性,将被覆盖
<script> var extend = function (to, from) { for (var property in from) { var descriptor = Object.getOwnPropertyDescriptor(from, property); if (descriptor && (!descriptor.writable || !descriptor.enumerable || !descriptor.configurable || !descriptor.get || !desciptor.set)) { Object.defineProperty(to, property, descriptor); //都拷贝 } else { to[property] = from[property]; //这个else遇存取器定义属性只拷贝值,适用于descriptor的5个特征都具备 } } return to; }; var aa = extend({a:1, b:2}, {a:11, bb:22}); console.log(aa); //{a:11, b:2, bb:22} </script>
2-5 控制对象的状态
- Object.preventExtensions(obj) //无法再添加新属性,严格模式下添加会抛错,可用delete删除现有属性
- Object.isExtensible(obj) //没使用Object.preventExtensions(obj)返回false,使用了返回true
- Object.seal(obj) //无法添加新属性,也无法删除旧属性,现有属性的configurable会变为false
- Object.isSeal(obj) //没使用Object.seal(obj)返回false,使用了返回true,且isExtensible返回false
- Object.isFrozen(obj) //是否被冻结,指不可扩展,所有属性不可配置,所有数据属性(即没有getter或setter组件的访问器的属性)都是不可写的。
2-5-eg:
<script> var o = new Object(); Object.preventExtensions(o); o.p = 1; console.log(o.p); // undefined console.log(Object.isExtensible(o)); //false </script>