详解javaScript的事件中,复杂数据类型的传参(数组,对象,函数)

本文涉及的产品
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
简介: 详解javaScript的事件中,复杂数据类型的传参(数组,对象,函数)

文章目录

在JavaScript这门编程语言学习中,有太多关键问题,比如如何传参,什么是变量提升,js代码预编译是怎么回事等等。要想成为一名优秀的js高手,这些内容是你不得不去认真研究和深挖的。今天我们就来具体看看这几件事,它到底是怎么回事儿。


一、何谓预编译,变量提升?

其实这一概念是javaScript非常基础而本质的内容,通过对一门编程语言 底层封装的一些内容、概念的深入研究。我们可以逐步接近语言的逻辑,本源,也能更好的理解编译这门语言的人当时的思路,这也算是跨越时空的对话了。


在js代码执行之前,会有一段时间。这段时间是用来给js代码做准备的。那么这段时间,我们称之为预编译时间。 而这段预编译时间,编译器干的最重要的事情就是给js代码进行变量提升,函数提升,以及形参实参的统一。



换句话说,女孩子出门游玩之前会有一段时间,这段时间是用来给她自己出去游玩做准备的。这段时间发生的最重要的事就是,化妆,换衣服等等。


预编译期间会将变量声明和函数声明提升到对应作用域的最顶端,只提升声明,不提升值!


如果是全局变量,则提升到全局作用域最顶端。如果是局部的变量,则提升到局部作用域的最顶端。


eg1:

      oo();
    function oo(){  
      console.log(`oo`);
    }
    // 此函数为普通函数,这样的函数预解析时有函数提升,所以可以在函数声明之前调用
    // 如果是下面这种写法,则预解析时,会被转化成上面这样,
    function oo(){  
      console.log(`oo`);
    }
    oo();

eg2

    var a = function(){
      console.log(`哈哈`);
    }
    a();
    // 函数表达式,只能在声明之后调用
    // 如果是var声明的函数表达式,具有变量提升的特点.但是只会返回undefined,因为只提升声明,不提升值
    // 如果是let,const声明的函数表达式,则不具备变量提升的特点.这是var特有的
    // 如果是上面这种写法,则预解析会被转化成下面这样
    var a;
    a = function(){
      console.log(`哈哈`);
    }
    a();

我相信js预解析,以及函数提升,变量提升到这应该是说明白了


二、复杂数据类型的传递

本人上一篇博客已经非常详细的解释了形参,实参,以及简单数据的传参。今天看看复杂数据类型的传递。


传参实质就是传实参给形参,最终让形参等于实参即可,形参用来接收实参的值,如此而已。


1.数组

代码如下:

<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <title></title>
  </head>
  <body>
    <div onclick="fun([1,2,'hh'],[45,'',47])">兔年大吉</div>
    <script>
    var a = 1;
    var b =8;
    function fun(a,b){
      console.log(a,b);
    }
    fun(a,b); 
    fun(3);
    </script>
  </body>
</html>

以上代码,我们是将数组[1,2,'hh'],[45,'',47]作为参数,进行传递的。


那么对应的a=[1,2,'hh'],而b=[45,'',47]


这个以数组作为参数传递,只有在点击的时候才会执行


另外一个问题就是,此函数被调用了两次,在不点击调用的情况下,自执行两次。

第一次执行fun(a,b)输出结果为1   8

此时a,b作为形参,而形参a,b两个变量被声明且赋值了,相当于初始化,分别给了a,b一个值,因此打印出来1   8

第二次执行fun(3)  输出结果为   3     undefined

这是因为,此时3作为实参,传给形参(a,b),但实参只有一个,而形参有两个。所以相当于给a赋值为3,b只声明未赋值,为undefined

总结:


形参和实参是一一对应关系,如同映射

当实参个数等于形参个数,则输出正确结果

当实参个数多于形参个数,只取形参个数

当实参个数小于形参个数,多出的形参只声明未赋值,属于未定义

  function full(person) {
        return person + '2' + person;
      }
  console.log(full());

上面输出 undefined2undefined

  function full(person) {
        return person + '2' + person;
      }
  console.log(full(1));

上面输出121

2.对象

代码如下:

<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <title></title>
  </head>
  <body>
    <div onclick="fn({name:'张三'})">法外狂徒</div>
    <script>
    function fn(n){
      console.log(n);
    }
    </script>
  </body>
</html>

这个就是点击事件以对象作为实参,进行传参的

当点击div以后,n={name:“张三”},这是一种赋值操作,只不过n作为变量没有被声明罢了

3.函数

这里我要重点说明一下


上面这两种写法,是完全不一样的,

A盒子表示将函数fk()的值作为参数进行传参

B盒子表示将函数fk作为参数进行传参

<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <title></title>
  </head>
  <body>
    <div onclick="fm(fk())" >哈哈哈</div>
    <div onclick="fm(fk)" >哈哈哈</div>
    <script>
    function fk(){
      console.log(3);
      return 10;
    }
    function fm(result){  
      console.log(result);
      console.log( typeof result);              
      fk()                              
      result()
      console.log( result.prototype === fk.prototype );
    }
    </script> 
  </body>
</html>


