调用了这么久的JS方法是长在对象、类、值本身还是原型链上?

js · 浏览次数 : 0

小编点评

JavaScript中,方法调用主要分为三种形式:对象方法、静态方法和原型方法。 1. 对象方法:直接在对象实例上调用。例如: ```javascript const obj = { a: 1, b: function () { console.log('Hello'); }, }; obj.b(); // 输出 "Hello" ``` 2. 静态方法:定义在类本身上,通过类名调用。例如: ```javascript class MyClass { static staticMethod() { console.log('This is a static method'); } } MyClass.staticMethod(); // 输出 "This is a static method" ``` 3. 原型方法:定义在类的原型上,可以通过实例调用。例如: ```javascript class MyClass { instanceMethod() { console.log('This is an instance method'); } } const obj = new MyClass(); obj.instanceMethod(); // 输出 "This is an instance method" ``` 关于基础类型值上调用方法的问题,实际上JavaScript引擎会在调用方法前将其转换为对应的包装类型。例如,当你调用一个字符串上的方法时,JavaScript引擎会先将字符串转换为String对象,然后调用其中的join方法,最后再将包装对象丢弃,还原为原始字符串。这个过程对用户来说是透明的,但实际上是发生的。 总结一下,JavaScript中的方法调用方式主要有对象方法、静态方法和原型方法。静态方法和原型方法可以通过类名或实例调用,而对象方法只能通过实例调用。此外,JavaScript会自动将基本类型值转换为包装类型,以便正确调用方法。

正文

调用了这么久的JS方法是长在对象、类、值本身还是原型链上?

JavaScript这门语言总是能带给我惊喜,在敲代码的时候习以为常的写法,退一步再看看发现自己其实对很多基操只有表面的使用,而从来没思考过为何要这样操作。

今天整理JS代码的时候突然发出灵魂三连问:

  • 为什么有些时候操作对象,可以直接调用对象上的方法,但有些时候我们使用类似Array.from()的写法?
  • 在对象上调用的方法跟在原型上调用的方法区别是什么?这两者相同么?
  • 为什么JS上可以直接在基础类型值上调用对象上面才存在的方法?基础类型值上调用的方法与在对象上调用的方法有区别么?

不同的方法调用方式

瞟了眼我的代码,立马就发现了一个调用类上方法的片段:

const obj = { a: 1 };
console.log(Object.hasOwn(obj, 'a')); // true
// 但是如果在对象上调用,则会抛不存在的错误
console.log(obj.hasOwn('a')); // TypeError: obj.hasOwn is not a function

在上面的例子里,Object.hasOwn是一个可以直接调用的方法,但令人困惑的是,当我们尝试直接在对象实例上调用hasOwn方法时,却抛出了一个类型错误,是不是有点反直觉? 我仔细想了一想突然发现,其实这只是一个基础JS概念的一个外在表现,只不过我们习惯了作为现象使用它,却很少会想到它背后的逻辑。

静态方法与实例方法

其实,我们需要做的只是区分JavaScript静态方法实例方法

静态方法 是定义在类上的方法,而不是在类的实例上,访问不到this与实例变量,所以只能通过类来调用这些方法,而不能通过一个实例来调用

class MyClass {
  static staticMethod() {
    console.log('这是个静态方法');
  }
}

MyClass.staticMethod(); // 正常执行
const myInstance = new MyClass();
myInstance.staticMethod(); // Error: myInstance.staticMethod is not a function

实例方法 是定义在类的原型上的方法,可以访问对象的属性,也可以访问this,实例化对象可以直接调用这些方法

class MyClass {
  instanceMethod() {
    console.log('这是个实例/对象方法');
  }
}

const myInstance = new MyClass();
myInstance.instanceMethod(); // 正常执行

概括来说,上面例子中Object.hasOwn()是一个需要传参的、在Object这个类上的静态方法,所以才需要在类上直接调用,而且不能在实例对象上调用;但在例如arr.sort()的调用,实际调用的是实例对象上的方法。

至于为何会做如此区分,原因是一个简单的面向对象编程需求:如果一个方法逻辑不涉及对象上的属性,但又逻辑上属于这个类,通过接受参数就可以实现功能的,则可以作为一个类的静态方法存在。但如果它需要直接访问类上属性,直接作为实例方法显然更加妥当。

