浅谈Javascript中的对象和继承


Posted in Javascript onApril 19, 2019

Javascript是一门函数式编程语言,Javascript当中函数是核心,在Javascript中函数也是对象,函数对象在创建的时候会被添加属性和方法。

在Javascript中函数对象有两种调用方式,一种是new关键字的调用,另一种是没有new关键字的调用,前者会返回一个对象,后者会返回return语句中的内容。

function Obj (name) {
 this.name = name;
 return name;
}

用new关键字来调用如下:

var obj = new Obj('张三') // 返回 { 'name': '张三' }

不用new关键字调用如下:

var obj = Obj('张三') // 返回 '张三'

说完了js当中的对象和调用方式,那让我们来理解下什么是对象。

什么是对象?

对象是类的一个实例(对象可不是女朋友),有状态和行为。例如:一个电脑就是一个对象,它的状态有:大小、颜色、品牌等,他的行为有:播视频、听音乐、聊天等。

而类是对象的抽象,可以理解为类是一个模板,它来描述一类对象的状态和行为。软件对象也有状态和行为,软件对象的状态就是属性,行为就是方法。在软件开发中,在方法中可以操作对象的内部状态,对象的相互调用也是通过方法来完成。

类既然可以理解为一个模板,我们通过一个Person的简单例子来理解下:

function Person (name, age, sex) {
 this.name = name;
 this.age = age;
 this.sex = sex;
 run = function () {
 console.log('Run')
 }
}

 在这个类中name,age,sex为这个类的属性,而run为这个类的方法;Person的职责是构造对象,进行对象的初始化。

接下来我们看下在js中如何声明并调用对象。

对象的创建(多种方法)

块级对象

var person = {
 name: '王小端coder',
 age: 29
}
console.log(person.name); // 王小端coder

构造函数 --- 系统自带

var obj = new Object();
obj.name = '王小端coder'

console.log(obj.name); // 王小端coder

系统自带的对象有:Object、Array、Number、Boolean、Date等

构造函数 --- 自定义

function Obj (name) {
 this.name = name;
}
var obj = new Obj('王小端coder');
console.log(obj.name); // 王小端coder

对象的增删改查

增:为对象添加一个属性

var person = {};
person.name = '王小端coder';
console.log(person.name);

删:通过delete操作符删除一个对象的属性

var person = {
 name: '王小端coder'
};
console.log(person.name); // 王小端coder
delete person.name;
console.log(person.name); // undefined

改:直接通过赋值来修改一个对象的属性

var person = {
 name: '王小端coder'
};
console.log(person.name); // 王小端coder
person.name = '王小端JS'
console.log(person.name); // 王小端JS

查:查询对象的属性,有两种方法

var person = {
 name: '王小端coder'
};
console.log(person['name']); // 第一种方法
console.log(person.name); // 第二种方法

原型的定义

原型是function对象的一个属性,它定义了构造函数制造出的对象的公共祖先。通过构造函数产生的对象,可以继承该原型的属性和方法,原型也是对象。

function Person (name, age) {
 this.name = name;
 this.age = age;
}

Person.prototype = {
 eat: function (food) {
 console.log('eat ' + food);
 },
 sleep: function () {
 console.log('sleeping');
 }
}
var person = new Person ('王小端coder', 29);
console.log(person.name); // 王小端coder
person.eat('apple') // eat apple

我们定义了一个Person构造函数,而属于Person构造对象共有的方法,则定义到了Person原型上。

查看构造函数原型的接口:隐士属性__proto__(其实我们能够访问原型的属性,或者说继承原型,靠的就是__proto__属性连接着构造函数和原型,可以说没有__proto__属性的存在,就无法实现原型的继承)直接通过new操作符创建的对象访问__proto__属性即可查看到原型。

原型链

原型链就是将一个个原型串联起来,形成一条原型继承的链子。Child继承Parent, Parent继承GrandParent, 而GrandParent没有自定义原型,所以默认为原型链的最顶端new Object();

对象的继承(多种方式)

创建的子类将继承超类的所有属性和方法,包括构造函数及方法的实现。记住,所有属性和方法都是公用的,因此子类可直接访问这些方法。子类还可添加超类中没有的新属性和方法,也可以覆盖超类的属性和方法。和其他功能一样,JavaScript实现继承的方式不止一种。这是因为JavaScript中的继承机制并不是明确规定的,而是通过模仿实现的。这意味着所有的继承细节并非完全由解释程序处理。作为开发者,你有权决定最适用的继承方式。

