箭头函数的意义和函数的二义性

简介: 这篇文章介绍了箭头函数的特点以及与普通函数的区别。它指出了箭头函数没有this、arguments对象和prototype原型的特点,因此不能作为构造函数使用。文章还解释了函数的二义性问题,即函数可以有多种调用方式,这在JS设计上存在缺陷。最后,文章指出箭头函数不再依赖面向对象的概念,因此没有this和原型的概念。

前言

说到箭头函数,可能很多人的第一反应就是和普通函数的区别:

  1. 箭头函数没有 this,普通函数的 this 指向依赖它是如何被调用的
  2. 箭头函数没有 arguments 对象,而是通过剩余参数(rest parameters)来获取所有参数的值
  3. 箭头函数没有 prototype 原型,不能用作构造函数,而普通函数可以
  4. 更加简洁的函数语法

相信一名前端开发者,对于这两者的区别,多多少少都能罗列一些,但是你们有没有想过,JS已经有普通函数了,为什么还要有箭头函数?难道仅仅只是语法简洁可读性好这么简单?

答案只有一个:消除函数的二义性

函数的二义性

那什么是函数的二义性呢?

我们创建一个普通函数:

function user(){
   }

这个时候就会出现歧义,因为这个函数有两种调用方式:

function user(){
   };

//  普通方式调用
user();

// 当做构造函数调用
new user();

这就是函数的二义性,至于为什么会出现这种情况,这个就要追溯到 JS 的历史渊源了,我感觉应该是 JS 这门语言在设计上的缺陷。

因为函数的二义性,导致 JS 函数的复杂度直线上升,因为函数在创建的时候,创建者不知道未来的调用者如何调用,可能直接调用,也有可能通过 new 方法调用,这就会存在很大的安全隐患。

后来开发者就在函数的命名上定义了一套规范,普通函数首字母小写,构造函数首字母大写,就像下面这样:

//  普通函数
function user(){
   };

// 构造函数
function User(){
   };

但毕竟这个不是强约制性的,普通函数依旧可以使用 new 调用,这只能说在一定程度上稍微缓解了这个问题,就像 JS 一些构造函数一样,两种调用方式都可以,比如:

//  Number
Number();
new Number()

// Date
Date();
new Date();

如果你想普通函数不能通过 new 来调用,你可以这么做:

function User(){
   
  if(new.target){
   
    throw('Uncaught TypeError: User is not a constructor')
  }
}

所以调用者压根就不清楚函数的调用方式,这个函数的二义性,会给开发者造成心智负担。
官方一直都知道这个问题,只是一直没解决,后来ECMAScript 6在给 JS 打补丁的时候,引入了两个概念:

  1. 箭头函数
  2. class实例

它们的作用都是为了消除函数的二义性

箭头函数只能这样调用:

const user = () => {
   };

// 报错 Uncaught TypeError: user is not a constructor
const a = new user();

// 正确
user();

class 只能这样调用:

class User{
   };

// 报错 Uncaught TypeError: Class constructor User cannot be invoked without 'new'
const user = User();

// 正确
const user = new User();

箭头函数

那为什么箭头函数里边没有this和原型?

const user = () => {
   };
console.log(user.prototype); // undefined

因为箭头函数跟实例无关,跟面向对象没关系,它已经脱离了面向对象的范畴,而 this 哪来的?this 来自于面向对象里面的概念,箭头函数里面没有原型的概念是一样的。

相关文章
|
开发框架 算法 .NET
一个简单高效低内存的.NET操作Excel开源框架 - MiniExcel
一个简单高效低内存的.NET操作Excel开源框架 - MiniExcel
314 0
|
监控 自动驾驶 安全
物联网卡在车联网中的作用
物联网卡(IoT SIM卡)在车联网中发挥着至关重要的作用,它是连接车辆与互联网、车辆与车辆(V2V)、车辆与基础设施(V2I)之间的桥梁。以下详细阐述了物联网卡在车联网中的几个主要作用:
|
前端开发 JavaScript 开发者
别再只用普通函数了!箭头函数的四大神奇区别,让你的代码飞起来!
【8月更文挑战第23天】在Web前端开发中,JavaScript的箭头函数(引入于ES6)提供了一种比传统函数更加简洁的定义方法。箭头函数使用 "=>" 替代 "function" 关键字,并且自动绑定外部 "this" 上下文,避免了传统函数中 "this" 值因调用方式不同而变化的问题。此外,箭头函数不拥有自己的 "arguments" 对象,但可以通过剩余参数语法获取所有参数。需要注意的是,箭头函数不能作为构造函数使用。理解这些差异有助于开发者编写更高效、清晰的代码。
493 0
|
11月前
|
前端开发 开发者
如何理解 package.json 中的 proxy 字段?
`package.json` 中的 `proxy` 字段用于配置代理服务器,帮助前端开发中解决跨域问题及模拟后端响应。其基本概念、使用场景及配置方法将在本文中详细探讨,助力开发者高效调试与测试。
352 4
|
Java
Java源文件声明规则详解
Java源文件的声明规则是编写清晰、可读且符合语法规范的Java程序的基础。这些规则包括文件名必须与公共类名相同、包声明位于文件顶部、导入声明紧跟其后、类声明需明确访问级别,并允许使用注释增强代码可读性。一个源文件可包含多个类,但只能有一个公共类。遵循这些规则有助于提升代码质量和维护性。
291 28
|
Java Linux 测试技术
JMeter 介绍与安装
Apache JMeter 是一款基于Java的开源性能和负载测试工具,常用于测试Web应用、Web服务、数据库及其他网络服务的性能。它具备跨平台特性,支持Windows、Mac及Linux系统,并可通过插件进行扩展。JMeter不仅能模拟大量用户访问以测试服务器承压能力,还适用于接口测试,支持分布式部署与UI及命令行操作模式。
|
关系型数据库 MySQL 数据库连接
win10关于mysql5.7数据库连接不上的解决方法
win10关于mysql5.7数据库连接不上的解决方法
375 0
|
监控 供应链 数据挖掘
ERP系统中的成本控制与降低成本策略解析
【7月更文挑战第25天】 ERP系统中的成本控制与降低成本策略解析
1120 3
|
移动开发 小程序 API
uniapp中uview组件库丰富的ActionSheet 操作菜单使用方法
uniapp中uview组件库丰富的ActionSheet 操作菜单使用方法
859 1
|
前端开发
ElementPlus中total出现el.pagination.total,显示总数没有出现怎样解决,出现的是英文,不是中文如何解决,这里如何配置中文,配置中文
ElementPlus中total出现el.pagination.total,显示总数没有出现怎样解决,出现的是英文,不是中文如何解决,这里如何配置中文,配置中文