笔记来源:尚硅谷最新版JavaScript基础全套教程完整版(140集实战教学,JS从入门到精通)_哔哩哔哩_bilibili

文章目录

  • 构造函数与原型对象
    • 1、使用工厂方法创建对象
    • 2、构造函数
      • 构造函数的执行流程
      • 构造函数修改
    • 3、原型对象
      • 原型prototype
      • hasOwnProperty
      • 原型的原型
      • toString
    • 4、垃圾回收(GC)

构造函数与原型对象

1、使用工厂方法创建对象

function createPerson(name, age, gender){// 创建一个新的对象var obj=new Object();//向对象中添加属性obj.name = name;obj.age = age;obj.gender = gender;obj.sayName = function(){console.log(this.name);};//将新的对象返回return obj;
}var obj1 = createPerson("孙悟空", 1000, "男");
var obj2 = createPerson("猪八戒", 3600, "男");
var obj3 = createPerson("沙悟净", 10000, "男");obj1.sayName(); // 孙悟空
obj2.sayName(); // 猪八戒
obj3.sayName(); // 猪八戒

使用工厂方法创建的对象,使用的构造函数都是Object

所以创建的对象都是Object这个类型,就导致我们无法区分出多种不同类型的对象

2、构造函数

创建一个构造函数,专门用来创建Person对象的构造函数就是一个普通的函数

创建方式和普通函数没有区别,不同的是构造函数习惯上首字母大写构造函数

和普通函数的区别就是调用方式的不同

  • 普通函数是直接调用
  • 构造函数需要使用new关键字来调用
function Person(){console.log(this); // Person{}
}
// 普通函数
var fun = Person();
console.log(fun); // undefined
// 构造函数
var person = new Person();
console.log(person); // Person{}

构造函数的执行流程

  1. 立刻创建一个新的对象
  2. 将新建的对象设置为函数中this,在构造函数中可以使用this来引用新建的对象
  3. 逐行执行函数中的代码
  4. 将新建的对象作为返回值返回
function Dog(){}function Person(name, age, gender){//向对象中添加属性this.name = name;this.age = age;this.gender = gender;this.sayHello = function(){console.log("My'name is " + this.name + ", " +"I'm " + this.age + " years old, " +"and I'm a " + this.gender + ".");};
}var person1 = new Person("孙悟空", 1000, "man");
var person2 = new Person("猪八戒", 3600, "man");
var person3 = new Person("沙悟净", 10000, "man");
var dog = new Dog();
person1.sayHello(); // My'name is 孙悟空, I'm 1000 years old, and I'm a man.
person2.sayHello(); // My'name is 猪八戒, I'm 3600 years old, and I'm a man.
person3.sayHello(); // My'name is 沙悟净, I'm 10000 years old, and I'm a man.
console.log(person1); // Person {name: "孙悟空", age: 1000, gender: "man", sayHello: ƒ}
console.log(person2); // Person {name: "猪八戒", age: 3600, gender: "man", sayHello: ƒ}
console.log(person3); // Person {name: "沙悟净", age: 10000, gender: "man", sayHello: ƒ}
console.log(typeof person1); // object
console.log(typeof person2); // object
console.log(typeof person3); // object

使用同一个构造函数创建的对象,我们称为一类对象,也将一个构造函数称为一个类。

我们将通过一个构造函数创建的对象,称为是该类的实例

使用instanceof可以检查一个对象是否是一个类的实例语法:对象 instanceof 构造函数

如果是则返回true,否则返回false

console.log(person1 instanceof Person); //true
console.log(person2 instanceof Person); //true
console.log(person3 instanceof Person); //true
console.log(dog instanceof Person);     //false

所有的对象都是Object的后代,所以任何对象和Object进行instanceof检查时都会返回true

console.log(person1 instanceof Object); //true
console.log(person2 instanceof Object); //true
console.log(person3 instanceof Object); //true
console.log(dog instanceof Object);     //true

this的情况:

  • 当以函数的形式调用时,thiswindow
  • 当以方法的形式调用时,谁调用方法this就是谁
  • 当以构造函数的形式调用时,this就是新创建的那个对象

构造函数修改

创建一个Person构造函数

