初步了解面向过程
面向过程是一种以过程为中心的编程思想。
面向过程也可以称之为‘面向记录’的编程思想,他们不支持丰富的面向对象特征(比如继承,多态),并且他们不允许混合持久化状态和域逻辑。就是分析出解决问题所需要的步骤,然后用函数把这些步骤一步一步实现,使用的时候一个一个依次调用就可以了。
面向过程其实是最为实际的一种思考方式,就算是面向对象的方法也是含有面向过程的思想。可以说面向过程是一种基础的方法。它考虑的是实际的实现。一般的面向过程是从上往下步步求精,所以面向过程最重要的是模块化的思想方法。对比面向过程,面向对象的方法主要是把事务给对象化,对象包括属性与行为。当程序规模不是很大时,面向过程的方法还会体验出一种优势。因为程序的流程很清楚,按着模块与函数的方法可以很好的组织。比如拿郝志枪喝茶这件事来说明面向过程,将过程拟为:
1:把昨天的茶叶倒掉
2:把杯子涮干净
3:倒进去茶叶
4:倒入开水
5:盖上杯盖等着喝
这几个步骤是一步一步去完成的,它的顺序很重要,只需要一个一个去实现就行了。而如果用面向对象放的的话,可能就只能抽象出一个郝志枪的类,它包括这四个方法,但是具体的顺序就不一定按照原来的顺序。
总结来说,面向过程就是分析出实现需求所需要的步骤,然后通过函数一步一步实现这个步骤,然后依次调用即可
概念
- 父类、子类
- 基类、派生类
浅谈面向对象
概念:
面向过程的编程方式由来已久。这种方式非常的直观,需要写一个功能,直接就写几行实现方法。比如你需要操作一个人移动到某个点,直接就写代码修改一个人的坐标属性,逐格的让他移动到目标点就行了
面向对象的编程方式,操作的是一个个的对象,比如你还是需要操作一个人的移动,你需要先实例化那个人的一个管理类对象,然后告诉这个“人”的对象,你需要移动到什么地方去。然后人就自己走过去了。至于具体是怎样走的,外部不关心,只有“人”对象本身知道
特性:
抽象性:
所谓的抽象性就是:如果需要一个对象描述数据,需要抽取这个对象的核心数据
提出需要的核心属性和方法
不在特定的环境下无法明确对象的具体意义
继承性:
所谓继承性就是自己没有但是别人有,拿过来成为自己的,就是继承,继承是实现复用的一种手段
在 Java 等语言中继承满足一个 class 的规则,类是一个 class,他规定了一个对象有什么属性和方法。
在这些语言中继承是 class 之间的继承,一个 class 继承另一个 class,那么该 class 就有了另一个 class 的成员,那么由该 class 创建出来的对象就同时具有两个 class 的成员。
在 JS 中没有明确的继承语法(ES6 提供了 class extend 语法),一般都是按照继承的理念实现对象的成员扩充实现继承,因此 JS 中实现继承的方法非常对多。
传统继承基于类,JS 继承基于对象
关于面向对象的一些其他概念:
类 class:在 JS 中就是构造函数:
在传统的面向对象语言中,使用一个叫类的东西定义模板,然后使用模板创建对象。
在构造方法中也具有类似的功能,因此也称其为类
实例(instance)与对象(object):
实例一般是指某一个构造函数创建出来的对象,我们称为 XXXX 构造函数的实例
实例就是对象。对象是一个泛称
实例与对象是一个近义词
键值对与属性和方法:
在 JS 中键值对的集合称为对象
如果值为数据(非函数),就称该键值对为属性
如果值为函数(方法),就称该键值对为方法 method
父类与子类(基类和派生类):
传统的面向对象语言中使用类来实现继承那么就有父类、子类的概念
父类又称为基类,子类又称为派生类
在 JS 中没有类的概念,在 JS 中常常称为父对象,子对象,基对象,派生对象。
构造函数
构造函数是干什么用的:
初始化数据的
在 js 中给对象添加属性用的,初始化属性用
创建对象的过程:
代码:var p = new Person();
首先运算符 new 创建了一个对象,类似于{},是一个没有任何(自定义)成员的对象。
使用 new 创建对象,那么对象的类型就是创建他的构造函数名
使用{}无论如何都是 Object 类型,相当于 new Object
然后调用构造函数,为其初始化成员
构造函数在调用的一开始,有一个赋值操作,即 this = 刚刚创建出来的对象
因此在构造函数中 this 表示刚刚创建出来的对象。
在构造函数中 利用 对象的动态特性 为其对象添加成员
作用域
什么是作用域:
域表示的就是范围,即作用域,就是一个名字在什么地方可以使用,什么时候不能使用。
简单的说,作用域是针对变量的,比如我们创建一个函数 a1,函数里面又包了一个子函数 a2。
| 1
2
3
4
5
6
7 | // 全局作用域
functiona a1() {
// a1 作用域
functiona2() {
// a2 作用域
}
} |
| — | — |
此时就存 在三个作用域:全局作用域,a1 作用域,a2 作用域;即全局作用域包含了 a1 的作用域,a2 的作用域包含了 a1 的作用域。
当 a2 在查找变量的时候会先从自身的作用域区查找,找不到再到上一级 a1 的作用域查找,如果还没找到就
到全局作用域区查找,这样就形成了一个作用域链。
js 中词法作用域的规则
函数允许访问函数外部的数据
整个代码结构只有函数可以限定作用域
作用规则首先使用提升规则分析
如果当前作用域中有了名字了,就不考虑外面的名字
属性搜索原则
所谓的属性搜索原则,就是对象在访问属性或方法的时候,首先在当前对象中查找
如果当前对象中存储着属性或方法,停止查找,直接使用该属性或方法
如果当前对象没有该成员,那么再在其原型对象中查找
如果原型对象中含有该成员,那么停止查找,直接使用
如果圆形中还没有,就到原型的原型中查找
如此往复,知道 Object.protitype 还没有,那么返回 undefined
如果是调用方法就报错,该 xxx 不是一个函数
闭包
对闭包的浅度理解
实用闭包主要是为了设计私有方法和变量。闭包的优点是可以避免全局变量的污染;缺点是闭包会常驻内存,增加内存使用量,使用不当很容易造成内存泄露。在 JavaScript 中,函数即闭包,只有函数才能产生作用域。
闭包的三个特性:
函数嵌套函数
在函数内部可以引用外部的参数和变量
参数和变量不会以垃圾回收机制回收
闭包有什么用(特性)
闭包的作用,就是保存自己私有的变量,通过提供的接口(方法)给外部使用,但外部不能直接访问该变量。
通过使用闭包,我们可以做很多事情,比如模拟面向对象的代码风格;更优雅,更简洁的表达出代码;在某些方面提升代码的执行效率。利用闭包可以实现如下需求:
匿名自执行函数::
一个匿名的函数,并立即执行它,由于外部无法引用它内部的变量,因此在执行完后很快就会被释放,关键是这种机制不会污染全局对象
缓存::
闭包正事可以做到这一点,因为它不会释放外部的引用,从而函数内部的值可以得到保留
实现封装::
模拟面向对象的代码风格::
闭包的基本模型
对象模式
函数内部定义个一个对象,对象中绑定多个函数(方法),返回对象,利用对象的方法访问函数内的数据
| 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21 | functioncreatePerson() {
varname = “”;
return{
getName: function() {
returnname;
},
setName: function( value ) {
// 如果不姓张就报错
if( value.charAt(0) === ‘张’) {
name = value;
} else{
thrownewError( ‘姓氏不对,不能取名’);
}
}
}
}
varp = createPerson();
p.set_Name( ‘张三丰’);
console.log( p.get_Name() );
p.set_Name( ‘张王富贵’);
console.log( p.get_Name() ); |
| — | — |
函数模式
函数内部定义一个新函数,返回新函数,用新函数获得函数内的数据
| 1
2
3
4
5
6
7
8
9
10
11 | functionfoo() {
varnum = Math.random();
functionfunc() {
returnmun;
}
returnfunc;
}
varf = foo();
// f 可以直接访问这个 num
varres1 = f();
varres2 = f(); |
| — | — |
沙箱模式
沙箱模式就是一个自调用函数,代码写到函数中一样会执行,但是不会与外界有任何的影响,比如 jQuery
| 1
2
3
4
5
6 | (function() {
varjQuery = function() { // 所有的算法 }
// …. // …. jQuery.each = function () {}
window.jQuery = window.$ = jQuery;
})();
$.each( … ) |
| — | — |
闭包的性能问题
js 垃圾回收机制,也就是当一个函数被执行完后,其作用域会被收回,如果形成了闭包,执行完后其作用域不会被收回
函数执行需要内存,那么函数中定义的变量,会在函数执行结束后自动回收,凡是因为闭包结构的,被引出的数据,如果还有变量引用这些数据的话,那么这些数据就不会被回收。因此在使用闭包的时候如果不使用某些数据了,一定要赋值一个 null
| 1
2
3
4
5
6
7
8
9 | varf = (function() {
varnum = 123;
returnfunction() {
returnnum;
};
})();
// f 引用着函数,函数引用着变量 num
// 因此在不使用该数据的时候,最好写上
f = null; |
| — | — |
原型
什么是原型
一句话说明什么是原型:原型能存储我们的方法,构造函数创建出来的实例对象能够引用原型中的方法。
JS 中一切皆对象,而每个对象都有一个原型(Object 除外),这个原型,大概就像 Java 中的父类,所以,基本上你可以认为原型就是这个对象的父对象,即每一个对象(Object 除外)内部都保存了它自己的父对象,这个父对象就是原型。一般创建的对象如果没有特别指定原型,那么它的原型就是 Object(这就很类似 Java 中所有的类默认继承自 Object 类)。
ES6 通过引入 class ,extends 等关键字,以一种语法糖的形式把构造函数包装成类的概念,更便于大家理解。是希望开发者不再花精力去关注原型以及原型链,也充分说明原型的设计意图和类是一样的。
查看对象的原型
当对象被创建之后,查看它们的原型的方法不止一种,以前一般使用对象的proto属性,ES6 推出后,推荐用 Object.getPrototypeOf()方法来获取对象的原型
| 1
2
3
4
5
6
7
8
9
10 | functionA(){
this.name=’lala’;
}
vara=newA();
console.log(a.proto)
//输出:Object {}
//推荐使用这种方式获取对象的原型
console.log(Object.getPrototypeOf(a))
//输出:Object {} |
| — | — |
无论对象是如何创建的,默认原型都是 Object,在这里需要提及的比较特殊的一点就是,通过构造函数来创建对象,函数 A 本身也是一个对象,而 A 有两个指向表示原型的属性,分别是proto和 prototype,而且两个属性并不相同
| 1
2
3
4
5
6
7
8
9
10
11 | functionA(){
this.name=’lala’;
}
vara=newA();
console.log(A.prototype)
//输出:Object {}
console.log(A.proto)
//输出:function () {}
console.log(Object.getPrototypeOf(A))
//输出:function () {} |
| — | — |
函数的的 prototype 属性只有在当作构造函数创建的时候,把自身的 prototype 属性值赋给对象的原型。而实际上,作为函数本身,它的原型应该是 function 对象,然后 function 对象的原型才是 Object。
总之,建议使用 ES6 推荐的查看原型和设置原型的方法。
原型的用法
其实原型和类的继承的用法是一致的:当你想用某个对象的属性时,将当前对象的原型指向该对象,你就拥有了该对象的使用权了。
| 1
2
3
4
5
6
7
8
9
10
11
12
13 | functionA(){
this.name=’world ‘;
}
functionB(){
this.bb=”hello”
}
vara=newA();
varb=newB();
//将 b 设置为 a 的原型,此处有一个问题,即 a 的 constructor 也指向了 B 构造函数,可能需要纠正
Object.setPrototypeOf(a,b);
a.constructor=A;
console.log(a.bb); //hello |
| — | — |
如果使用 ES6 来做的话则简单许多,甚至不涉及到 prototype 这个属性
| 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15 | class B{
constructor(){
this.bb=’hello’
}
}
class A extends B{
constructor(){
super();
this.name=’world’;
}
}
vara=newA();
console.log(a.bb+” “+a.name); //hello world
console.log(typeof(A)) //“function” |
| — | — |
怎么样?是不是已经完全看不到原型的影子了?活脱脱就是类继承,但是你也看得到实际上类 A 的类型是 function,所以说,本质上 class 在 JS 中是一种语法糖,JS 继承的本质依然是原型,不过,ES6 引入 class,extends 来掩盖原型的概念也是一个很友好的举动,对于长期学习那些类继承为基础的面对对象编程语言的程序员而言。
我的建议是,尽可能理解原型,尽可能用 class 这种语法糖。
好了,问自己两个问题:
- 为什么要使用原型?——提高函数的复用性。
- 为什么属性不放在原型上而方法要放在原型上?
利用对象的动态特性:构造函数.prototype.xxxx = vvv
利用直接替换
| 1
2
3
4 | Student.prototype = {
sayHello : function(){},
study : function(){}
}; |
| — | — |
原型链
什么是原型链
凡是对象就有原型,那么原型又是对象,因此凡是给定一个对象,那么就可以找到他的原型,原型还有原型,那么如此下去,就构成一个对象的序列,称该结构为原型链。
每个实例对象都有一个proto_
属性,该属性指向它原型对象,这个实例对象 的构造函数有一个原型属性 prototype
,与实例的proto**
属性指向同一个对象。当一个对象在查找一个属性的时, 自身没有就会根据**proto__
向它的原型进行查找,如果都没有,则向它的原型的原型继续查找,直到查到 Object.prototype._proto_
为 null
,这样也就形成了原型链
。
这个概念其实也变得比较简单,可以类比类的继承链条,即每个对象的原型往上追溯,一直到 Object 为止,这组成了一个链条,将其中的对象串联起来,当查找当前对象的属性时,如果没找到,就会沿着这个链条去查找,一直到 Object,如果还没发现,就会报 undefined。
原型链的结构
凡是使用构造函数,创建出对象,并且没有利用赋值的方式修改原型,就说该对象保留默认的原型链。
默认原型链结构是什么样子呢?
| 1
2
3 | functionPerson(){}
varp = newPerson();
//p 具有默认的原型链 |
| — | — |
默认的原型链结构就是:当前对象 -> 构造函数.prototype -> Object.prototype -> null
在实现继承的时候,有时候会利用替换原型链结构的方式实现原型继承,那么原型链结构就会发生改变
| 1
2
3 | functionDunizbCollection(){}
DunizbCollection.prototype = [];
vararr = newDunizbCollection(); |
| — | — |
此时 arr 对象的原型链结构被指向了数组对象的原型链结构了:arr -> [] -> Array.prototype -> Object.prototype -> null
用图形表示对象的原型链结构
以如下代码为例绘制原型链结构
| 1
2 | functionPerson(){}
varp = newPerson(); |
| — | — |
原型链结构图为:
使用原型需要注意两点:
- 原型继承链条不要太长,否则会出现效率问题。
- 指定原型时,注意 constructor 也会改变。
继承
实现继承有两种常见方式
混合式继承
最简单的继承就是将别的对象的属性强加到我身上,那么我就有这个成员了
混合式继承的简单描述:
| 1
2
3
4
5
6
7
8
9
10
11 | functionPerson() {};
Person.prototype.extend = function( o ) {
for( vark ino ) {
this[ k ] = o[ k ];
}
};
Person.prototype.extend({
run: function() { console.log( ‘我能跑了’); },
eat: function() { console.log( ‘我可以吃了’); },
sayHello: function() { console.log( ‘我吃饱了’); }
}); |
| — | — |
原型继承
利用原型也可以实现继承,不需要在我身上添加任何成员,只要原型有了我就有了
借用构造函数继承
这种技术的基本思想相当简单,即在子类型构造函数的内部调用超类型的构造函数,而函数只不过是在特定环境中执行代码的对象,因此通过使用 apply()和 call()方法也可以在(将来)新创建的对象上执行构造函数
| 1
2
3
4
5
6
7
8
9
10
11 | functionPerson ( name, age, gender ) {
this.name = name;
this.age = age;
this.gender = gender;
}
// 需要提供一个 Student 的构造函数创建学生对象
// 学生也应该有 name, age, gender, 同时还需要有 course 课程
functionStudent ( name, age, gender, course ) {
Person.call( this, name, age, gender );
this.course = course;
} |
| — | — |
函数的四种调用模式
函数模式
就是一个简单的函数调用。函数名的前面没有任何引导内容。
| 1
2
3
4
5
6 | functionfoo () {}
varfunc = function() {};
…
foo();
func();
(function() {} )(); |
| — | — |
this 的含义:在函数中 this 表示全局对象,在浏览器中式 window
方法模式
方法一定式依附与一个对象,将函数赋值给对象的一个属性,那么就成为了方法。
| 1
2
3
4
5
6 | functionf() {
this.method = function() {};
}
varo = {
method: function() {}
} |
| — | — |
this 的含义:这个依附的对象
构造器调用模式
创建对象的时候构造函数做了什么?由于构造函数只是给 this 添加成员,没有做其他事情。而方法也可以完成这个操作,就是 this 而言,构造函数与方法没有本质的区别。
特征:
使用 new 关键字,来引导构造函数。
构造函数中的 this 与方法中的一样,表示对象,但是构造函数中的对象是刚刚创建出来的对象
构造函数中不需要 return ,就会默认的 return this。
如果手动添加 return ,就相当于 return this
如果手动的添加 return 基本类型,无效,还是保留原来 返回 this
如果手动添加的 return null,或 return undefined ,无效
如果手动添加 return 对象类型,那么原来创建的 this 就会被丢掉,返回的是 return 后面的对象
上下文调用模式
上下文就是环境。就是自己定义设置 this 的含义。
语法
函数名.apply( 对象, [ 参数 ] );
函数名.call( 对象, 参数 );
描述
函数名就是表示函数本身,使用函数进行调用的时候默认 this 是全局变量
函数名也可以是方法提供,使用方法调用的时候,this 是指向当前对象
使用 apply 进行调用后,无论是函数还是方法都无效了,我们的 this ,由 apply 的第一个参数决定
参数问题
无论是 call 还是 apply 在没有后面的参数的情况下(函数无参数,方法五参数)是完全一致的
| 1
2
3
4
5 | functionfoo(){
console.log( this);
}
foo.apply( obj );
foo.call( obj ); |
| — | — |
第一个参数的使用也是有规则的:
如果传入的是一个对象,那么就相当于设置该函数中的 this 为参数
如果不传入参数,或传入 null、undefined 等,那么相当于 this 默认为 window
| 1
2
3 | foo();
foo.apply();
foo.apply( null); |
| — | — |
如果传入的是基本类型,那么 this 就是基本类型对应的包装类型的引用
在使用上下文调用的时候,原函数(方法)可能会带有参数,那么这个参数再上下文调用中使用 第二个(第 n 个)参数来表示
| 1
2
3
4
5
6 | functionfoo( num ) {
console.log( num );
}
foo.apply( null, [ 123 ] );
// 等价于
foo( 123 ); |
| — | — |
设计模式
工厂模式
构造模式
原型模式
单例模式
发布订阅
观察者模式
- Post link: https://blog.gaocaipeng.com/2020/03/12/mi580a/
- Copyright Notice: All articles in this blog are licensed under unless otherwise stated.