JavaScript【八】JavaScript函数高级

news2025/1/17 1:17:50

文章目录

  • 🌟前言
  • 🌟回调函数
  • 🌟递归函数
  • 🌟闭包
    • 🌟什么是闭包函数?
    • 🌟 闭包三个特性
    • 🌟 闭包的缺点
    • 🌟JavaScript的垃圾回收机制
    • 🌟闭包小案例
      • 🌟局部变量长期驻扎内存
      • 🌟避免全局变量污染
      • 🌟私有成员的存在
  • 🌟[ES6]块级作用域
    • 🌟什么是块级作用域
    • 🌟 函数声明与块级作用域
  • 🌟函数扩展
    • 🌟[ES6]函数参数的解构赋值
    • 🌟[ES6]箭头函数
      • 🌟 基本用法
      • 🌟使用注意点
  • 🌟写在最后

在这里插入图片描述

🌟前言

哈喽小伙伴们,本文将收录在JavaScript【前端童子功】这个专栏里;这个专栏里边会收录一些JavaScript的基础知识和项目实战;希望大家可以多多支持,你们的支持就是我创作的动力;让我们一起来看看吧🤘

🌟回调函数

js代码会至上而下一条线执行下去,但是有时候我们需要等到一个操作结束之后再进行下一个用户操作,这时候就需要用到回调函数来控制下一个操作的功能。

把一个函数的指针作为另一个函数的参数,当调用这个参数时,被当做参数的这个函数就叫 回调函数

function A(callback) {
    callback();
    console.log('主函数');
}

function cb(){
    console.log("回调函数")
}
A(cb)

把函数整体作为参数传进去

function A(callback) {
    callback();
    console.log('主函数');
}


A(function (){
    console.log("回调函数")
})

传参给回调函数

function A(callback) {
    var str = "传递给回调的参数";
    callback(str);
    console.log('主函数');
}

A(function (str){
    //str 是接收主函数传递的参数
    console.log("主函数传递的参数为",str);
    console.log("回调函数");
})

在执行回调的时候确保它是个函数

function Fn(options, callback) {
    if (typeof callback === "function") {
        callback(options);
    }
}

示例:封装each函数简化for循环

function each(n,callback){
    for(var i=0;i<n;i++){
        callback(i)
    }
}

each(10,function(i){
    console.log(i)
})

🌟递归函数

所谓的 递归函数 就是在函数体内调用本函数。使用递归函数一定要注意,处理不当就会进入死循环。

  • 在函数内部直接或间接引用自身。
  • 每个递归函数里必定会有终止条件。

求阶乘:

function factorial(c){
  if(c == 1){
      return c;
  }else{
      return c * factorial(c-1);
  }
}
alert(factorial(5));

多维数组遍历:

var new_array=[];
function _getChilds(data){
    if(typeof data != "object" || data == null){
        new_array.push(data);
    }else{
        getChilds(data);
    }
}
function getChilds(data){
    for(var i in data){
        _getChilds(data[i]);
    }
}
var json = {
    "aa" : {"l" : 1,"m" : 3},
    "bb" : 1,
    "cc" : "abc",
    "dd" : true,
    "ee" : null
}
getChilds(json)
console.log(new_array)

🌟闭包

闭包是JavaScript语言的一大特点,主要应用场合为:设计私有的方法和变量。

🌟什么是闭包函数?

理解闭包前,需要先理解 全局作用域和局部作用域 的区别。函数内部可以访问全局作用域下定义的全局变量,而函数外部却无法访问到函数内部定义(局部作用域)的局部变量。

  • 概念: 闭包(closure)是定义在一个外部函数内部,并且能够访问外部函数中变量的函数。
  • 原理: 作用域链

创建闭包的常见方式,就是在一个函数内部创建另一个函数并返回:

function fun(num){
    return function (){
        return num;
    }
}
var end = fun(100);
end();  //  100

