zoukankan      html  css  js  c++  java
  • JS:JS中常见的 “函数名 is not a function” 错误

    js中常见的错误,例如Uncaught TypeError: x is not a function

    其原因除了函数本身有错之外,还有一种很奇怪的情况:函数本身没有错,但是运行时就是不能正常运行。这种情况与javascript的特性有关变量与函数声明前置的优先级。

    总结:

    js有声明前置,函数和变量的声明都会前置,即会在整个js代码的最开始,不管声明部分在什么位置。

    js中函数的声明优先于同名变量的声明,不管先后顺序如何。

    函数的声明会在整段js代码的最前面,不管function() 函数在js的什么位置。

    当先声明了名为x()的函数,再声明名为var x的变量时,变量名会覆盖函数,使得在同名变量定义之后,函数变得未定义,即同名变量定义之后 调用同名函数会报错,即 x is not a function。

    当有两个同名变量和两个同名函数一起定义时,四个量的名字相同,那么后一个函数或者变量会覆盖掉前一个对应的量(函数/变量)且四个中最后一个定义量之后在引用此量,该量的类型(变量/函数)是最后一次定义此名量时的类型(变量/函数)。

    1-4点解释如下:

    首先看代码:

    console.log(x)
    console.log(x());
    var x=1;
    function x(){
        console.log(5);
    }
    console.log(x)
    console.log(x());//此时x变成了一个变量

    输出结果:

    function x(){
                    console.log(5);
     }

    5

    1
     Uncaught TypeError: x is not a function

     ----------------------------

    js解释器在对其上下文进行解释执行时分为三个阶段来进行:声明阶段、初始化阶段、执行阶段。

    针对js上下文,首先会进行声明阶段,声明阶段中的特点是声明前置;声明又会包括变量声明前置和函数声明前置,鉴于以上代码的输出结果,我们可以得出函数声明前置优先于变量声明前置的特点,并且如果变量名和函数名冲突会忽略变量的声明,因此声明过得变量名或函数名不会重复声明,这样也可以很好地解释为什么第一次输出的是函数而不是undefined。根据js的这些特点我们可以将以上代码解析成如下:

    //声明阶段
    function x(){//函数声明
        console.log(5);
            }
    var x;//变量声明,因为x已经声明过了,此处不进行声明(忽略)
    //执行阶段
    console.log(x);
    console.log(x());
    x=1;
    console.log(x);
    console.log(x());

    如上代码所述,js将变量和函数的声明前置,然后再执行代码。

    • 第二次输出时,因为声明阶段已经声明过名为x的函数,所以在执行阶段中调用x函数,会执行函数体中的内容。
    • 第三次输出时,输出1,因为x被赋值为1.
    • 第四次输出时,因为x此时是一个变量而不是一个函数,所以js无法解释“变量()”这样的格式,就会提示“x is not a function”。

    第5点解释如下:当有两个同名变量和两个同名函数一起定义时,四个量的名字相同,那么后一个函数或者变量会覆盖掉前一个对应的量(函数/变量)。

    js中声明过得变量名或函数名不会重复声明,如果js代码中有同名的函数或同名的变量时,程序如何运行,如下代码:

    console.log(x)
    console.log(x());
    var x=1;
    var x=100;
    function x(){
    console.log(5);
    }
    function x(){
    console.log(500);
    }
    console.log(x)
    console.log(x());//此时x变

    根据js解析代码的特点,将代码解析成如下:

    //声明阶段
    function x(){//函数声明
    //console.log(5);此句会被下句代码覆盖
    console.log(500);
    }
    var x;//变量声明,因为x已经声明过了,此处不进行声明(忽略)
    //执行阶段
    console.log(x);
    console.log(x());
    x=1;
    x=100;//x的值被覆盖
    console.log(x);
    console.log(x());//此时x变成了一个变量

    所以输出的结果就是:

    function x(){
                    console.log(500);
     }

    500

    100
     Uncaught TypeError: x is not a function

    ----------------------------

    所以:如果声明了同名的函数其定义会被后者覆盖,声明了同名的变量其值也会被后者覆盖

  • 相关阅读:
    SpringBoot 官方文档中文版
    SpringBoot 官方文档中文版
    SpringBoot 官方文档中文版
    RK 打包脚本 make_ext4fs system.img
    Linux安装docker-compose
    Linux使用docker安装Jenkins 并进行jar包发布
    JAVA把InputStream 转 字节数组(byte[])
    SpringBoot简单整合分布式任务调度平台(XXL-JOB)
    SpringBoot项目bootstrap.yml配置文件不加载
    JAVA中Base64和byte数组(byte[]) 相互转换
  • 原文地址:https://www.cnblogs.com/forforever/p/12374589.html
Copyright © 2011-2022 走看看