第二个div,在这里将fk函数作为实参进行传参的时候,本质上fk是一个变量;

fk() = result();

result作为形参相当于一个声明,利用变量名调用函数;

result相当于   var  result = function fk(){};
总结

js预解析是非常重要的,这里面注意var与const和let的区别

以数组,对象等数据类型传参,记得注意参数之间存在一一对应关系

可以在调用函数的时候,把括号里面的内容看作实参,但本质上是变量。函数本身在某些场景下也可以理解为未声明的变量,但这仍然需要深度思考。函数作为实参是怎么传的,怎么执行的。文章只能给大家作为参考,最重要还是要自己深度思考。

创作不易,如果对你有帮助,希望不吝一键三连。


相关文章
|
1月前
|
JavaScript 前端开发
如何在 JavaScript 中使用 __proto__ 实现对象的继承?
使用`__proto__`实现对象继承时需要注意原型链的完整性和属性方法的正确继承,避免出现意外的行为和错误。同时,在现代JavaScript中,也可以使用`class`和`extends`关键字来实现更简洁和直观的继承语法,但理解基于`__proto__`的继承方式对于深入理解JavaScript的面向对象编程和原型链机制仍然具有重要意义。
|
1月前
|
Web App开发 JavaScript 前端开发
如何确保 Math 对象的方法在不同的 JavaScript 环境中具有一致的精度?
【10月更文挑战第29天】通过遵循标准和最佳实践、采用固定精度计算、进行全面的测试与验证、避免隐式类型转换以及持续关注和更新等方法,可以在很大程度上确保Math对象的方法在不同的JavaScript环境中具有一致的精度,从而提高代码的可靠性和可移植性。
|
5天前
|
存储 JavaScript 前端开发
JavaScript中的数据类型以及存储上的差别
通过本文的介绍,希望您能够深入理解JavaScript中的数据类型及其存储差别,并在实际编程中灵活运用这些知识,以提高代码的性能和稳定性。
22 3
|
29天前
|
JSON 前端开发 JavaScript
JavaScript中对象的数据拷贝
本文介绍了JavaScript中对象数据拷贝的问题及解决方案。作者首先解释了对象赋值时地址共享导致的值同步变化现象,随后提供了五种解决方法:手动复制、`Object.assign`、扩展运算符、`JSON.stringify`与`JSON.parse`组合以及自定义深拷贝函数。每种方法都有其适用场景和局限性,文章最后鼓励读者关注作者以获取更多前端知识分享。
18 1
JavaScript中对象的数据拷贝
|
1月前
|
JavaScript 前端开发 Java
[JS]同事:这次就算了,下班回去赶紧补补内置函数,再犯肯定被主管骂
本文介绍了JavaScript中常用的函数和方法,包括通用函数、Global对象函数以及数组相关函数。详细列出了每个函数的参数、返回值及使用说明,并提供了示例代码。文章强调了函数的学习应结合源码和实践,适合JavaScript初学者和进阶开发者参考。
42 2
[JS]同事:这次就算了,下班回去赶紧补补内置函数,再犯肯定被主管骂
|
29天前
|
存储 JavaScript 前端开发
js中的数据类型
JavaScript 中的数据类型包括五种基本类型(String、Number、Undefined、Boolean、Null)和三种引用类型(Object、Array、Function,以及ES6新增的Symbol)。基本类型直接存储值,引用类型存储的是指向实际数据的内存地址。了解它们的区别对于掌握 JavaScript 的变量赋值和函数传参至关重要。
21 1
|
1月前
|
前端开发 JavaScript 开发者
除了 Generator 函数,还有哪些 JavaScript 异步编程解决方案?
【10月更文挑战第30天】开发者可以根据具体的项目情况选择合适的方式来处理异步操作,以实现高效、可读和易于维护的代码。
|
1月前
|
JavaScript 前端开发 图形学
JavaScript 中 Math 对象常用方法
【10月更文挑战第29天】JavaScript中的Math对象提供了丰富多样的数学方法,涵盖了基本数学运算、幂运算、开方、随机数生成、极值获取以及三角函数等多个方面,为各种数学相关的计算和处理提供了强大的支持,是JavaScript编程中不可或缺的一部分。
|
2月前
|
JavaScript 前端开发
JavaScript 函数语法
JavaScript 函数是使用 `function` 关键词定义的代码块,可在调用时执行特定任务。函数可以无参或带参,参数用于传递值并在函数内部使用。函数调用可在事件触发时进行,如用户点击按钮。JavaScript 对大小写敏感,函数名和关键词必须严格匹配。示例中展示了如何通过不同参数调用函数以生成不同的输出。
|
2月前
|
存储 JavaScript 前端开发
JS函数提升 变量提升
【10月更文挑战第6天】函数提升和变量提升是 JavaScript 语言的重要特性,但它们也可能带来一些困惑和潜在的问题。通过深入理解和掌握它们的原理和表现,开发者可以更好地编写和维护 JavaScript 代码,避免因不了解这些机制而导致的错误和不一致。同时,不断提高对执行上下文等相关概念的认识,将有助于提升对 JavaScript 语言的整体理解和运用能力。
下一篇
DataWorks