上述案例中,num 是fun函数内部的变量,本来只能fun函数内部使用,但是返回的函数也有权限访问num。所以在函数外部通过调用返回的函数即可得到函数内部的变量

function out (){
   var n = 1;
   return function (){
       return n++;
} }
var fn = out();
console.log(fn()) //1
console.log(fn()) //2
console.log(fn()) //3

🌟 闭包三个特性

  1. 函数嵌套函数
  2. 函数内部可以引用外部的参数和变量
  3. 参数和变量不会被垃圾回收机制回收

🌟 闭包的缺点

闭包的缺点就是常驻内存,会增大内存使用量,使用不当很容易造成内存泄漏。 一般函数执行完毕后,局部活动对象就会被销毁,内存中仅仅保存全局作用域。但闭包的情 况不同。

🌟JavaScript的垃圾回收机制

  1. 在JavaScript中,如果一个对象不再被引用,那么这个对象就会被GC回收。
  2. 如果两个对象互相引用,而不被第三者所引用,那么这两个互相引用的对象也会被回收。

🌟闭包小案例

那么使用闭包有什么好处呢?

  1. 希望一个局部变量长期驻扎在内存中
  2. 避免全局变量的污染
  3. 私有成员的存在

🌟局部变量长期驻扎内存

function out (){
    var n = 1;
    return function (){
        return n++;
    }
}
var fn = out();
console.log(fn())
console.log(fn())
console.log(fn())

在循环中直接找到对应的元素的索引

/ <ul>
  <li>1111111</li>
  <li>1111111</li>
  <li>1111111</li>
</ul>
var lis=document.getElementsByTagName("li");
for(var i=0;i<lis.length;i++){
   (function(i){
       lis[i].onclick=function(){
           alert(i);//0 1 2
       }
   })(i)
}
for(var i=0;i<10;i++){
    setTimeout(function(){
        console.log(i);    //10
    }, 1000)
}

🌟避免全局变量污染

但我们声明一个函数的时候其实就相当于创建一个全局变量 上一个例子中总共占有2个全局变量。

函数表达式 函数自调用:

(function(){
})()
 var out = (function(){
    var n1 = 1;
    var n2 = 100;
    return function(){
        n1++;
        n2--;
        return n1+n2;
} })()
out() // 1
out() // 2

🌟私有成员的存在

模块化代码

var aa=(function(){
   var a=10;
   function aaa(){
       a++;
       alert(a);
   }
   function bbb(){
       a+=10;
       alert(a);
   }
   return {
      a:aaa,
      b:bbb
   }
})()
aa.a(); //11
aa.b(); //21

这样做的好处就是 bbb aaa 这两个函数我们在外面是访问不到的

  • JavaScript内部原理系列-闭包(Closures)

由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包,否则会造成网页的性能问题,在IE中可能导致内存泄露。

🌟[ES6]块级作用域

🌟什么是块级作用域

JS中作用域有:全局作用域、函数作用域。没有块作用域的概念。ECMAScript 6(简称ES6)中新增了块级作用域。

块作用域由 { } 包括,if语句和for语句里面的{ }都属于块作用域。

ES5没有块级作用域概念:

{
  var num = 10;
}
console.log(num)    // 10

// for循环结束后,变量i依然会存在
for(var i = 10; i<5;i++){ }
console.log(i);   // 5

ES6的块级作用域:

{
  let num = 10;
}
console.log(num)    // 报错: num is not defined

// 由于i是存在于块级作用域中,所以i只能在循环{}中使用
for(let i = 10; i<5;i++){ }
console.log(i);   // 报错: i is not defined