在Person构造函数中,为每一个对象都添加了一个sayName方法,目前我们的方法是在构造函数内部创建的

也就是构造函数每执行一次就会创建一个新的sayName方法也是所有实例的sayName都是唯一的

function Person(name, age, gender){this.name = name;this.age = age;this.gender = gender;this.sayHello = function(){console.log("My'name is " + this.name + ", " +"I'm " + this.age + " years old, " +"and I'm a " + this.gender + ".");};
}

这样就导致了构造函数执行一次就会创建一个新的方法,执行10000次就会创建10000个新的方法,而10000个方法都是一模一样的

这是完全没有必要,完全可以使所有的对象共享同一个方法

function Person(name, age, gender){this.name = name;this.age = age;this.gender = gender;this.sayHello = fun;
}
// 将sayName方法在全局作用域中定义
function fun(){console.log("My'name is " + this.name + ", " +"I'm " + this.age + " years old, " +"and I'm a " + this.gender + ".");
};

将函数定义在全局作用域,虽然节省了空间,但却污染了全局作用域的命名空间

而且定义在全局作用域中也很不安全

3、原型对象

原型prototype

我们所创建的每一个函数(不论是普通函数还是构造函数),解析器都会向函数中添加一个属性prototype

function Person(){}function MyClass(){}console.log(Person.prototype);
// {constructor: ƒ}
//      constructor: ƒ Person()
//          arguments: null
//          caller: null
//          length: 0
//          name: "Person"
//          prototype: {constructor: ƒ}
//          __proto__: ƒ ()
//          [[FunctionLocation]]: 09-原型对象.html:8
//          [[Scopes]]: Scopes[1]
//      __proto__: Object
console.log(Person.prototype == MyClass.prototype); // false

当函数以普通函数的形式调用prototype时,没有任何作用

当函数以构造函数的形式调用prototype时,它所创建的对象中都会有一个隐含的属性,指向该构造函数的原型对象,我们可以通过__proto__来访问该属性

var mc1 = new MyClass();
var mc2 = new MyClass();
var mc3 = new MyClass();
console.log(mc1.__proto__ == MyClass.prototype); // true
console.log(mc2.__proto__ == MyClass.prototype); // true
console.log(mc3.__proto__ == MyClass.prototype); // true

原型对象就相当于一个公共区域,所有同一个类的实例都可以访问到这个原型对象

我们可以将对象中共有的内容,统一设置到原型对象中

// 向MyClass中添加属性a
MyClass.prototype.a = "123";
console.log(mc1.a);  // 123
// 向MyClass中添加方法sayHello
MyClass.prototype.sayHello = function(){alert("hello");
}
mc3.sayHello();

当我们访问对象的一个属性或方法时,它会先在对象自身中寻找,如果有则直接使用,如果没有则会去原型对象中寻找,如果找到则直接使用

mc2.a = "456";
console.log(mc2.a);  // 456

以后我们创建构造函数时,可以将这些对象共有的属性和方法,统一添加到构造函数的原型对象中

这样不用分别为每一个对象添加,也不会影响到全局作用域,就可以使每个对象都具有这些属性和方法了

hasOwnProperty

function MyClass(){}
MyClass.prototype.name = "I'm prototype's name.";
var mc = new MyClass();
mc.age = 18;
// 使用in检查对象中是否含有某个属性时,如果对象中没有但是原型中有,也会返回true
console.log("name" in mc); // true
console.log("age" in mc); // true
// 可以使用对象的hasOwnProperty()来检查对象自身中是否含有该属性
// 使用该方法只有当对象自身中含有属性时,才会返回true
console.log(mc.hasOwnProperty("name")); // false
console.log(mc.hasOwnProperty("age"));  // true
console.log(mc.hasOwnProperty("hasOwnProperty"));  // false

那么,hasOwnProperty是原型对象中定义的方法吗?

因为对象中没有定义hasOwnProperty方法,那应该就是在原型对象中定义的了,果真如此吗?

我们用hasOwnProperty方法看下有没有hasOwnProperty它自己

console.log(mc.__proto__.hasOwnProperty("hasOwnProperty"));  // false

我们发现,原型对象中也没有hasOwnProperty方法,那hasOwnProperty究竟是哪里来的呢?

