JavaScript 中的提升

javascript 中的提升

什么是 javascript 提升?

提升是指 javascript 中在执行之前进行声明的过程。首先处理变量和函数声明。因此,即使变量在声明之前被引用,也不会导致错误,而是返回 undefined。对于函数声明,整个函数被提升,这意味着它可以在代码中定义之前使用。此过程在执行开始之前将声明放入堆栈中。

简单来说:

用 var 声明的变量在提升期间被初始化为未定义。

函数声明已完全提升,可以在编写代码之前调用。

提升过程可确保这些声明在执行堆栈中得到识别,无论它们在代码中的位置如何。

立即学习“Java免费学习笔记(深入)”;

需要注意的是,只有声明被提升,而不是赋值。作业保留在您撰写作业的同一位置。

注意:那些说在提升过程中代码移到顶部的人实际上是错误的。代码永远不会向上移动。

例子:

console.log(myvariable); // undefined      var myvariable = 10;

此代码同时执行两个操作。首先,它声明变量 myvariable 并将其提升到作用域,但其值仍然未定义。因此,console.log 命令将未初始化的值显示为未定义。之后,值 10 被分配给 myvariable。

但是,如果您尝试更改它,则不会对之前的声明产生任何影响。例如:

console.log(myvariable); // referenceerror: myvariable is not defined      myvariable = 10;

这里,我们还没有声明 myvariable 变量,因此它在尝试更改它之前会抛出 referenceerror。

总之,javascript 首先读取所有代码,然后将所有声明提升到外部,同时将赋值保留在其原始位置。这个过程称为提升。

var 和 let 也被提升:

var 和 let 都在 javascript 中被提升,但它们的行为略有不同。

变量:

当你使用 var 声明一个变量时,它的声明被提升到了作用域之外,你可以在声明之前访问该变量,但在它被赋值之前,它的值将是未定义的。

例子:

console.log(myvariable); // undefinedvar myvariable = 10;

在上面的代码中,声明 var myvariable 被提升到范围之外,但赋值 myvariable = 10 保留在原处。因此,console.log 语句输出 undefined,因为变量存在但尚未赋值。

让:

另一方面,当您使用 let 声明变量时,提升行为有点不同。变量声明已提升,但您无法在声明之前访问该变量。这被称为“暂时死区”。如果您尝试在声明之前访问 let 变量,您将收到 referenceerror。

例子:

console.log(myvariable); // referenceerror: myvariable is not definedlet myvariable = 10;

在这种情况下,let 声明被提升,但在声明之前无法访问该变量。因此,console.logstatement 会抛出 referenceerror,因为变量尚未定义。

SpeakingPass-打造你的专属雅思口语语料 SpeakingPass-打造你的专属雅思口语语料

使用chatGPT帮你快速备考雅思口语,提升分数

SpeakingPass-打造你的专属雅思口语语料 25 查看详情 SpeakingPass-打造你的专属雅思口语语料

这里基本上在let的情况下,即使提升了,变量的值仍然是未定义的。因为内存中没有空间容纳该变量。因此无法引用该地址。因为内存中没有myvariable = 10的地址,所以给出引用错误。

常量:

const 关键字允许您创建常量,这些变量一旦赋值就无法重新赋值。

示例:1

const pi = 3.142;pi = 22/7; // attempt to reassign the value of piconsole.log(pi); // output: typeerror: assignment to constant variable

在这个例子中,我们将 pi 定义为一个常量,初始值为 3.142。当我们尝试为 pi 重新分配新值时,会抛出 typeerror,因为常量无法重新分配。

示例:2

const pi;console.log(pi); // output: syntaxerror: missing initializer in const declarationpi = 3.142;

在这种情况下,我们声明一个常量 pi 而不对其进行初始化。此代码会抛出 syntaxerror,因为常量必须同时声明和初始化。

示例:3

function getcircumference(radius) {  console.log(circumference);  circumference = pi * radius * 2;  const pi = 22/7;}getcircumference(2); // referenceerror: circumference is not defined

在这里,在 getcircumference 函数中,我们尝试在声明之前访问它的周长。它会抛出 referenceerror,因为变量尚未定义。

使用const时,需要在使用变量之前声明并初始化变量。

总体而言,var 和 let 都在 javascript 中被提升,但它们的行为和临时死区的概念仅适用于 let 变量。 const 创建一旦初始化就无法重新分配的常量。

所有未声明的变量都是全局变量:

function hoisted() {a = 20;var b = 100;}hoisted();console.log(a); // 20// can be accessed as a global variable outside the hoisted() function.console.log(b);// as it is declared, it is bound within the bounds of the hoisted() function. we cannot print it outside the hoisted() function.output: referenceerror: b is not defined

函数作用域变量

我们可以看到,函数 hoist() 作用域中 var 消息变量的声明将到达函数的顶部。
为了避免这个问题,我们将确保在使用变量之前声明它。

在您提供的两个示例中,输出都将是未定义的。

示例:1

function hoist() {  console.log(message);  var message = 'hoisting is all the rage!';}hoist(); // output: undefined