🌟 函数声明与块级作用域

  • ES5中,函数只能在顶层作用域和函数作用域中声明,不能在块级作用域中声明。
  • ES6中,在块级作用域内部声明的变量函数,其行为类似于let,即块级作用域外不可使用
  function fun(){
    console.log("I am outside function.")
  }
  if (true) {
    function fun() {
      console.log("I am inside function.");
    }
    fun();    // I am inside function.
  }
  fun();    // I am inside function.
  'use strict'      //es6 中的运行结果
  function fun(){
    console.log("I am outside function.")
  }
  if (true) {
    function fun() {
      console.log("I am inside function.");
    }
    fun();    // I am inside function.
  }
  fun();    // I am outside function.

🌟函数扩展

🌟[ES6]函数参数的解构赋值

ES6 允许按照一定模式,从数组和对象中提取值,对变量进行赋值,这被称为解构

只需要保证实参和形参数据类型同为数组或对象,即可进行函数参数的解构赋值:

function add([x, y]){
  return x + y;
}

add([1, 2]); // 3
function fun({num1:x, num2:y}) {
  return x + y;
}
fun({num1: 1, num2: 2})  // 3

🌟[ES6]箭头函数

🌟 基本用法

ES6允许使用“箭头”(=>)定义函数。

var f = v => v;

上面的箭头函数等同于:

var f = function(v) {
return v;
};
  • 如果箭头函数不需要参数,就使用一个()_代表参数部分。
  • 如果箭头函数需要多个参数,就使用一个()代表参数部分。
var f = () => 5;
var f = _ => 5;
// 等同于
var f = function () { return 5 };

var sum = (num1, num2) => num1 + num2;
// 等同于
var sum = function(num1, num2) {
return num1 + num2;
};

如果箭头函数的代码块部分多于一条语句,就要使用大括号将它们括起来。

var sum = (num1, num2) => {
    console.log(num2);
    console.log(num1);
return num1 + num2;
}

console.log(sum(1,2));     // 2  1  3

由于大括号被解释为代码块,所以如果箭头函数直接返回一个json,必须在对象外面加上括号。

var person = age => ({ name: "Tom", age: age});
console.log(person(21));     //Object {name: "Tom", age: 21}

箭头函数可以与变量解构结合使用。

var full = ({ name, age}) => name + '-' + age;
console.log(full({name:"Tom",age:21}));     // Tom-21

箭头函数使得表达更加简洁。

const isEven = n => n % 2 == 0;
const square = n => n * n;

箭头函数的其中一个用处是简化回调函数。

// 正常函数写法
var arr1 = [1,2,3].map(function (x) {
return x * x;
});
console.log(arr1);    //[1,4,9]

// 箭头函数写法
var arr2 = [1,2,3].map(x => x * x);
console.log(arr2);    //[1,4,9]

🌟使用注意点

  1. this对象的指向是可变的,但是在箭头函数中,它是固定的。

    function foo() {
    setTimeout(() => {
      console.log('id:', this.id);
    }, 100);
    setTimeout(function () {
      console.log("id:",this.id);
    },100)
    }
    
    var id = 21;
    foo.call({ id: 42 });
    //id: 42
    //id: 21
    
  2. 不能当做构造函数

  3. 箭头函数没有arguments对象

🌟写在最后

更多JavaScript知识以及API请大家持续关注,尽请期待。各位小伙伴让我们 let’s be prepared at all times!

✨原创不易,还希望各位大佬支持一下!
👍 点赞,你的认可是我创作的动力!
⭐️ 收藏,你的青睐是我努力的方向!
✏️ 评论,你的意见是我进步的财富!

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

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

相关文章

STM32F407串口通信

本文是基于江科大B站视频编写&#xff0c;参考各种资料进行学习。 1、USART串口协议 硬件电路&#xff08;接线&#xff09; 2、USART串口外设 常用波特率为9600、115200 必须对应特定引脚&#xff0c;才能实现通信&#xff0c;如果引脚冲突&#xff0c;看看有没有重…

数据结构和算法学习记录——层序遍历(层次遍历)、二叉树遍历的应用(输出二叉树中的叶节点、求二叉树的高度、二元运算表达式树及其遍历、由两种遍历序列确定二叉树)