原型链与方法调用

JavaScript中的每个对象都有一个原型(prototype)(除了Object.protoype也就是所有原型的尽头),对象的方法实际上是定义在原型链上的。虽然我们可能是在对象上调用了一个方法,实际上JavaScript引擎会沿着原型链查找该方法并调用。

const arr = [1, 2, 3];
console.log(arr.join('-')); // "1-2-3"
console.log(Array.prototype.join.call(arr, '-')); // "1-2-3"

上面的例子里,join方法是数组的实例方法。实例方法可以直接在数组的实例上调用,也可以通过Array.prototype.join.call的方式来调用,这俩本质上是一样的。唯一区别是Array.prototype.join.call允许我们在任何类似数组的对象上调用这个方法,哪怕它不是一个真正的数组。
等等?我们可以在不是数组的值上调用join?是的

const pseudoArray = { 0: 'one', 1: 'two', 2: 'three', length: 3 };

// ❌显然object上没有join方法,这样调用会报错
pseudoArray.join(','); // Error: pseudoArray.join is not a function

// 成功在object上调用join!!
const result = Array.prototype.join.call(pseudoArray, ',');
console.log(result); // "one,two,three"

所以,在对象上调用实例方法,等同于按照这个对象的原型链一层一层向父类上找同名方法来调用。

基础类型的自动包装

虽然其他支持面向对象编程范式的语言也有类似行为,也就是对基本类型的自动包装自动拆包,但为了百分百掌握JavaScript的行为与他们的异同,还是再来确定一遍吧

每当我们在基本类型值上(例如"hello"6)上调用方法,JavaScript引擎都会先使用基本类型对应的包装类型对值进行包装,调用对应的方法,最后将包装对象丢掉还原基础类型。这是个引擎内部的隐式操作,所以我们没有任何的感知。

JavaScript对于以下的基本类型,都有对应的包装类型。可以通过typeof操作结果是基本类型名还是object来确认:

  • string - String
  • number - Number
  • boolean - Boolean
  • symbol - Object
  • bigint - Object

让我们列一下他们基本类型对应包装类型的使用:

// string
const primitiveString = "hello";
const objectString = new String("hello");
console.log(typeof primitiveString); // "string"
console.log(typeof objectString); // "object"

// number
const primitiveNumber = 42;
const objectNumber = new Number(42);
console.log(typeof primitiveNumber); // "number"
console.log(typeof objectNumber); // "object"

// boolean
const primitiveBoolean = true;
const objectBoolean = new Boolean(true);
console.log(typeof primitiveBoolean); // "boolean"
console.log(typeof objectBoolean); // "object"

// symbol
const primitiveSymbol = Symbol("description");
const objectSymbol = Object(primitiveSymbol);
console.log(typeof primitiveSymbol); // "symbol"
console.log(typeof objectSymbol); // "object"

// bigint
const primitiveBigInt = 123n;
const objectBigInt = Object(primitiveBigInt);
console.log(typeof primitiveBigInt); // "bigint"
console.log(typeof objectBigInt); // "object"

所以,在基本类型上调用方法,等同于创建这个基本类型对应的包装类型的对象并调用方法,最后拆包并返回原始类型的值。本质上还是调用了同类型包装行为创建的对象上的方法。

"str".toUpperCase();
// 等同于
(new String("str")).toUpperCase()
// 当然,这里巧了,toUpperCase()本来也没想返回包装类型的对象

结语

哈哈,原来这个类、对象方法调用现象的原因其实一直都在我的大脑里,这只是面向对象编程中的一个很稀松平常的事实,但平时从来只是使用,还从来没联想过为何他会这样。

不知道你有没有感受到这种编程语言带来的实践经验与基础理论交融的乐趣。在一点点的实践中才会慢慢发现原来看似“这样写就能跑”的一些代码,其实背后都有曾经学习、分析过的程序概念和理论的支撑。这种感受或许就是编程快乐的其中之一个源头吧。

为大家的好奇心与耐心致敬。

与调用了这么久的JS方法是长在对象、类、值本身还是原型链上?相似的内容:

调用了这么久的JS方法是长在对象、类、值本身还是原型链上?