示例:2

function hoist() {  var message;  console.log(message);  message = 'hoisting is all the rage!';}hoist(); // output: undefined

在这两种情况下,输出都是未定义的,因为变量被提升到各自作用域的顶部,但它们的赋值随后按照代码的原始顺序发生。

函数表达式

函数表达式不会被提升,并抛出 typeerror,因为该表达式被视为变量而不是函数。

expression(); // output: "typeerror: expression is not a functionvar expression = function() {  console.log('will this work?');};

严格模式

从它的名字来看,它是 javascript 的一个受限变体,不允许在声明变量之前使用变量。在严格模式下运行我们的代码:

通过将一些静默的 javascript 错误更改为显式抛出错误来消除它们。

修复了 javascript 引擎难以执行优化的错误。

你可能会错过声明变量的机会,使用 strict 会抛出引用错误来阻止你。

'use strict';console.log(hoist); // Output: ReferenceError: hoist is not definedhoist = 'Hoisted';

结论

在声明变量和函数时了解 javascript 中的提升非常重要,随着对提升的解释,您将了解 javascript 代码的实际处理方式。

以上就是JavaScript 中的提升的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/451813.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月7日 23:35:58
下一篇 2025年11月7日 23:36:50

相关推荐

  • c++ 作用域解析运算符怎么用

    在 c++++ 中,作用域解析运算符 (::) 用于明确指定标识符的作用域。1) 解决命名冲突,如访问全局变量 (::count)。2) 访问命名空间成员 (math::calculatearea)。3) 访问类的静态成员 (myclass::staticvar)。合理使用该运算符可以提高代码的清晰…

    2025年12月18日
    000
  • c++ 内联函数怎么使用

    c++++ 内联函数通过将函数体嵌入调用处来提升性能。1) 使用 inline 关键字定义内联函数,如 inline int add(int a, int b) { return a + b; }。2) 编译器决定是否内联,基于函数大小和复杂度。3) 适用于小型、频繁调用的函数,避免过度使用以防代码…

    2025年12月18日
    000
  • c语言函数的基本要求有和定义

    C语言函数本质上是代码模块化,封装了代码段并提供了一个名称以便重复使用。函数定义包括参数列表(值或地址传递)、返回值类型和作用域,局部变量只在函数内部有效。函数指针和回调函数允许将函数作为参数传递。性能优化建议权衡函数粒度和选择合适的参数传递方式。最佳实践强调代码可读性和可维护性,包括清晰的注释、规…

    2025年12月18日
    000
  • c语言函数名定义

    C语言函数名定义包括:返回值类型、函数名、参数列表和函数体。函数名应清晰、简洁、统一风格,避免与关键字冲突。函数名具有作用域,可在声明后使用。函数指针允许将函数作为参数传递或赋值。常见错误包括命名冲突、参数类型不匹配和未声明的函数。性能优化重点在函数设计和实现上,而清晰、易读的代码至关重要。 C语言…

    2025年12月18日
    000
  • c语言函数的嵌套调用和递归调用分别是啥意思

    C语言函数调用可分为嵌套调用和递归调用。嵌套调用是指函数内调用其他函数,层层嵌套。递归调用是指函数自身调用自身,可用于处理自相似结构问题。关键区别在于嵌套调用中的函数依次调用,相互作用域独立,而递归调用中函数不断自调用,需注意递归基和栈溢出问题。选择哪种调用方式取决于问题的具体需求和性能要求。 C语…

    2025年12月18日
    000
  • 使用gdb快速调试

    本文介绍使用GDB命令行调试程序的常用技巧。以下步骤演示如何在命令行快速调试代码: 编译代码: 使用 gcc myprogram.c -g -o myprogram 命令编译您的C程序,-g 选项生成调试信息。 启动GDB: 使用 gdb -tui myprogram 命令启动GDB,-tui 选项…

    2025年12月18日
    000
  • c语言函数定义格式有哪些

    C语言函数定义的关键元素包括:返回类型(定义函数返回的值)、函数名(遵循命名规范,决定作用域)、参数列表(定义函数接受的参数类型、数量和顺序)和函数体(实现函数的逻辑)。明确这些元素的意义和微妙关系至关重要,能帮助开发者避免“坑”,编写更高效、更优雅的代码。 C语言函数定义:那些你可能不知道的细节 …

    2025年12月18日
    000
  • 理解 C/C++ 中的 char**

    在c++和c++中,char**是指向char类型的指针。它通常用于表示字符串数组,例如命令行参数 (argv)、动态字符串数组或每行都是一个字符串的二维数组。尽管一开始会令人困惑,但通过一些示例,您将看到它的操作方式与处理“字符串表”类似。 什么是 char* *?char* 是指向 char 的…

    2025年12月18日
    000
  • C++ 中如何正确声明和使用友元类

    友元类是一种可访问其他类私有成员的特殊类。声明友元类时使用 friend 关键字,如:friend class friendclass; 在友元类中,可通过作用域解析运算符 (::) 访问友元类的私有成员:friendclass::accessprivatemember(outer); 实战案例:在…

    2025年12月18日
    000
  • C++ 函数的类方法是如何声明和实现的?

    c++++中类方法声明是在类定义中使用访问控制修饰符声明的,实现则在类定义之外,使用类名作为作用域解析运算符。例如,public类方法可在对象上调用,protected和private方法受保护或私有访问限制。 C++ 函数的类方法 声明 类方法是类的一部分,可以在对象上调用。它们在类定义中使用 p…

    2025年12月18日
    000
  • C++ 函数性能优化中内存管理的技巧

    c++++ 函数性能优化中,内存管理至关重要,可通过以下技巧提升效率:代码重复优化:使用内存池预分配对象,避免频繁内存分配/释放。智能指针:自动管理指向对象的指针,释放内存,防止内存泄漏和悬垂指针。局部变量优化:利用 raii 技术,在变量作用域结束时自动释放资源。值/引用传递:通过引用传递参数,节…

    2025年12月18日
    000
  • C++ 函数类的成员函数如何重载?

    在 c++++ 中,函数类成员函数可通过不同的参数列表进行重载,允许同一函数名称存在多个实现。重载的成员函数在类作用域内定义,可以重载标准或自定义运算符,以修改其行为。例如,在 myfunction 类中,运算符()成员函数重载为接受 int 参数返回 int 值和接受两个 double 参数返回 …

    2025年12月18日
    000
  • C++ 函数的类方法如何使用 RAII?

    c++++ 中可以通过函数模板 std::function 使用 raii(资源获取即初始化)技术,实现以下步骤:构造函数获取资源。析构函数释放资源。创建 raii 封装类,在作用域结束时释放资源。在作用域内使用资源。离开作用域后,析构函数自动释放资源。 C++ 函数的类方法如何使用 RAII RA…

    2025年12月18日
    000
  • C++ 类方法的 this 指针解析

    C++ 类方法的 this 指针解析 this 指针是在类方法中隐含定义的特殊指针,它指向调用该方法的特定对象实例。理解 this 指针至关重要,因为它允许类方法访问和修改对象的成员变量和方法。 语法 class ClassName {public: void methodName() { // 可…

    2025年12月18日
    000
  • C++ 函数的 STL 迭代器有哪些?

    C++ 函数的 STL 迭代器 STL(标准模板库)迭代器是一个通用的机制,用于遍历容器中元素。C++ 函数支持几种 STL 迭代器类型,用于不同的目的。 类型 以下是 C++ 函数支持的常见 STL 迭代器类型: 输入迭代器 (InputIterator):只支持单向遍历,元素不可改变。输出迭代器…

    2025年12月18日
    000
  • C++ 友元函数引发访问权限设置异常的解决方法

    友元函数访问权限设置异常的解决方法:声明友元函数为 friend;确保友元函数具有足够的访问权限,可通过 getter 方法访问私有成员。 C++ 友元函数引发访问权限设置异常的解决方法 友元函数是一种允许非成员函数访问类的私有成员的特殊机制。然而,有时在使用友元函数时可能会遇到访问权限设置异常。 …

    2025年12月18日
    000
  • 剖析 C++ 函数性能优化盲区,深入优化见真章

    优化 c++++ 函数性能需要识别并消除常见的盲区,包括:1. 过量内存分配;2. 复制操作;3. 函数调用开销;4. 缓存局部性;5. 分支错误预测。通过采用内存池、移动语义、内联函数、优化缓存访问和分支预测,可以显著提升函数性能。 剖析 C++ 函数性能优化盲区,深入优化见真章 优化 C++ 函…

    2025年12月18日
    000
  • 析构函数在释放资源时执行了什么操作?

    析构函数自动执行,释放对象分配的资源,包括:释放分配的内存关闭文件或连接释放外部资源(例如网络连接或数据库句柄) 析构函数:释放资源的诀窍 简介析构函数是C++中的一种特殊函数,它在对象超出作用域时自动执行。其主要目的是释放对象分配的资源,确保程序安全、有效地运行。 作用析构函数主要执行以下操作: …

    2025年12月18日
    000
  • 在 C 语言中使用函数指针时需要特别注意什么问题?

    在 c 语言中使用函数指针时,应注意函数原型匹配、函数指针的有效性、常量性、类型安全。这些注意事项可避免未定义的行为或程序崩溃。函数指针可用于实现通用排序算法,例如通过函数指针对整数数组进行快速排序。 在 C 语言中使用函数指针时应注意的注意事项 函数指针在 C 语言中是一种强大的工具,它允许您以动…

    2025年12月18日
    000
  • C++ 类方法的内存管理优化

    C++ 类方法的内存管理优化 在 C++ 中,类方法的内存管理至关重要,因为它影响着应用程序的性能和资源占用。本文将探讨优化类方法内存管理的最佳实践,并提供实际示例来说明这些技术。 局部变量 避免在类方法中分配大型局部变量,因为它们将在栈上占据内存空间。取而代之,使用堆上的动态分配来存储大型数据结构…

    2025年12月18日
    000

发表回复

登录后才能评论
关注微信