目录 层序遍历 思路图解 代码实现 二叉树遍历的应用 输出二叉树中的叶节点 代码实现 求二叉树的高度 思路图解 代码实现 二元运算表达式树及其遍历 由两种遍历序列确定二叉树 层序遍历 层序遍历可以通过一个队列来实现&#xff0c;其基本过程为&#xff1a; 先根…

计算机网络P5~P7

目录 1.计算机网路性能指标 2.吞吐量 3.时延 4.时延带宽积 5.往返时间 6.利用率 7.丢包率 8.计算机体系结构 1.计算机网路性能指标 速率和带宽 速率 &#xff1a;每秒传输的比特的个数 单位 b/s kb/s mb/s gb/s 带宽&#xff1a;表示传送数据的能力 。带宽可以理解为高…

医院手术麻醉信息管理系统源码

医院手术麻醉信息管理系统源码 实现整个围术期术前、术中、术后的全数字化和信息化。 医院手术麻醉临床信息管理系统是一种基于云计算技术的信息系统&#xff0c;它可以帮助医院更好地管理手术麻醉临床信息&#xff0c;提高手术麻醉的安全性和效率。 首先&#xff0c;医院手术…

数学分析:多元微积分1

卓里奇的数学分析的好处在于直接从多元函数来入手多元微积分&#xff0c;引出矩阵&#xff0c;十分自然。 紧集的概念&#xff0c;感觉直接用闭集去理解就行&#xff0c;&#xff08;对于图形学来说&#xff09;。 多元函数的极限&#xff0c;其实和单元函数并没有什么区别。 这…

在你眨眼时,VR这样欺骗你的视觉

VR的沉浸感&#xff0c;可以让人仿佛置身于现实之外&#xff0c;还可以利用各种生物传感特性&#xff0c;来模拟一种具有存在感的虚拟空间。实际上&#xff0c;VR可以很好的欺骗人的感官&#xff0c;比如尽管你身处空间有限的房间中&#xff0c;但在VR中你仿佛已来到开阔的户外…

网络应用程序设计(idea版)——实验三:Web组件重用与JavaBeans

目录 实验预习报告 一、实验目的 二、实验原理 三、实验预习内容 实验报告 一、实验目的 二、实验要求 三、实验内容及要求 实验预习报告 一、实验目的 1. 理解静态包含和动态包含的概念&#xff0c;掌握相关指令和动作的使用&#xff1b; 2. 掌握JavaBeans的含义和…

路科验证UVM入门与进阶详解实验3

在UVM——lab3中&#xff0c;将SV环境移植到UVM的重点内容如下&#xff1a; TLM的单向、多向通信端口使用TLM的通信管道UVM的回调类型 uvm_callbackUVM的一些仿真控制函数TLM单向、多向通信 1、何为事务级、何为事务级通信&#xff08;TLM&#xff09;&#xff1f;有什么用&am…

禁用linux系统默认的22端口号

前提条件&#xff1a;在云平台上开放一个代替22的端口。 1、需查看服务器的防火墙是否开启&#xff0c; 命令&#xff1a;systemctl status firewalld.service 如果防火墙的状态为开启&#xff0c;在运行&#xff0c;需要先在防火墙增加端口 1.命令&#xff1a;firewall-cm…

【消息队列】Kafka如何实现高性能IO

我们直到Kafka是一个自称高性能的消息队列引擎&#xff0c;一般来说对于中间件的设计需要从计算、存储、网络三方面进行下手&#xff0c;而消息从产生到消费&#xff0c;也会经历多个流程&#xff0c;比如在生产者端采用异步\同步方式发送&#xff0c;采用高效的压缩算法&#…

JavaWeb03 Cookie和Session

