js创建对象的方法及其优缺点 JavaScript是面向对象还是基于对象?
JavaScript是面向对象还是基于对象?
什么是面向对象在编程的世界里,有一种思想非常重要,那就是面向对象的思维。掌握了这个思路,就意味着你不再是一个程序员新手,开始向开发者的目标迈进了。
那么,到底什么是面向对象的思维呢?说到这,我可以告诉你这个东西的来历。以前的编程是面向过程的。那么什么是面向过程的呢?之前在网上看到一句话,觉得形容的很好。让我借一下。
一般来说,如果现在有人让你把大象放进冰箱,我们如何用面向过程的方法来做?我们可以把它分成以下几个步骤:
1.打开冰箱门。
2.把大象放进冰箱。
3.关上冰箱门。
我们程序员会把这三个步骤写成三个函数:
()
2.putElephantIntoFridge()
()
然后我们可以依次调用这些函数。嗯,你以为可以下班了,但是过几天,你会发现这种需求会演变成很多奇怪的趋势,比如:
请把狮子放进冰箱。
请把大象放进微波炉里。
请把猴子放在微波炉里。
把其他动物放在冰箱里,但是不要放。;不要关门。
5.……
诸如此类。这时,要以面向过程的实现这些需求,就需要重新定义实现这些需求的功能。这无疑是一件令人抓狂的事情。但是老板和客户出钱了,你就得做!所以你必须加班,所以你牺牲了你的工作...
所以,为了你的生活,你必须想出一个方法,不。;不需要每次需要时都重新定义实现的功能,也就是说,面向对象。
我们的想法是:如果每次我们想改变需求,我们不要 不要自己去做这些过程,而是指导别人去做,不是吗?;没关系吗?所以我们面向对象思维的第一个转变就是做一个执行者,成为一个指挥者。
如果用定向思维来完成把大象放进冰箱的需求。我们的方法变成这样:
1.找到冰箱,命令冰箱自己打开冰箱门。
2.找到大象,命令它自己进入冰箱。
命令冰箱再次自行关门。
所以实现这个需求需要的实体是:大象和冰箱。我们把出现在实现需求中的实体称为对象。大象要能自己进入冰箱,冰箱要能自己开关门。进入冰箱、开门和关门的能力,我们称之为对象,通常用编程方法来表达。
所以总结一下:
1.面向过程是实现需求的第一步,任何工作都需要自己去做。
2.面向对象意味着把一切都交给能做的人。
所以现在的问题是,如果有必要,当求变变成上面说的那些,如何解决面向对象的问题?我们现在要做的是分析需求中出现的对象(实体),然后赋予它们相应的能力。
在新的需求中,大象、狮子和猴子应该被放进冰箱和微波炉。这时,这里出现的物体(实体)是:动物和容器。动物需要的方法(能力)是进入容器,容器需要的方法(能力)是开关门。
因此,上述所有要求都可能变成:
1.[集装箱]开门。
2.[动物]进入[容器]。
3.关闭[容器](或者如果需要关门,可以省略这一步)
所以这样一来,我们就不 不需要重复定义函数来实现这些需求。甚至在未来,需求变成了把动物从容器里拿出来,我们只要在动物物体上拓展把动物从容器里拿出来的方法,就可以快速完成需求。这个时候,你牺牲工作的几率就小多了。
如何实现面向对象
说了这么多,大家大概也能明白什么是面向对象了。那么如何用js写代码实现面向对象呢?
在JavaScript中,我们使用构造函数来创建对象。
函数大象(){
}
大象会有一些独特的数据,比如它的体重、品种、年龄等等。我们称这些独特的数据为:属性。每头大象 的数据就不一样了。这种差异在代码中是如何体现的?
功能象(年龄、体重、类型){
}
我们将每个数据作为形式参数传入构造函数,然后在创建时决定每个大象的实际数据。最终的构造函数被写成:
功能象(年龄、体重、类型){
年龄
这个重量重量
这种类型
}
现在,如果我们想要一头大象,我们只需要用new的创建这个对象。
//这是一头2岁重200kg的非洲象。
Var ele1新象(2,200kg,非洲象)
//这是一头3岁的美国大象,体重250kg。
Var ele2新象(3,250kg,美国象)
现在大象有了,我们应该教大象进入容器的能力。这是方法。主要的编写方法是将这些方法编写到构造函数中。
功能象(年龄、体重、类型){
年龄
这个重量重量
这种类型
this.enterContainer函数(){}}
大象造好了,接下来要做的就是把冰箱变成一个物体。我们还为冰箱写了一个构造函数。
功能冰箱(){
}
同样,冰箱这个对象也有其独特的属性(数据),比如冰箱的高度和宽度。我们还将这些属性写入构造函数。
功能冰箱(宽、高){
宽度宽度
这个高度高度
}
现在根据需求,冰箱应该有开关门的方法,我们也写在构造函数上。
功能冰箱(宽、高){
宽度宽度
这个高度高度
函数(){}
函数(){}
}
此时,我们需要下面的代码来完成 "把大象放进冰箱 "
// 1找一个冰箱物体,这个物体的宽度和高度足以把大象放进去。
var冰箱贴新款冰箱贴(4m,4m)
// 2给冰箱一个开门指令。
()
// 3找到一个大象对象。
Var大象新象(2,200kg,非洲象)
// 4给大象下达指令进入冰箱。
elephant.enterContainer()
// 5向冰箱发出关闭指令。
()
但是这个时候,我们要实现把狮子放进冰箱的需求,就要写一段代码,描述狮子的属性和方法。而这个代码几乎和描述大象的代码一模一样。
功能狮子(年龄、体重、类型){
年龄
这个重量重量
这种类型
this.enterContainer函数(){}
}
这时我们分析,无论大象、狮子、猴子的属性(年龄、体重、各种类型)和方法(进入容器)是否相同,这些都是我们需要的动物,所以让 让我们直接写一个描述动物的代码。
功能动物(年龄、体重、类型){
年龄
这个重量重量
这种类型
this.enterContainer函数(){}
}
当我们想把大象放进冰箱时:
Var ele新动物(2,250kg,非洲象)
ele.enterContainer()
当我们想把狮子放进冰箱时:
定义变量狮子新动物(2250公斤,美洲狮)
lion.enterContainer()
此时,不需要重复编写代码来达到类似的要求。但这时有同学想说,动物中猴子会爬树,大象会 如果要求猴子爬树,can 为什么我们不在动物构造器中增加一个爬树的方法呢?这显然是合理的!
当然不是!在解决这个同学之前 的问题,让 让我们做个总结。只是为了解决我们国家把动物放进冰箱的问题,我们把面向过程的方法改成了面向对象的方法。我们在上面的代码中只使用了面向对象思想的第一个特性:封装。所谓封装,就是把对象(需求中的实体)的属性(特征)和方法(能力)抽象出来,形成一个又一个的分类。在js中,es6之前没有类的概念,所以我们用构造函数来表示每个分类。抽象对象,只要你想用,就用构造函数里的new操作,新建一个。
继承
接下来解决猴子爬树的问题。当然,要解决这个问题,我们需要利用面向对象思维的另一个特性:继承。继承意味着子类可以享受父类的属性和方法(从这里开始不再解释类似的属性基本概念)。那么什么是子类和超类呢?为了解决把动物放入冰箱的问题,我们定义了一个动物构造器,我们理解为父类,然后提出这个问题:不是所有的动物都有相同的方法。猴子会爬树,但大象会。;t .这个时候我们需要重新定义猴子的构造函数。我们把这种猴子理解为一个子类。
功能猴(年龄、体重、类型){
年龄
这个重量重量
这种类型
函数(){}
this.enterContain:原型。让 ■首先在控制台中输出一个构造函数:
console.dir(数组)
此时,在控制台中,我们可以看到数组构造函数有一个prototype属性。这个属性就是我们所说的原型。
通过扩展这个原型属性,我们发现通常使用的数组的方法都是从这个原型派生出来的。换句话说,原型方法可以由实例对象共享。
然后我们会用原型来解决猴子代码重复太多的问题。我们发现动物构造函数和猴子构造函数都需要一个函数来进入enterContainer。为了去掉这个重复的代码,我们在Animal这个相当父类的构造函数中声明,将Monkey的原型指向Animal的一个实例。
功能动物(年龄、体重、类型){
年龄
这个重量重量
这种类型
this.enterContainer函数(){
Console.log(进入容器)
}
}
功能猴(年龄、体重、类型){
年龄
这个重量重量
这种类型
函数(){}
}
新动物()
此时,我们新建了一个Monkey实例,并发现这个实例可以调用容器方法。
Var猴新猴(2,25kg,金丝猴)
monkey.enterContainer()
此时,进入容器的方法enterContainer就可以共享了。然而,这种写作有一个缺点。我们写的方法都是写在构造函数里的,每次新建对象都会在内存里声明一个函数,而且这个函数的代码每次都是一样的。这是非常不必要的。
Var m1新款猴子(1,15kg,长臂猴子)
Var m2新猴(2,25kg,猕猴)
console.log(m1,m2)
我们模仿原生js的,在原型上写方法来解决这个问题。
功能动物(年龄、体重、类型){
年龄
这个重量重量
这种类型
}
函数(){
Console.log(进入容器)
}功能猴(年龄、体重、类型){
年龄
这个重量重量
这种类型
}
新动物()
函数(){
Console.log(小猴子正在爬树)
}
首先从内存分析,对象上没有对象方法。
然后从控制台观察。
Var m1新款猴子(1,15kg,长臂猴子)
Var m2新猴(2,25kg,猕猴)
console.log(m1,m2)
()
()
这是因为我们在原型上写方法,原型上的方法可以被实例共享。M1和m2都是猴子的实例,可以调用爬树的方法。
利用构造函数实现属性继承
到目前为止,我们已经解决了一些代码的重用问题。我们发现有些代码还是重复的,这部分代码是对象的属性。
在js中,我们可以使用借用构造函数来实现属性的继承。
什么是借款?这其实是一个所有函数都可以调用的方法:call方法。另一个功能是在函数执行时修改这个点。举个简单的例子:
函数fn(){
console.log(this)
}
fn()
这段代码在正常情况下的结果是输出窗口对象。
但是如果我们用借用的方法:
函数fn(){
consol:小明})
控制台中的输出是:我们正在使用call方法的第一个参数。利用这个特性,我们可以借用构造函数。
具体代码如下
功能动物(年龄、体重、类型){
年龄
这个重量重量
这种类型
}
功能猴(年龄、体重、类型){
(这个,年龄,体重,类型)
}
至此,Monkey中重复的属性代码没有了。那么让我们 让我们试试Monkey的一个实例是否会有这些属性。
Var m1新款猴子(1,15kg,长臂猴子)
Var m2新猴(2,25kg,猕猴)
console.log(m1,m2)
所以最后,我们的代码是这样写的
功能动物(年龄、体重、类型){
年龄
这.魏重量轻
这种类型
}
函数(){
Console.log(进入容器)
}
功能猴(年龄、体重、类型){
(这个,年龄,体重,类型)
}
新动物()
函数(){
Console.log(小猴子正在爬树)
}
这个时候,如果是所有动物的方法,我们只需要回到世间。如果是猴子 我们将把它写在世界上。这就是js中实现定向的过程。我们把以上两种实现继承的称为:组合继承。
更简单的语法实现面向对象
上面写的是我们在es5标准下的面向对象过程。这个过程有点麻烦。在es6的新标准下,我们有了更简单的语法来实现面向对象。
类别关键字
首先,让我们 让我们理解es6中的一个新关键字:class,它允许我们快速定义类。语法如下:
类别类别名称{
}
然后在里面写这个类的构造函数。
类别类别名称{
构造函数(){
}
}
例如,我们定义一个动物类。
动物类{
建造者(年龄、体重、类型){
年龄
这个重量重量
这种类型
}
}
当我想要一个动物实例时,我只需要新的。
Var a1新动物(2,200kg,非洲象)
console.log(a1)
这个语法的本质也是利用函数和原型来实现的,所以我想实现前面的动物和冰箱的问题,如果在这个新的语法中实现会更快。
扩展关键字
在新语法的情况下,如果我们想实现继承,我们只需要使用一个新的关键字 "扩展 "。语法如下:
类子类扩展父类{
构造函数(){}
}
但重要的是,在这种新语法下实现的继承,必须先在子类的构造函数中调用父类的构造函数。
类子类扩展父类{
构造函数(){
超级()
}
}
所以现在要认识到Mokey 从动物的遗传,我们不得不这样写。
动物类{
建造者(年龄、体重、类型){
年龄
这个重量重量
这种类型
}
enterContainer(){
Console.log(进入容器)
}
}
类Mokey扩展动物{
建造者(年龄、体重、类型){
超级(年龄、体重、类型)
}
climbingTree(){
Consol
javascript怎么实现像其他语言一样,一个类一个文件?应该注意哪些问题?
我不 不知道有没有学过java,就每个类写一个文件,单独介绍一下。建议看看JavaScript面向对象编程。网上有很多专门介绍的文章,很全面,所以我赢了 不要在这里张贴代码。
版权声明:本文内容由互联网用户自发贡献,本站不承担相关法律责任.如有侵权/违法内容,本站将立刻删除。