【ES6】ECMAS6新特性概览(一):变量声明let与const、箭头函数、模板字面量全面解析

news2024/11/14 13:24:09

在这里插入图片描述

🔥 个人主页:空白诗
🔥 热门专栏:【JavaScript】

在这里插入图片描述

文章目录

    • 🌿 引言
    • 一、 `let` 和 `const` - 变量声明的新方式 🌟
      • 📌 `var`的问题回顾
      • 📌 `let`的革新
      • 📌 `const`的不变之美
    • 二、 Arrow Functions - 箭头函数,简洁之美 ✨
      • 📌 语法精简
      • 📌 普通函数与`this`的动态绑定
      • 📌 箭头函数与静态`this`绑定
        • 适用场景
        • 注意事项
      • 📌 小结
    • 三、 Template Literals - 字符串模板,插值新体验 💬
      • 📌 基础插值
      • 📌 多行字符串与保留格式
      • 📌 表达式与函数
      • 📌 标签化模板字面量
      • 📌 小结
    • 四、 Default Parameters - 参数默认值,告别undefined ❓
      • 📌 基本用法
      • 📌 复合默认值与逻辑判断
      • 📌 注意事项与实践建议
      • 📌 小结
    • 总结

在这里插入图片描述

🌿 引言

ES6,作为ECMAScript 2015的简称,标志着JavaScript编程语言的一个重要进化节点。它不是渐进的变化,而是一次飞跃式的更新,为开发者带来了一系列强大的新特性与语法糖,极大提升了代码的简洁性、可读性和运行效率。从新的变量声明方式letconst,到优雅的箭头函数模板字符串,再到让对象操作更为灵活的解构赋值增强的对象字面量ES6的每项改进都旨在让JavaScript适应日益复杂的应用场景,同时保持其作为脚本语言的活力与魅力。本文是深入探索这些核心特性的起点,为你铺开一条通向高效、现代JavaScript编程实践的道路。


一、 letconst - 变量声明的新方式 🌟

ES6之前的JavaScript世界,var作为声明变量的主力军,虽功不可没,却也因其变量提升(variable hoisting)和函数作用域(function scope)的特性,给开发者带来了不少困惑和潜在错误。随着ES6的登场,letconst的引入如同一股清流,为变量管理领域带来了革新与严谨性。

📌 var的问题回顾

  • 变量提升(Variabe Hoisting): 当你在函数内部使用var声明一个变量时,这个声明会被JavaScript引擎默默地“提升”到当前作用域的最顶部。尽管这不会提升变量的赋值操作,但会导致在声明前访问变量时不会抛出ReferenceError错误,而是输出undefined。这种机制有时会导致难以预料的结果,尤其是在涉及条件或循环时。

    function illustrateVarHoist() {
        console.log(bar); // 输出:undefined,证明变量被提升了
        var bar = "I'm hoisted!";
        console.log(bar); // 输出:I'm hoisted!
    }
    illustrateVarHoist();
    
  • 作用域限制(Scope Limitations): var声明的变量遵循函数作用域(function scope)或全局作用域(global scope),而不是块级作用域(block scope)。这意味着,在一个函数内部声明的var变量对整个函数都是可见的,而在任意代码块(如if语句或for循环)中声明的var变量实际上属于包含它的函数作用域或全局作用域。这可能导致在循环或条件判断中预期之外的变量共享问题。

    for (var i = 0; i < 3; i++) {
        setTimeout(function() {
            console.log(i); // 输出:3,3,3,因为i是在同一个作用域内被修改的
        }, 1000);
    }
    

以上代码展示了由于var的作用域特性,在循环结束后每个setTimeout回调访问的i都是循环结束时的值(3),而非每次迭代时的值。这强调了使用var时可能遇到的意料之外的副作用。