我们先定义一个父类

// 定义一个父类
function Person (name) {
 this.name = name;
 this.sleep = function () {
 console.log('朕正在睡觉!');
 }
}
// 在父类原型上面增加一个方法
Person.prototype.eat = function (food) {
 console.log('朕现在吃的食物是:' + food);
}

下面为您介绍几种具体的继承方式:

原型链继承

// 定义一个婴儿来继承人的父类
function Baby () {
}
Baby.prototype = new Person();
Baby.prototype.name = '小端';

var baby = new Baby()
console.log(baby.name); // 小端
baby.eat('milk'); // 朕现在吃的食物是:milk

优点: 简单、易于实现;父类新增的原型方法和属性子类都能访问到;

缺点: 无法实现多继承;来自原型的对象的所有属性被所有实例共享;如果为子类增加属性和方法,无法放倒构造器中;创建子类实例时无法向父类构造函数传参;

构造继承

// 定义一个婴儿来继承人的父类
function Baby (name) {
 Person.call(this);
 this.name = name || 'Coder'
}

let baby = new Baby();
console.log(baby.name); // Coder
baby.sleep(); // 朕正在睡觉!
console.log(baby instanceof Person); // false
console.log(baby instanceof Baby); // true

优点: 可以实现多继承;创建子类时可以向父类传递参数;子类可以共享父类引用的属性;

缺点: 实例不是父类的实例;只能继承父类的实例属性和方法;无法实现函数复用,每个子类都有父类实例函数的副本,影响性能;

实例继承

function Baby (name) {
 var instance = new Person();
 instance.name = name || 'Coder';
 return instance;
}
let baby = new Baby();
console.log(baby.name); // Coder
baby.sleep() // 朕正在睡觉!

优点: 不限制调用方式,返回的对象具有相同的效果;

缺点: 不支持多继承;实例是父类的实例,不是子类的实例;

拷贝继承

// 定义一个婴儿来继承人的父类
function Baby (name) {
 var person = new Person();
 for (let p in person){
  Baby.prototype[p] = person[p];
 }
 Baby.prototype.name = name || 'Coder';
}
let baby = new Baby();
console.log(baby.name); // Coder
baby.sleep() // 朕正在睡觉!

优点: 支持多继承;

缺点: 效率较低,内存占用高;无法获取父类不可枚举的方法;

注意:不可枚举的方法是不能使用for in访问到的方法

组合继承

// 定义一个婴儿来继承人的父类
function Baby (name) {
 Person.call(this);
 this.name = name || 'Coder';
}
Baby.prototype = new Person();
Baby.prototype.constructor = Baby;

let baby = new Baby();
console.log(baby.name); // Coder
baby.eat('milk'); // 朕现在吃的食物是:milk

优点: 不存在引用属性共享问题,可以传参,函数可复用;

缺点: 调用两次构造函数,生成了两份实例;

寄生组合继承

// 定义一个婴儿来继承人的父类
function Baby (name) {
 Person.call(this);
 this.name = name || 'Coder';
}
(function () {
 let Super = function () {};
 Super.prototype = Person.prototype;
 Baby.prototype = new Super();
})();

let baby = new Baby();
console.log(baby.name); // Coder
baby.eat('milk'); // 朕现在吃的食物是:milk

优点: 整合了以上几种继承的优点;

缺点: 实现复杂;

ES6的class

ES6 提供了更接近传统语言的写法,引入了 Class(类)这个概念,作为对象的模板。通过class关键字,可以定义类。基本上,ES6的class可以看作只是一个语法糖,它的绝大部分功能,ES5 都可以做到,新的class写法只是让对象原型的写法更加清晰、更像面向对象编程的语法而已。

可以像如下定义:

class Person {
 constructor (name) {
  this.name = name;
 }
 sleep () {
  console.log('朕正在睡觉!');
 }
 eat (food) {
  console.log('朕现在吃的食物是:' + food);
 }
}

let person = new Person('小端');
console.log(person.name); // 小端
person.eat('milk'); // 朕现在吃的食物是:milk

ES6 class的继承

// 定义一个婴儿来继承人的父类
class Baby extends Person {
 constructor(name){
  super(name);
 }
}
let baby = new Baby('小端');
console.log(baby.name); // 小端
baby.eat('milk'); // 朕现在吃的食物是:milk