调用了这么久的JS方法是长在对象、类、值本身还是原型链上? JavaScript这门语言总是能带给我惊喜,在敲代码的时候习以为常的写法,退一步再看看发现自己其实对很多基操只有表面的使用,而从来没思考过为何要这样操作。 今天整理JS代码的时候突然发出灵魂三连问: 为什么有些时候操作对象,可以直接调用对

第139篇:JS数组常用方法(map(),reduce(),foreach())

好家伙,本篇为MDN文档数组方法的学习笔记 Array.prototype.reduce() - JavaScript | MDN (mozilla.org) 数组方法这块的知识缺了,补一下 1.map()方法 map() 方法创建一个新数组,这个新数组由原数组中的每个元素都调用一次提供的函数后的返

《爆肝整理》保姆级系列教程-玩转Charles抓包神器教程(8)-Charles如何进行断点调试

1.简介 Charles和Fiddler一样也有个强大的功能,可以修改发送到服务器的数据包,但是修改前需要拦截,即设置断点。设置断点后,开始拦截接下来所有网页,直到取消断点。这个功能可以在数据包发送之前,修改请求参数;在收到应答包,在js解析和浏览器渲染之前,修改返回结果。有了这个功能,开发者就可以

Web攻防--JS算法逆向--断点调试--反调试&&代码混淆绕过

# Web攻防--JS算法逆向--断点调试--反调试&&代码混淆绕过 ## JS算法逆向 在进行渗透测试过程中,在一些功能点进行参数注入或者枚举爆破等过程中,会出现参数进行加密的情况,但是我们输入参数并不是加密状态,即便测试点存在漏洞也不可能测试成功,这时候便需要将所提交参数进行加密后在进行注入,针

初探富文本之OT协同实例

初探富文本之OT协同实例 在前边初探富文本之OT协同算法一文中我们探讨了为什么需要协同、为什么仅有原子化的操作并不能实现协同、为什么要有操作变换、如何进行操作变换、什么时候能够应用操作、服务端如何进行协同调度等等,这些属于完成协同所需要了解的基础知识,实际上当前有很多成熟的协同实现,例如ot.js、

[转帖]堆和栈的概念

定义: 栈: 栈是为执行线程留出的内存空间。当函数被调用的时候,栈顶为局部变量和一些 bookkeeping 数据预留块。当函数执行完毕,块就没有用了,可能在下次的函数调用的时候再被使用。栈通常用后进先出(LIFO)的方式预留空间;因此最近的保留块(reserved block)通常最先被释放。这么

C#的关于窗体的类库方案 - 开源研究系列文章

这次想到了以前编写的关于应用的那个类库,不过当时的只是定义了显示接口,然后调用窗体显示。现在想到要把这个关于窗体的类库进行集合,统一调用,于是就把原来的代码进行了修改完善,终于得到了这次这个例子。 这个例子主要实现了4种关于窗体的形式。第1种为普通的显示文件的信息(即程序集信息里的那些信息);第2种

Beetl 源码解析:GroupTemplate 类

本文首发于公众号:腐烂的橘子 前言 Beetl 是一款 Java 模板引擎,在公司的项目中大量运用,它的作用是写通用代码时,有一些差异化的逻辑需要处理,这时可以把这些差异化的逻辑写在模板里,程序直接调用,实现了代码的低耦合。 有人问差异化的东西为什么不能通过配置实现?原因是配置只能将一些差异化的值抽

[转帖]系统调用的实现

1.概述 系统调用是内核与用户空间程序交互的接口 很多情况下,用户进程需要获得系统服务(调用系统程序),这时就必须利用系统提供给用户的“特殊接口”——系统调用了,它的特殊性主要在于规定了用户进程进入内核的具体位置;换句话说,用户访问内核的路径是事先规定好的,只能从规定位置进入内核,而不准许肆意跳入内

浅谈服务接口的高可用设计

作为一个后端研发人员,开发服务接口是我正常不过的工作了,这些接口不管是面向前端HTTP或者是供其他服务RPC远程调用的,都绕不开一个共同的话题就是“高可用”,接口开发往往看似简单,但保证高可用这块实现起来却不并没有想想的那么容易,接下来我们就看一下,一个高可用的接口是该考虑哪些内容,同时文中有不足的欢迎批评指正。