📌 let的革新

  • 块级作用域(Block Scoping): 引入let关键字是ES6对变量作用域处理的一次重大改进。与var的函数作用域不同,let声明的变量严格遵循块级作用域原则。这意味着,如果你在一个特定的代码块(例如if语句、for循环或者一对大括号内)使用let声明变量,那么这个变量仅在此块内有效,出了这个块就无法访问,从而有效避免了变量泄漏到外部作用域,减少了意外的变量覆盖问题。

    {
        let y = 5;
        console.log(y); // 输出 5
    }
    // console.log(y); // 这里会报错:y is not defined,体现了let的严格块级限制
    
    for (let i = 0; i < 3; i++) {
        setTimeout(function() {
            console.log(i); // 分别输出 0, 1, 2,每一次迭代i都是新的绑定
        }, 1000);
    }
    

在上面的循环例子中,每次循环迭代都会创建一个新的i变量实例,每个setTimeout回调都能记住自己迭代时i的值,解决了使用var时的闭包陷阱问题,这是let在处理循环中的一个显著优势,极大增强了代码的逻辑清晰度和可预测性。

📌 const的不变之美

  • 常量声明(Constant Declaration): const关键字的引入,为JavaScript开发者提供了一种定义不可变变量的方式。一旦使用const声明了一个变量并赋予初始值,该变量的值便被视为固定,任何尝试修改的操作都将引发错误。这促进了代码的清晰度和稳定性,鼓励开发者明确哪些数据是程序状态中的常量,减少意外变更的风险。

    const MAX_SCORE = 100; // 游戏中的最高分,不应改变
    // MAX_SCORE = 200; // 这里会导致错误:Assignment to constant variable.
    
    // 对象和数组的特殊情况:
    const settings = { difficulty: 'hard' };
    // settings = {}; // 错误:不能重新赋值
    settings.difficulty = 'easy'; // 但是可以修改对象的属性,因为const冻结的是引用,非对象内容本身
    console.log(settings); // 输出:{ difficulty: 'easy' }
    

通过const,开发者能够明确标识出那些设计为不应更改的值,无论是基础数据类型还是对象的引用,这对于维护大型代码库尤其重要,因为它提高了代码的可读性和可维护性。结合let的块级作用域特性,两者共同为JavaScript的变量管理带来了一场革命,推动了更加严谨和高效的编程实践。

这里只做一些简单介绍,更多关于 varletconst以及变量提升作用域逻辑TDZ 的内容在之前的一篇博客:
JavaScript基础知识强化:变量提升、作用域逻辑及TDZ的全面解析


二、 Arrow Functions - 箭头函数,简洁之美 ✨

箭头函数(=>)作为ES6的一项革新特性,显著简化了函数的语法,同时以一种新颖的方式解决了长期困扰开发者的关于this关键字上下文绑定的问题,为JavaScript函数表达式赋予了新的生命和优雅。

📌 语法精简

与传统函数声明或函数表达式不同,箭头函数采用了一种更加紧凑的语法结构,摒弃了function关键字,并且在函数体仅包含一个表达式时,允许省略花括号 {}return 语句,直接返回该表达式的结果。

// 传统函数声明
function add(a, b) {
    return a + b;
}

// 箭头函数表达式
const add = (a, b) => a + b; // 直接返回求和结果

// 传统函数表达式
[1, 2, 3].map(function(x) {
    return x * x;
});

// 使用箭头函数,提升代码简洁度
[1, 2, 3].map(x => x * x);

📌 普通函数与this的动态绑定

在经典的函数实现中,this的值由函数调用的环境在运行时动态确定,这可能导致在复杂调用链或异步编程中出现预期之外的this行为。

function identify() {
    console.log(this.name);
}

const obj = { name: "Object Context", identify: identify };

// 正常情况下的`this`绑定
obj.identify(); // 输出 "Object Context"

// 独立调用时,非严格模式下的`this`通常指向全局对象
identify(); // 可能在浏览器中输出 "Window" 或在Node.js中输出 "global"

在面试时经常会遇到的一个问题是:如何改变普通函数中this的指向❓
一种是使用.call(), .apply(), 或 .bind()方法,这三种方法在本文中不做详细解释,详细区别可查看另一篇博客:JavaScript中call、apply与bind的区别
一种是手动设置that = this_this去保存当前的this值,然后在需要的地方使用这个变量
还有一种就是使用箭头函数,将在下文详细解释