我们可以看到Class可以通过extends关键字实现继承,这比ES5的通过修改原型链实现继承,要清晰和方便很多。

我之后会单独用一章来详细讲ES6中的Class和继承。

以上所述是小编给大家介绍的Javascript中的对象和继承详解整合,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Javascript 相关文章推荐
基于jQuery的js分页代码
Jun 10 Javascript
jQuery插件实现屏蔽单个元素使用户无法点击
Apr 12 Javascript
关于Javascript作用域链的八点总结
Dec 06 Javascript
Jquery-1.9.1源码分析系列(十一)之DOM操作
Nov 25 Javascript
深入浅析Extjs中store分组功能的使用方法
Apr 20 Javascript
jquery实现瀑布流效果 jquery下拉加载新数据
Dec 12 Javascript
JavaScript实现body内任意节点的自定义属性功能示例
Sep 18 Javascript
Angular.js通过自定义指令directive实现滑块滑动效果
Oct 13 Javascript
解决select2在bootstrap modal中不能正常使用的问题
Aug 09 Javascript
Next.js实现react服务器端渲染的方法示例
Jan 06 Javascript
微信小程序个人中心的列表控件实现代码
Apr 26 Javascript
使用vant的地域控件追加全部选项
Nov 03 Javascript
详解如何在Vue项目中导出Excel
Apr 19 #Javascript
vue cli使用融云实现聊天功能的实例代码
Apr 19 #Javascript
详解vue中使用微信jssdk
Apr 19 #Javascript
vue router导航守卫(router.beforeEach())的使用详解
Apr 19 #Javascript
this.$toast() 了解一下?
Apr 18 #Javascript
Vue-input框checkbox强制刷新问题
Apr 18 #Javascript
vue axios封装及API统一管理的方法
Apr 18 #Javascript
You might like
浅析PKI加密解密 OpenSSL
2013/07/01 PHP
CI(CodeIgniter)框架中的增删改查操作
2014/06/10 PHP
win平台安装配置Nginx+php+mysql 环境
2016/01/12 PHP
thinkphp5使html5实现动态跳转的例子
2019/10/16 PHP
js apply/call/caller/callee/bind使用方法与区别分析
2009/10/28 Javascript
JavaScript初学者建议:不要去管浏览器兼容
2014/02/04 Javascript
JavaScript中的toLocaleLowerCase()方法使用详解
2015/06/06 Javascript
详解jQuery中的元素的属性和相关操作
2015/08/14 Javascript
基于jQuery Ajax实现上传文件
2016/03/24 Javascript
JavaScript中输出信息的方法(信息确认框-提示输入框-文档流输出)
2016/06/12 Javascript
JS实现滑动门效果的方法详解
2016/12/19 Javascript
基于jQuery实现照片墙自动播放特效
2017/01/12 Javascript
Openlayers显示瓦片网格信息的方法
2020/09/28 Javascript
微信小程序中target和currentTarget的区别小结
2020/11/06 Javascript
Vue——解决报错 Computed property "****" was assigned to but it has no setter.
2020/12/19 Vue.js
初步解析Python中的yield函数的用法
2015/04/03 Python
python常见数制转换实例分析
2015/05/09 Python
Python 文件管理实例详解
2015/11/10 Python
详解使用pymysql在python中对mysql的增删改查操作(综合)
2017/01/18 Python
基于使用paramiko执行远程linux主机命令(详解)
2017/10/16 Python
Python寻找两个有序数组的中位数实例详解
2018/12/05 Python
NumPy 基本切片和索引的具体使用方法
2019/04/24 Python
python获取点击的坐标画图形的方法
2019/07/09 Python
python通过http下载文件的方法详解
2019/07/26 Python
Python之指数与E记法的区别详解
2019/11/21 Python
Python操控mysql批量插入数据的实现方法
2020/10/27 Python
六道php面试题附答案
2014/06/05 面试题
安全生产投入制度
2014/01/29 职场文书
违纪检讨书2000字
2014/02/08 职场文书
学习党的群众路线剖析材料
2014/10/09 职场文书
初婚初育证明范本
2015/06/18 职场文书
休学证明范本
2015/06/19 职场文书
2016年“节能宣传周”活动总结
2016/04/05 职场文书
Python实现单例模式的5种方法
2021/06/15 Python
分析JVM源码之Thread.interrupt系统级别线程打断
2021/06/29 Java/Android
浅谈MySQL函数
2021/10/05 MySQL