一个网站怎么证明你来过&#xff1f; 1.首次访问时服务器给客户端一个cookie&#xff0c;下次客户端再次访问会自动携带cookie&#xff0c;注意cookie可以是多个 2.首次访问时服务器登记了客户端一系列信息&#xff0c;下次客户端再进行访问时服务器自动匹配此客户端是否访问…

Win10搭建我的世界Minecraft服务器「内网穿透远程联机」

文章目录1. Java环境搭建2.安装我的世界Minecraft服务3. 启动我的世界服务4.局域网测试连接我的世界服务器5. 安装cpolar内网穿透6. 创建隧道映射内网端口7. 测试公网远程联机8. 配置固定TCP端口地址8.1 保留一个固定tcp地址8.2 配置固定tcp地址9. 使用固定公网地址远程联机转载…

Mac远程控制工具有哪些

适用于Mac的远程控制工具有很多&#xff0c;这里我们给大家列举五个常用软件。 1、Apple Remote Desktop 苹果自带远程桌面正如其名称所承诺的那样。作为 Apple 出品的应用程序&#xff0c;您可以想象它的配置和上手是多么容易。从 App Store 下载 Apple Remote Desktop 后&a…

ThreadPoolExecutor获取原始异常

ThreadPoolExecutor作用 ThreadPoolTaskExecutor是Spring框架提供的一个线程池实现&#xff0c;它是基于Java的ThreadPoolExecutor实现的。ThreadPoolTaskExecutor可以管理线程池中的线程&#xff0c;以满足多线程并发执行任务的需要。 FutureTask作用 FutureTask的主要作用…

自动化篇 | 13 | app自动化:airtest

1 airtes简介 1.1 参考地址 http://airtest.netease.com/ # airtest官网 https://airtest.doc.io.netease.com/ # airtest操作方法 https://airtest.doc.io.netease.com/IDEdocs/faq/1_code_examples/ # 代码示例2 airtest架构 Airtest框架 3 airtest使用 3.1 打开界…

【C语言】猜数字游戏的实现

简单不先于复杂&#xff0c;而是在复杂之后。 目录 1.思路&#xff08;代码会附在最后面&#xff09; 2.随机数函数rand 1.思路&#xff08;代码会附在最后面&#xff09; 首先实现一个猜数字游戏我们先要有一个思路&#xff1a;1.产生一个随机数 2.猜数字 3.可能猜大了 4.可能…

java简单一写一个springboot入门案例

首先 我们先打开idea编辑器 然后 选择File New>Project… 项目的话 我们选择 Spring lnitializr 右边的路径要改一下 然后 选择好路径之后 一定要要在后面加上 \项目名 例如 我这里 选择了 D盘下的学习案例下的springbootDom 项目名叫 threshold 那就这样配置 java版本选…

探索【PhySO】

目录0、 初步报错情况1、230、 初步报错情况 根据原链接&#xff0c;已确定安装成功&#xff0c;运行demo时出现报错&#xff0c;不能正常运行 报错&#xff1a; .Dummy epoch time 238.366736 ms .Dummy epoch time (w duplicate elimination) 180.852427 ms (found 15/1…

【结构型模式】代理模式

文章目录优秀借鉴1、简介2、结构3、实现方式3.1、案例引入3.2、静态代理3.3、JDK动态代理3.4、CGLIB动态代理4、区别对比4.1、静态代理和动态代理4.2、JDK动态代理和CGLIB动态代理5、代理模式优缺点6、应用场景优秀借鉴 设计模式&#xff08;四&#xff09;——搞懂什么是代理…

线段树的懒标记

上次看的那个视频讲线段树的时候压根没讲懒标记&#xff0c;然后我今天去写题目直接被薄纱&#xff01;都是70分&#xff0c;剩下3个节点tml&#xff01;&#xff01;&#xff01; 懒标记 我们在修改一些区间的时候&#xff0c;按照我昨天来学的来修改要改到最下面的叶节点去…