📌 箭头函数与静态this绑定

箭头函数的核心变革在于其处理this的方式。它们不会创建自己的this,而是直接沿用外层(词法作用域)的this值,这意味着箭头函数内的this在定义时即被固定,调用上下文的改变不会影响到它。

const objArrow = { name: "Arrow Context" };

// 箭头函数捕获定义时的`this`
const identifyArrow = () => console.log(this.name);

// 即便函数被赋值或传递,其`this`依然绑定到定义时的环境
objArrow.identifyArrow = identifyArrow;
objArrow.identifyArrow(); // 明确输出 "Arrow Context"

// 直接调用,箭头函数的`this`遵循外层作用域
identifyArrow(); // 输出取决于外部上下文,可能是全局对象的名称
适用场景
  • 事件处理与回调:在事件监听或异步操作如setTimeoutPromise回调中,使用箭头函数可以避免this丢失问题。

  • 对象方法:虽然通常不推荐将箭头函数直接作为对象的方法(因为它无法直接绑定对象实例作为this),但在一些场景下,如需要保持外层作用域的this时,箭头函数依然有用。

  • 函数内部函数:在另一个函数内部定义的箭头函数,通常用于保留外层的this,避免闭包问题。

注意事项
  • 不可用作构造函数:箭头函数不能用new调用,因为它没有自己的this,没有原型属性。
  • argumentssuper:箭头函数不绑定自己的arguments,且不能使用super关键字,需要时考虑使用其他方案。

📌 小结

  • 普通函数this是动态的,调用时根据上下文确定,适用于需要灵活上下文的场景。
  • 箭头函数通过静态绑定this,确保了函数内部的上下文一致性,特别适合用于闭包、事件处理和异步编程等对this稳定有高要求的场景。

通过对比分析,我们深入理解了箭头函数在处理this问题上的独特优势,以及它如何通过简化语法和增强逻辑清晰度来提升JavaScript代码的质量和可维护性。在实际编码实践中,合理选择普通函数与箭头函数,能够有效提升程序的可靠性和开发效率。


三、 Template Literals - 字符串模板,插值新体验 💬

