一、prototype的应用
每一个函数都拥有一个属性 prototype
给构造函数提供共有的属性和方法
// prototype 原型对象 共有属性和方法
// 如何封装构造函数
// 普通的实现
// 把普通代码中的变量变成属性
// 把普通代码的代码块变成方法
// 注意:this的改变
// 提供共用的属性和方法
Date.prototype.format = function () {
// this -> 实例化的这个对象,不是构造函数
let y = this.getFullYear();
let m = this.getMonth() + 1;
let d = this.getDate();
return y + '-' + m + '-' + d
}
let date = new Date();
let res = date.format();
console.log(res); // 2022-1-6
二、 原型
原型 __proto__
每一个对象都有__proto__属性 -> 指向对应的构造函数的原型对象
去查找一个对象是否有某个属性或方法
构造函数 + 原型对象
constructor __proto__
hasOwnProperty() -- 用于判断属性或者方法是不是自己的(还有可能是继承
来的)
function Dog(type) {
this.type = type ;
this.say = '汪'
}
Dog.prototype.color = 'red'
const d = new Dog('二哈')
console.log(d);
console.log(d.say); // 汪
console.log(d.color); // red
console.log(d.constructor); // 构造器
console.log(d.__proto__); // 原型对象
三、原型对象 prototype
function Person(name , age) {
this.name = name ;
this.age = age ;
}
Person.prototype.xing = '陈'
const p = new Person('小花',16)
// p 自己没有xing这个属性
console.log(p.xing); // 陈
console.log(p.hasOwnProperty('name')); // true
// name是p自身的属性
console.log(p.hasOwnProperty('xing')); // false
// xing不是p自身的属性,是继承而来的
console.log(p.constructor);
四、原型链
原型链
每一个对象都有原型,也就是__proto__这个属性,这个属性会指向这个构造
函数函数的原型对象,也就是prototype
构造函数的原型对象也是对象,也有原型,他指向对象也就是object的原型对
象
最终Object的原型对象也是对象,它的原型最终指向null
// 原型会指向构造函数的原型对象
console.log(p.__proto__ == Person.prototype); // true
// 原型对象也是一个对象, 因此也有原型,它的原型就是Object.prototype
console.log(Person.prototype.__proto__ == Object.prototype); // true
// Object.prototype 也是原型对象,指向null
console.log(Object.prototype.__proto__ == null);//true
console.log(p.__proto__.__proto__.__proto__ == null); // true
// 函数也是对象,因此函数也有原型
console.log(Person.__proto__ == Function.prototype); // true
//
console.log(Function.prototype.__proto__ == Object.prototype);
console.log(Object.prototype.__proto__ == null);
五、instanceOf
// Array.isArray() // 布尔值
// constructor // 可以判断数组
// instanceof 用于判断一个对象是否是一个构造函数的实例化对象
// instanceof 判断某一个对象是不是这个构造函数的实例化对象
console.log(arr instanceof Array); // true
console.log(arr.constructor == Array); // true
console.log(arr.__proto__ == Array.prototype);//true
六、new
new的贡献
1 创建了一个对象
2 this指向了这个对象
3 把这个对象的原型指向了构造函数的原型对象
4 返回了这个对象
function Person(type) {
// const obj = {}
// this -> obj
// obj.__proto__ = Person.prototype
// return obj
console.log(this);
// 这个obj的原型是Object.prototype
}
const p = new Person()
console.log(p.__proto__ == Person.prototype); // true
const obj = {}
console.log(obj.__proto__ == Object.prototype);// true
七、原型的引用
const oPs1 = document.querySelectorAll('p')
// NodeList
const oPs2 = document.getElementsByTagName('p')
// HTMLCollection
console.log(oPs1,oPs2);
// NodeList forEach
// HTMLCollection 没有forEach
oPs1.forEach( v => {
console.log(v);
})
const len = oPs2.length
oPs2.__proto__ = Array.prototype // 强行改变了原型的指向 length 没了
oPs2.length = len
console.log(oPs2);
oPs2.forEach( v => {
console.log(v);
})
// oPs2.push('a') push会改变数组