面向对象的历程
面向对象
一、单例模式
1.1 对象数据类型的作用:
把描述一个对象的属性和方法放在一个单独的空间,与其他的对象分割开,即时出现属性名相同的情况,也不会产生冲突
var name="xiao 1" var age=12 var name="xiao 2" var aeg=13 //单例模式 var person1={ name="xiao 1", age=12 } var person2={ name="xiao 2", age=13 }
单例模式就好像上面对象数据类型一样,使用分组编写代码思想。在单例模式中person1叫命名空间,只是存放空间的一个名字而已。
1.2 模块化开发:
在项目中使用单例模式来开发一个大型项目,把当前的需求分成若干个功能模块,每个人负责一部分,同时开发,最后把所有人的代码整合在一起。
// 公共模块 var utils = { select: function () { } } // tab模块 var tabrender = { change: function () { utils.select() //公共模块方法调用 } } // search模块 var searchrender = { change: function () { this.clickevent() //自己命名空间下方法调用,this可以避免修改名字的麻烦 }, clickevent: function () { } }
缺点:单例模式手工生产输出单个个体,需要批量多个是就麻烦。
二、工厂模式
2.1 工厂模式:
用函数封装实现同一件事相同的代码,要实现该功能,只需要执行函数就可以了。
function creattable(w,h) { var obj = {}; obj.width = w; obj.height = h; obj.fn = function () { console.log("the table width is "+this.width+" and height is "+this.height) } return obj; } var mytable=creattable(200,100) mytable.fn()
2.2 低耦合高内聚:
减少页面中的冗余代码,提高代码的重复利用率
三、一些概念
- 继承:子类继承父类的属性和方法
- 封装:把相同的代码放在一个函数里面,以后要实现该功能执行函数就可以了。
- 多态:当前方法的多种形态。后台语言中包括重载(传入的参数不一样,方法也不一样,js没有重载)和重写(子类重写父类的方法通过原型链)。
- 对象:js中万物皆是对象。一个泛指
- 类:对象的具体细分
- 实例:一个类别中具体的一个个体
- 举个例子 对象:大自然;类:动物、植物、人 实例:猫狗 你我他
+内置类:
number,string,boolean,null,undefined
object、array、regexp、date、function...
四、构造函数模式
主要的目的创建一个自定义类,并创建这个类的实例。
构造函数和工厂模式区别
4.1 执行的时候
构造函数要在类(首字母一般大写,不是函数和方法)(所有的类都是函数数据类型)前加new,返回的是类的实例
var a=creattable(10,10) var b=new creattable(10,12)
4.2 代码执行
先创建一个私有的作用域,形参赋值,进行预解释,代码执行从上到下执行。
特别的地方,不用手动创建空的对象obj,浏览器会默认帮我们创建一个对象数据类型(这个对象是我们当前类的一个实例)(用this代表),代码执行,把属性名、属性值赋予当前的实例,最后浏览器返回当前的实例。
function creattable(w,h) { // 创建mytable实例 也就是this this.width = w; // 赋值 mytable.width this.height = h; this.fn = function () { console.log("the table width is "+this.width+" and height is "+this.height) } // 返回实例 } var mytable=new creattable(200,100) mytable.fn()
知识点:
1、js所有的类都是函数数据类型、所有的实例都是对象数据类型
2、类中this是类的一个实例
3、this.xxx=xxx都是私有属性,不同实例私有属性不相等
4.3 构造函数扩展
1、构造函数中new fn() 执行,如果没有传递参数,后面的括号可以省略new fn。
2、this在类中出现指的是当前类的一个实例,在一个属性值是方法时,要看执行的时候前面有".",有就是点前面的,没有则是window
function fn() { this.a=50; this.geta=function () { console.log(this.a); } } var f1=new fn; // 类后面可以不加括号 f1.geta() //50 this-->f1 var f2=f1.geta //f2 --> function(){console.log(this.a);} f2() //undefined this-->window
3、在构造函数中,浏览器会默认的返回一个对象数据类型的值,如果我们手动的创建返回的值。返回的是基本数据类型的值,创建的实例对象没有影响。返回的是引用数据类型的值,当前的实例会被返回值替换。
function fn() { this.x=1; this.getx=function() { console.log(this.x); } return {x:233} //基本数据类型没有影响 } var f1=new fn; console.log(f1) //{x: 233}
4.4 检测类型
- 检测某一个实例是否属于这个类 instanceof
console.log(f1 instanceof fn) true
- 检测属性属于对象 in
console.log(x in f1) true
- 是否是私有属性 hasownproperty
console.log(f1.hasownproperty("grtx")) true
- 是否是公有属性
function haspunproperty(obj,attr) { retrun (attr in obj) && !obj.hasownproperty(attr); } console.log(haspunproperty(f1,"getx")) //flase
五、原型链模式
5.1 概述
构造函数模式有类和实例的概念,把两者区分开,那如何识别每一个类,就需要原型链模式。
把实例上公有的属性和方法放到类的原型上(xxxx.prototype)
function creattable(w, h) { this.width = w; this.height = h; } creattable.prototype.fn = function () { console.log("the table width is " + this.width + " and height is " + this.height) }
- 1、每一个函数数据类型(类)都有一个天生自带的属性(prototype),属性值是对象数据类型。
- 2、在prototype上浏览器天生给他加上一个constructor属性,属性值是当前函数(类)本身。
- 3、每一个对象数据类型(实例,对象,prototype)天生自带一个属性_proto_,属性值是当前实例所属类的原型(prototype)
5.2 原型链
一个实例的方法在运行的时候,如果这个方法是自己私有的,那么就直接用,如果不是私有的,那么通过__proto__去所属类的原型上去查找,如果还没有就通过原型的__proto__一直查到基类的object.如果还没有报错,如果有就直接用了。我们把这种通过__proto__查找的机制叫做原型链.
5.3 批量设置原型上的公有属性和方法
1、起别名
function fn() { this.x = 100 } var pro = fn.prototype; pro.getx = function () { } pro.gety = function () { }
2、重构原型上的对象
重新开辟一个堆内存,存储自定义公有的属性和方法,原来浏览器开辟的堆内存会给替换掉(没有constructor指向不是创建实例的类而是向上查找的object,为了保持一致可以手动增加constructor的指向)浏览器在空闲的时候会销毁回收内存。
function fn() { this.x = 100 } fn.prototype={ constructor: fn, getx: function () { }, gety: function () { } }
用上面的方法给内置的类增加公有的方法,会把原来的属性和方法替换,浏览器会屏蔽这种方法。
但是我们可以一个个去修改内置的方法,重复修改没有则增加(用特殊的命名标记自己的方法)
array.prototype = { constructor: array, unique: function() {} }; console.dir(array.prototype); //// array.prototype.sort1 = function () { console.log("ok");//this->ary 我们当前要操作的这个数组 }; var ary = [1, 2, 2, 1, 2, 3, 4, 2, 1, 3]; var res=ary.sort1(); //把sort1 改成sort 就是修该内置的sort console.log(ary); console.log(res);
5.4 原型,模式中this指向问题。
数组去重 array.prototype.unique = function () { var obj = {} for (var i = 0; i < this.length; i++) { var cur = this[i] if (obj.cur === cur) { this[i] = this[this.length - 1] this.length-- i-- continue } obj.cur = cur } obj = null return this } ary = [2, 12, 121, 12, 1, 2, 2, 1, 3, 41, 1] ary.unique()
链式写法:执行完成数组的一个方法可以紧接着执行下一个方法
//数组从小到大,再倒序排列,去除最后的一个数,返回去除的数,原数组改变 array.sort(function (a, b) {return a-b }).reverse().pop()
5.5 可枚举和不可枚举
object.prototype.lol=function () { } var obj={name:"小a",age=12} for(var key in obj){ //for in默认的情况下只会遍历到实例的私有属性和我们在实例所属类的原型上增加的方法属性。 if(obj.propertyisenumerable(key)){ //原型上的方法和属性都是不可枚举的不管是内置还是增加的 console.log(key) } if(obj.hasownperporty(key)) { //另一种方法,只枚举公有的方法 console.log(key) } }
5.6 原型继承
子类和父类通过原型链有了关联,也是通过查找原型链来实现,
function a() { this.x = 100 } a.prototype.getx = function () { } function b() { this.y = 200 } //让b继承a的方法和私有属性 b.prototype = new a; //父类中公有和私有都会变成子类原型上的公有的属性和方法
上一篇: 个人注意