自从ES6引入模板字面量(Template Literals)以来,处理字符串和嵌入变量或表达式变得更加直观和简洁。这种以反引号(`)包围的字符串格式,不仅改善了代码的可读性,还提供了强大的插值功能,让我们能够以一种接近自然语言的方式构造复杂的字符串。

📌 基础插值

基本的插值操作通过${expression}来完成,其中expression可以是任何有效的JavaScript表达式,包括变量、运算符乃至函数调用。

const name = "Alice";
const age = 28;

// 使用模板字面量进行字符串拼接
console.log(`My name is ${name}, and I am ${age} years young.`);

// 输出:My name is Alice, and I am 28 years young.

📌 多行字符串与保留格式

除了变量插值,模板字面量还天然支持多行字符串,无需使用转义字符,这对于编写HTML片段、SQL查询或是任何需要跨多行的文本内容尤其方便。

const poem = `The woods are lovely, dark and deep,
But I have promises to keep,
And miles to go before I sleep,
And miles to go before I sleep.`;

console.log(poem);

📌 表达式与函数

模板字面量中的插值不仅仅局限于简单变量,你甚至可以直接执行函数或计算表达式,并将其结果嵌入到字符串中。

function getGreeting(hour) {
    return hour < 12 ? "Good morning" : "Good afternoon";
}

const currentHour = new Date().getHours();
console.log(`${getGreeting(currentHour)}, ${name}! How's your day going?`);

📌 标签化模板字面量

进一步地,通过“标签模板”功能,你可以自定义模板字面量的处理逻辑。这为字符串的预处理(如过滤、格式化等)提供了无限可能。

function highlight(text) {
    return text.map((part, index) => {
        if (index % 2 === 0) return part;
        else return `<strong>${part}</strong>`;
    }).join('');
}

const sentence = `Cup of coffee, slice of cake`;
console.log(highlight`I'd like a ${sentence}`);

📌 小结

模板字面量以其便捷的插值语法、多行文本的支持以及通过标签化进行的高级处理能力,彻底改变了JavaScript中字符串处理的方式。它们让代码更易于阅读和维护,同时也大大提高了开发者在构建动态内容时的灵活性和效率。掌握这一特性,对于现代JavaScript编程至关重要。


四、 Default Parameters - 参数默认值,告别undefined ❓

ES6的参数默认值功能是JavaScript函数定义的一次重要进化,它赋予了开发者为函数参数指定默认值的能力,从而在调用函数时即使未提供相应实参,也能避免undefined的困扰。此特性显著提升了代码的健壮性与灵活性,使函数调用更加优雅流畅。

📌 基本用法

在函数声明时,通过在参数名后紧接赋值运算符=及默认值即可设置参数的默认值。如下例所示,当调用greet()时未提供参数,name自动采用默认值Stranger

function greet(name = "Stranger") {
    console.log(`Hello, ${name}!`);
}

greet(); // 输出:Hello, Stranger!
greet("Alice"); // 输出:Hello, Alice!

📌 复合默认值与逻辑判断

默认值的设定远不止限于简单字面量,它能够是任何有效的JavaScript表达式,甚至是函数调用,这为参数处理引入了丰富的逻辑潜力。

function calculateArea(width, height = width * 2) {
    return width * height;
}

console.log(calculateArea(10)); // 未指定高度,使用默认值(宽度的两倍),输出:200
console.log(calculateArea(10, 5)); // 指定了高度,覆盖默认逻辑,输出:50

📌 注意事项与实践建议

  • 顺序与依赖: 如果函数有多个参数且其中某些依赖前面的参数值来计算默认值,务必确保前面的参数都有明确的值,否则默认值逻辑可能不会按预期执行。

  • 默认值共享: 注意默认值如果是引用类型(如数组、对象),所有函数调用将共享同一默认实例,修改一个实例会影响到其他调用。为避免这种情况,可以考虑使用函数来生成默认值。

  • 默认值惰性评估: 默认值在函数定义时计算,对于包含复杂计算或副作用的默认值需谨慎使用。

📌 小结

默认参数机制的加入,标志着JavaScript函数设计步入了一个更加人性化和健壮的新阶段。它不仅简化了函数接口的使用,减少了因遗漏参数导致的错误,还促进了代码的清晰度与可维护性,无疑是现代JavaScript开发中不可或缺的一环。掌握并善用此特性,将使你的代码更加稳健且优雅。


总结

本文深入探讨了ECMAS6中对JavaScript变量声明方式的重要革新,重点聚焦于letconst的引入,以及箭头函数与模板字面量的使用,这些特性如何重塑了现代JavaScript编程实践。

  • letconst声明:打破了var的限制,let带来了块级作用域,确保变量在定义块内有效,避免了变量污染;const确保不可变性,加强了代码的安全性与可读性。

  • 箭头函数:简化了函数语法,提升了可读性,解决了this绑定难题,尤其在异步编程与事件处理中的应用显著。

  • 模板面量:提供了一种动态构造字符串的新方式,支持多行文本与表达式插值,增强了代码的自然度与灵活性。

这些特性综合起来,ECMAS6不仅增强了JavaScript的严谨性与代码质量,还极大地提高了开发效率,使开发者能编写出更健壮、易维护、可读性强的应用程序。理解这些核心机制,是成为JavaScript高手的关键路径。

后续我们将继续深入了解ES6新特性,例如class类、模块、异步处理promise、生成器函数generatorsmapset数据结构…等等。当然其中有些内容在之前的博客也有详细探讨过,内容都在👉🏻 🔥 专栏:【JavaScript】。感谢大家支持~ ❤️

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1702701.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

双指针法练习题(2024/5/27)

1 反转字符串 II 给定一个字符串 s 和一个整数 k&#xff0c;从字符串开头算起&#xff0c;每计数至 2k 个字符&#xff0c;就反转这 2k 字符中的前 k 个字符。 如果剩余字符少于 k 个&#xff0c;则将剩余字符全部反转。如果剩余字符小于 2k 但大于或等于 k 个&#xff0c;则…

【管理咨询宝藏114】贝恩为某知名化妆品战略规划方案

本报告首发于公号“管理咨询宝藏”&#xff0c;如需阅读完整版报告内容&#xff0c;请查阅公号“管理咨询宝藏”。 本报告首发于公号“管理咨询宝藏”&#xff0c;如需阅读完整版报告内容&#xff0c;请查阅公号“管理咨询宝藏”。 【管理咨询宝藏114】贝恩为某知名化妆品战略…

导入 FDTD 仿真的 S 参数到 INTERCONNECT 的器件中

导入 FDTD 仿真的 S 参数到 INTERCONNECT 的器件中 正文正文 很多时候,仿真链路比较大时,我们可以将仿真的每个部分分隔开来,用 FDTD 计算出每一部分的 S 参数,然后将这些 S 参数导入 INTERCONNECT 中得到最终的仿真结果。这里我们来介绍一下这种方法。 首先,我们从右侧…

洗地机哪个品牌的质量比较好?家用洗地机品牌排行榜

随着科技的迅速发展和生活水平的不断提高&#xff0c;洗地机凭借其集吸尘、拖地和洗地于一体的技术优势&#xff0c;成为了家庭清洁的理想选择。洗地机不仅能够高效清理各种地面污渍&#xff0c;还能同时处理干湿垃圾&#xff0c;极大地提升了清洁效率。然而&#xff0c;市场上…

【SpeedAI科研小助手】2分钟极速解决知网维普重复率、AIGC率过高,一键全文降!文件格式不变,公式都保留的!

知网、维普极速降重、降AIGC率方法 非常简单&#xff0c;打开SpeedAI科研小助手&#xff0c;使用一键降重&#xff0c;上传自己的论文文件&#xff0c;等待即可。 等待弄完了之后&#xff0c;直接下载&#xff0c;可以发现word格式保持不变。直接交就完事了&#xff0c;全程2…

YOLOv10:全面的效率-准确性驱动模型设计

YOLOv10&#xff1a;全面的效率-准确性驱动模型设计 提出背景精细拆分解法双重标签分配一致的匹配度量以效率为导向的模型设计 YOLO v10 总结1. 双重标签分配策略2. 一致匹配度量策略 论文&#xff1a;https://arxiv.org/pdf/2405.14458 代码&#xff1a;https://github.com/T…

Mac 安装 PostgreSQL简易教程

Mac 安装 PostgreSQL简易教程 下载安装包 下载安装包 下载地址 我下载的文件&#xff1a;Postgres-2.7.3-16.dmg 双击 dmg 文件安装 拖拽图标到右边的文件&#xff0c;然后到应用程序中找到 Postgres.app 双击打开。 然后点击 Initialize 按钮 配置$PATH 到命令下工具&#…

智慧管网 | “数字大脑”加速“能源动脉”新升级

行业背景 我国作为全球最大的发展中国家&#xff0c;随着工业化、城镇化的发展&#xff0c;工业企业与居民对原油、天然气消费需求不断增长。而油气管网作为一组连接石油和天然气生产基地、储气库、终端市场等节点的管道网络系统&#xff0c;是油气上下游衔接协调发展的关键环…

148.栈与队列:前K个高频元素(力扣)

代码解决 class Solution { public:// 自定义比较类&#xff0c;用于优先队列&#xff08;小顶堆&#xff09;class mycomparison{public:// 重载操作符&#xff0c;用于比较两个pair&#xff0c;基于pair的第二个值&#xff08;频率&#xff09;bool operator()(const pair&l…

【机器学习】随机梯度下降算法以及优化

一、概述&#xff1a; 什么是梯度下降&#xff1f; 梯度下降法的基本思想可以类比为一个下山的过程。 假设这样一个场景:一个人被困在山上&#xff0c;需要从山上下来(i.e.找到山的最低点&#xff0c;也就是山谷)。但此时山上 的浓雾很大&#xff0c;导致可视度很低。因此&am…

全球伦敦金交易时间每天都一样吗?

伦敦金市场是一个全球化的市场&#xff0c;它全天的交易盘面由亚洲、欧洲和北美市场无缝地连接而成&#xff0c;无论来自世界上什么地方的投资者参与其中&#xff0c;都可以得到全天接近24个小时的交易行情&#xff0c;只要有足够的精力&#xff0c;根本不用担心没有交易获利的…

ResNet残差网络的学习【概念+翻译】

基于何明凯前辈论文的学习 1.主要内容&#xff08;背景&#xff09; 1、首先提了一个base&#xff1a;神经网络的深度越深&#xff0c;越难以训练。 2、原因&#xff1a;因为随着神经网络层数的增加&#xff0c;通常会遇到梯度消失或梯度爆炸等问题&#xff0c;这会导致训练变…

小预算大效果:揭秘品牌如何用创新方法实现低成本传播

说到品牌&#xff0c;我们都知道&#xff0c;没钱是真的难搞。 品牌建设就像跑马拉松&#xff0c;得慢慢来&#xff0c;持续投入&#xff0c;一点一滴积累声誉&#xff0c;这样才能培养出忠实的粉丝团。 但别急&#xff0c;就算资金紧张&#xff0c;我们也有办法让品牌慢慢站…

for循环绑定id,更新html页面的文字内容

需求&#xff1a;将方法中内容对齐 实现方式 给for循环中每个方法添加一个动态的id在DOM结果渲染完后&#xff0c;更新页面数据&#xff0c;否则会报错&#xff0c;找不到对应节点或对应节点为空 <view v-for"(item, index) in itemList" :key"index"…

Linux学习(十二)-- 用户管理与用户组管理、su与exit命令、sudo命令

目录 1. 用户管理 注&#xff1a; 以下命令需root用户执行 1.1 创建用户 1.2 删除用户 1.3 查看用户所属组 1.4 修改用户所属组 2.用户组管理 注&#xff1a; 以下命令需root用户执行 2.1 创建用户组 2.2 删除用户组 拓展&#xff1a; 3. su命令与exit命令 4. sudo…

计算机网路概述

目录 计算机网络的概念 计算机网络的定义&#xff1a; 计算机网络的组成&#xff1a; 终端系统/资源子网 通信子网 计算机网络的类型 按照拓扑分类​编辑 按照范国分类&#xff1a; 按传输方式进行分类 计算机网络体系结构 传输方式 按照传输方向区分 按照传输对象…

移动应用平台—WorkPlus企业级移动应用平台解决方案

在当今数字化和移动化的时代&#xff0c;企业需要一个强大的、灵活性高的企业级移动应用平台来连接员工、提高工作效率和创新能力。企业级移动应用平台是一种专门为企业构建和管理移动应用的解决方案&#xff0c;为企业提供了强大的功能和高度可定制的移动应用开发工具。 一、…

USB外设管理软件是什么?有哪些特别好用的USB管理软件

USB外设管理软件是什么&#xff1f;有哪些特别好用的USB管理软件 USB外设管理软件是一种专门用于监控和管理计算机上连接的USB设备的软件工具。这类软件通常提供多种功能&#xff0c;以便用户或管理员能够更好地控制和管理USB设备&#xff0c;从而提高计算机系统的安全性和工作…

HubSpot企业商机管理和销售自动化:提升业务效率的利器

在当今数字化时代&#xff0c;企业出海已成为拓展市场、增加营收的重要途径。然而&#xff0c;如何高效地管理商机和实现销售自动化&#xff0c;成为许多企业面临的挑战。HubSpot作为一款强大的营销、销售和服务自动化平台&#xff0c;为企业提供了全方位的解决方案。今天运营坛…

【数据结构】图解红黑树以及代码实现

目录 一、相关概念 性质 二、图解 1、插入操作 2、parent在左边情况1&#xff1a;cur为红色节点parent也是红色节点、uncle也为红色节点 3、parent在左边情况2&#xff1a;cur为红色节点parent也是红色节点、uncle为黑色或者是空&#xff0c;cur是parent的left 4、parent…