原型的原型

原型对象也是对象,所以它也有原型,当我们使用一个对象的属性或方法时

  • 会先在自身中寻找,自身中如果有则直接使用

  • 如果没有则去原型对象中寻找,有则使用

  • 如果没有则去原型的原型中寻找,直到找到Object对象的原型

  • Object对象的原型没有原型,如果在Object中依然没有找到,则返回undefined

    console.log(mc.helloWorld);  // undefined
    

那么,按照这个原理,我们在原型的原型中使用hasOwnProperty方法看看

console.log(mc.__proto__.__proto__.hasOwnProperty("hasOwnProperty"));  // true

那既然原型对象有原型,那原型的原型还有原型吗?

话不多说,直接打印看下

console.log(mc.__proto__.__proto__.__proto__);  // null

根据上述原理,mc.__proto__.__proto__就是Object对象了

Object对象虽然没有原型,但也有__proto__,只是为null而已

toString

当我们直接在页面中打印一个对象时,事件上是输出的对象的toString()方法的返回值(这里并非视频中所说的那样,有待确认

如果我们希望在输出对象时不输出[object Object],可以为对象添加一个toString()方法

function Person(name, age, gender){this.name = name;this.age = age;this.gender = gender;
}
var per1 = new Person("孙悟空", 1000, "man");
var per2 = new Person("猪八戒", 3600, "man");
// 当我们直接在页面中打印一个对象时,事件上是输出的对象的`toString()`方法的返回值
console.log(per1); // Person {name: "孙悟空", age: 1000, gender: "man"}
console.log(per1.toString()); // [object Object]
// 如果我们希望在输出对象时不输出`[object Object]`,可以为对象添加一个`toString()`方法
per1.toString = function(){return "Person[name=" + this.name + ", age=" + this.age + ", gender=" + this.gender + "]";
}
console.log(per1); // Person {name: "孙悟空", age: 1000, gender: "man", toString: ƒ}
console.log(per1.toString()); // Person[name=孙悟空, age=1000, gender=man]

上述只是修改per1对象的toString方法,不会对其他对象产生影响

如果想要所有对象都执行该方法,可以修改Person原型的toString

console.log(per2.toString()); // [object Object]
// 修改Person原型的toString
Person.prototype.toString = function(){return "Person[name=" + this.name + ", age=" + this.age + ", gender=" + this.gender + "]";
}
console.log(per2.toString()); // Person[name=猪八戒, age=3600, gender=man]

4、垃圾回收(GC)

就像人生活的时间长了会产生垃圾一样,程序运行过程中也会产生垃圾这些垃圾积攒过多以后,会导致程序运行的速度过慢

所以我们需要一个垃圾回收的机制,来处理程序运行过程中产生垃圾

当一个对象没有任何的变量或属性对它进行引用,我们将永远无法操作该对象

此时这种对象就是一个垃圾,这种对算过多会占用大量的内存空间,导致程序运行变慢

在JS中拥有自动的垃圾回收机制,会自动将这些垃圾对象从内存中销毁,我们不需要也不能进行垃圾回收的操作

我们需要做的只是要将不再使用的对象设置null即可

var obj = new Object();
// ...
obj = null

【JS从入门到精通】08-构造函数与原型对象相关推荐

  1. 【Vue学习笔记】尚硅谷Vue2.0+Vue3.0全套教程丨vue.js从入门到精通

    尚硅谷Vue2.0+Vue3.0全套教程丨vue.js从入门到精通 1.Vue核心部分 1.1 Vue简介 1.1.1 Vue是什么? Vue是一套用于构建用户界面的渐进式JavaScript框架. ...

  2. 前端-Vue.js从入门到精通基础笔记(理论+实操+知识点速查)

    #[2022.3]尚硅谷Vue.js从入门到精通基础笔记(理论+实操+知识点速查) 前言 本文完全基于 参考资料:加工整理而成,包括其代码,案例,资源等.前置知识是学习尚硅谷的视频教程,本文配合其教程 ...

  3. [转]JavaScript构造函数及原型对象

    JavaScript中没有类的概念,所以其在对象创建方面与面向对象语言有所不同. JS中对象可以定义为"无序属性的集合".其属性可以包含基本值,对象以及函数.对象实质上就是一组没有 ...

  4. 深入理解Javascript中构造函数和原型对象的区别

    在 Javascript中prototype属性的详解 这篇文章中,详细介绍了构造函数的缺点以及原型(prototype),原型链(prototype chain),构造函数(constructor) ...

  5. JavaScript 进阶 35 -- 构造函数、原型对象、实例之间的关系详解

    说到继承 ,先要弄明白 :构造函数.原型对象.实例之间的关系. 先来了解几个简单的概念: 构造函数:通过 关键字 new 来调用的 函数,var p = new Person():Person就称为构 ...

  6. 面向对象;构造函数;原型对象

    构造函数 构造函数 1.什么是构造函数 2.为什么要使用构造函数 如何封装一个构造函数 3.构造函数的执行过程 4.构造函数的返回值 5.与普通函数的区别 5.1调用方式的不同 5.2 返回值不同 5 ...

  7. JavaScript进阶-编程思想、构造函数的原型对象、对象原型、原型继承以及原型链

    编程思想 面向过程 面向过程就是分析出解决问题所需要的步骤,然后用函数把这些步骤一步一步实现,使用的时候再一个一个的依次 调用就可以了. 优点: 性能比面向对象高,适合跟硬件联系很紧密 的东西,例如单 ...

  8. javascript面向对象系列第一篇——构造函数和原型对象

    前面的话 一般地,javascript使用构造函数和原型对象来进行面向对象编程,它们的表现与其他面向对象编程语言中的类相似又不同.本文将详细介绍如何用构造函数和原型对象来创建对象 构造函数 构造函数是 ...

  9. javascript面向对象精要学习总结(第四章 构造函数和原型对象)

    constructor 实例的构造函数属性 它是实例的构造函数属性,指向创建它的构造函数 不要用它来检测对象的类型,因为它可以被覆盖,并不准确 检测对象类型最好使用 instanceof functi ...

最新文章

  1. sqlrelay mysql_数据库连接池SQL Relay安装使用-Java架构师必看
  2. 3.C#中泛型类的进一步探讨
  3. PHP 实现Office word 关键词添加背景色
  4. TCP拥塞控制算法 — CUBIC的补丁(四)
  5. 操作系统【连续式分配方式、隐式链接、显示链接、索引方式、混合索引、位示图、成组链接】
  6. pycharm上传代码到github
  7. 开机发现超级管理员账户不见了
  8. 泛型类java_Java泛型 - 通用类( Generic Classes)
  9. windows 系统文件 —— 特殊文件及文件类型
  10. 智能交通灯linux代码实现,基于Linux的智能交通灯控制器设计
  11. Winform自动升级系统的设计与实现(源码)
  12. 将bilibili缓存文件转换为MP4格式
  13. 计算机退出域后无法加域,win10退出域后无法再加入域
  14. ajax怎么会突然出现401,当jquery ajax遇上401请求
  15. 小米系统shell_分享自用小米手机ADB命令删除系统内置应用
  16. 广东财经大学理工科毕业论文word模版
  17. HTAP混合事务/分析数据库调研
  18. matlab已知函数表达式画函数图像,怎么用matlab画已知函数表达式的一个函数图像?函数比较复杂的……...
  19. 记一次使用可调电源修复饿死严重亏电的电动车电瓶
  20. 放下一切包袱,轻装前行

热门文章

  1. Ubuntu虚拟机关机之后 重新打开黑屏或者花屏
  2. 自然语言处理深度生成模型相关资源、会议和论文分享
  3. 论文:Threat of Adversarial Attacks on Deep Learning in Computer Vision: A Survey翻译工作
  4. 网络安全——使用反弹木马进行提权获取主机Shell
  5. 在centos7下安装python3.7.9并搭建scrapy2环境
  6. linux c 出现报错warning: implicit declaration of function ‘atoi’问题
  7. Ubuntu20.04+MAVROS+PX4+Gazebo保姆级安装教程
  8. 7款安全测试自动化工具推荐!
  9. TI C6000 TMS320C6678 DSP+ Zynq-7045的PS + PL异构多核案例开发手册(3)
  10. [转]睡五分钟等于六钟头的方法