JavaScript进阶-高阶技巧

文章目录

  • 高阶技巧
    • 深浅拷贝
      • 浅拷贝
      • 深拷贝
    • 异常处理
      • throw抛异常
      • try/caych捕获异常
      • debugger
    • 处理this
      • this指向
      • 改变this
    • 性能优化
      • 防抖
      • 节流


高阶技巧

深浅拷贝

只针对引用类型

浅拷贝

拷贝对象后,里面的属性值是简单数据类型直接拷贝值,如果属性值是引用数据类型则拷贝的是地址
常见语法:
1.拷贝对象:Object.assgin(新变量,被拷贝对象)const 变量名 = {...对象}
2.拷贝数组:Array.prototype.concat()[...arr]
注意:如果是简单数据类型拷贝值,引用数据类型拷贝的是地址(如果是单层对象没有问题,如果是多层对象就有问题)

深拷贝

拷贝的是对象,不是地址
常见方法:

  • 通过递归实现深拷贝
    函数递归:如果一个函数在内部可以调用其本身,那么这个函数就是递归函数
    由于递归容易发生“栈溢出”错误。所以必须要加退出条件return
const obj = {
	uname:'pink',
	age:18,
	hobby:['乒乓球','足球'],
	family:{
		baby:'xiaop'
	}
}
const o = {}
// 拷贝函数
// 深拷贝,拷贝的新对象不会影响旧对象,遇到数组和对象就再次调用递归函数,先数组再对象
function deepCopy(newObj,oldObj) {
	for(let k in oldObj) {
		// 一定先写数组,在写对象,因为万物皆对象
		// 处理数组的问题
		if (oldObj[k] instanceof Array) {
			newObj[k] = []
			// 函数递归
			deeepCopy(newObj[k],oldObj[k])
		} 
		// 处理对象的问题
		else if (oldObj[k] instanceof Object) {
			newObj[k] = {}
			deepCopy(newObj[k],oldObj[k])
		} else {
			// k属性名,oldObj[k]属性值
			// newObj[k] === o.uname
			newObj[k] = oldObj[k]
		}
	}
}
deepCopy(o,obj)// 函数调用两个参数
o.age = 20
console.log(o)
o.hobby[0] = '篮球'
o.family.baby = 'oldp'
console.log(obj)
  • lodash/cloneDeep
    JavaScript库lodash里面cloneDeep内部实现了深拷贝
    语法:const deep = _.cloneDeep(object)
<!-- 先引用 -->
<script src="./lodash.min.js"></script>
<script>
	const obj = {
		uname:'pink',
		age:18,
		hobby:['乒乓球','足球'],
		family:{
			baby:'xiaop'
		}
	}
	const o = _.cloneDeep(obj)
	o.family.baby = 'oldp'
	console.log(obj)
</script>
  • 通过JSON.stringfy()实现
const obj = {
	uname:'pink',
	age:18,
	hobby:['乒乓球','足球'],
	family:{
		baby:'xiaop'
	}
}
// JSON.stringify() 把对象转换为JSON字符串
// JSON.parse() 把JSON字符串转换为对象
const o = JSON.parse(JSON.stringify(obj))
o.family.baby = 'oldp'
console.log(obj)

异常处理

异常处理是指预估代码执行过程中可能发生的错误,然后最大程度的避免错误的发生导致整个程序无法继续运行

throw抛异常

1.throw抛出异常信息,程序也会终止执行
2.throw后面跟的是错误提示信息
3.Error对象配合throw使用,能够设置更详细的错误信息

function fn(x,y) {
	if (!x || !y) {
		// throw '没有参数传递进来'
		throw new Error('没有参数传递进来')
	}
	return x + y
}

try/caych捕获异常

通过try/catch捕获错误信息(浏览器提供的错误信息)
1.try…catch用于捕获错误信息
2.将预估可能发生了错误的代码写在try代码段中
3.如果try代码段中出现错误后,会执行catch代码段,并截获到错误信息
4.finally不管是否错误,都会执行

function fn() {
	try {
		// 可能发生错误的代码要写到try
		const p = document.querySelector('.p')
		p.style.color = 'red'  
	} catch (err) {
		// 拦截错误,提示浏览器提供的错误信息,但是不中断程序的执行
		console.log(err.message)
		// 需要return中断程序,可以用throw代替
		throw new Error('错误')
	} finally {
		// 不管你程序对不对,一定会执行的代码
		alert('弹出对话框')
	}
}

debugger

在代码中需要调试的地方写debugger,打开控制台后自动来到debugger的位置,类似于断点
在这里插入图片描述

处理this

this指向

  • 普通函数
    普通函数的调用方式决定了this的值,即谁调用就指向谁
    普通函数没有明确调用者时this值为window,严格模式下没有调用者时this的值为undefined
  • 箭头函数
    箭头函数中的this与普通函数完全不同,也不受调用方式影响,事实上箭头函数中并不存在this
    1.箭头函数会默认帮我们绑定外层this的值,所以在箭头函数中this的值和外层的this是一样的
    2.箭头函数中的this引用的就是最近作用域中的this
    3.向外层作用域中,一层一层查找this,直到有this的定义
    注意:
    1.在开发中使用箭头函数前需要考虑函数中this的值,事件回调函数使用箭头函数时,this为全局变量时,this为全局的window
    因此DOM事件回调函数如果里面需要DOM对象的this,则不推荐使用箭头函数
    2.同样由于箭头函数this的原因,基于原型的面向对象也不推荐采用箭头函数

改变this

  • call()
    使用call()方法调用函数,同时指定被调用函数中的this的值
    语法:fn.call(thisArg,arg1,arg2,...)
    thisArg:在fn函数运行时指定的this值
    arg1,arg2:传递的其他参数
    返回值就是函数的返回值,因为它就是调用函数
  • apply()
    使用apply()方法调用函数,同时指定被调用函数中的this的值
    语法:fn.apply(thisArg,[argsArray])
    thisArg:在fn函数运行时指定的this值
    argsArray:传递的值,必须包含在数组里面
    返回值就是函数的返回值,因为它就是调用函数
    因此apply()主要跟数组有关系,比如使用Math.max()求数组的最大值
const obj = {
	age:18
}
function fn(x,y) {
	console.log(this)// {age:18}
	console.log(x + y)// 3
}
// 调用函数
// 改变this指向
fn.apply(obj,[1,2])
// 3.返回值 就是函数的返回值

// 使用场景:求数组最大值
const arr = [1,2,3]
const max = Math.max.apply(Math,arr)
// console.log(Math.max(...arr))
console.log(max)// 3
  • bind()(重点)
    bind()方法不会调用函数,但是能改变函数内部this指向
    语法:fn.bind(thisArg,arg1,arg2...)
    thisArg:在fn函数运行时指定的this的值
    arg1,arg2:传递的其他参数
    返回由指定的this值和初始化参数改造的原拷贝函数(新函数)
    因此当我们只是想改变this指向,并且不想调用这个函数的时候,可以使用bind,比如改变计时器内部的this指向
const obj = {
	age:18
}
function fn() {
	console,log(this)
}
// 返回值是个函数,但是这个函数里面的this是更改过的
const fn = fn.bind(obj)
// console.log(fn)
fn()// {age:18}
<body>
	<button>发送</button>
	<script>
		// 需求,有一个按钮,点击里面就禁用,2秒钟之后开始
		const btn = document.querySelector('button')
		btn.addEventListener('click', function() {
			// 禁用按钮
			this.display = true//this指向btn
			setTimeout(function() {
				// 在这个普通函数里面,要this由原来的window改为btn
				this.disabled = false
			}.bind(this),2000)//这个this是上面指向btn的this
		})
	</script>
</body>

性能优化

防抖

防抖:单位时间内,频繁触发事件,只执行最后一次
使用场景:
1.搜索框搜索输入。只需要用户最后一次输入完,再发送请求
2.手机号、邮箱验证输入检测
在这里插入图片描述
常用方法:

  • lodash提供的防抖来处理
    语法:_.debounce(func,[wait=0],[options=])
    func:要防抖的函数
    [wait=0]:需要延迟的毫秒数
    [options=]:选项对象
  • 手写防抖函数
    核心思路:利用定时器(setTimeout)来实现

例如:

<style>
	.box {
		width:500px;
		height:500px;
		background-color:#ccc;
		color:#fff;
		text-align:center;
		font-size:100px;
	}
</style>
<body>
	<div class="box"></div>
	<script src='./lodash.min.js'>
	<script>
		// 利用防抖实现性能优化
		// 需求:鼠标在盒子上移动,里面的数字就会变化+1
		const box = document.querySelector('.box')
		let i = 1
		function mouseMove() {
			box.innerHTML = i++
			// 如果里面存在大量消耗性能的代码,比如DOM操作、数据处理,可能造成卡顿
		}
		
		// 添加事件
		// box.addEventListener('mousemove',mouseMove)
		
		// 利用lodash库实现防抖 500毫秒之后+1
		box.addEventListener('mousemove',_.debounce(mouseMove,500))
		
		// 手写函数部分
		// 1.声明定时器变量
		// 2.每次鼠标移动(事件触发)的时候都要先判断是否有定时器,如果有先清除以前的定时器
		// 3.如果没有定时器,则开启定时器,存入到定时器变量里面
		// 4.定时器里面写函数调用
		function debounce(fn,t) {
			let timer
			// return返回一个匿名函数
			return function() {
				if(timer) {
					clearTimeout(timer)
				}	
				timer = setTimeout(function() {
					fn()//加小括号调用fn函数
				},t)	
			}
		}
		box.addEventListener('mousemove',debounce(mouseMove,500))
	</script>
</body>

节流

节流:单位时间内,频繁触发事件,只执行一次
使用场景:
高频事件:鼠标移动mousemove、页面尺寸缩放resize、滚动条滚动scroll
在这里插入图片描述
常用方法:

  • lodash提供的节流函数来处理
    语法:_.throttle(func,[wait=0],[options=])
    func:要节流的函数
    [wait=0]:需要节流的毫秒数
    [options=]:选项对象
    [options.leading=true]:指定调用在节流开始前
    [options.leading=false]:指定调用在节流结束后
  • 手写一个节流函数来处理
    核心思路:利用定时器(setTimeout)来实现

例如:

<style>
	.box {
		width:500px;
		height:500px;
		background-color:#ccc;
		color:#fff;
		text-align:center;
		font-size:100px;
	}
</style>
<body>
	<div class="box"></div>
	<script src='./lodash.min.js'>
	<script>
		// 利用节流实现性能优化
		// 需求:鼠标在盒子上移动,里面的数字就会变化+1
		const box = document.querySelector('.box')
		let i = 1
		function mouseMove() {
			box.innerHTML = i++
			// 如果里面存在大量消耗性能的代码,比如DOM操作、数据处理,可能造成卡顿
		}
		
		// 添加事件
		// box.addEventListener('mousemove',mouseMove)
		
		// 利用lodash库实现节流 500毫秒之后+1
		box.addEventListener('mousemove',_.throttle(mouseMove,500))

		// 手写函数部分
		// 1.声明一个定时器变量
		// 2.当鼠标每次滑动都先判断是否有定时器,如果有定时器则不开启新定时器
		// 3.如果没有定时器则开启定时器,记得存到变量里面
		// 3.1定时器里面调用执行的函数
		// 3.2定时器里面要把定时器清空
		function throttle(fn,t) {
			let timer = null
			return function() {
				if(!timer) {
					timer = setTime(function() {
					fn()
					// 清空定时器
					// 在setTimeout中是无法删除定时器的,因为定时器还在运作,故不使用clearTimeront(timer)
					timer = null
					},t)
				}
			}
		}
		box.addEventListener('mousemove',throttle(mouseMove,500))
	</script>
</body>

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

matlab 写入格式化文本文件

目录 一、save函数 二、fprintf函数 matlab 写入文本文件可以使用save和fprintf函数 save输出结果: fprintf输出结果: 1.23, 2.34, 3.45 4.56, 5.67, 6.78 7.89, 8.90, 9.01 可以看出fprintf输出结果更加人性化,符合要求,下面分别介绍。 一、save函数 …

【前端寻宝之路】学习如何使用HTML实现简历展示和填写

&#x1f308;个人主页: Aileen_0v0 &#x1f525;热门专栏: 华为鸿蒙系统学习|计算机网络|数据结构与算法 ​&#x1f4ab;个人格言:“没有罗马,那就自己创造罗马~” #mermaid-svg-iJ3Ou0qMGFVaqVQq {font-family:"trebuchet ms",verdana,arial,sans-serif;font-siz…

哈希表是什么?

一、哈希表是什么&#xff1f; 哈希表&#xff0c;也称为散列表&#xff0c;是一种根据关键码值&#xff08;Key value&#xff09;直接进行访问的数据结构。它通过把关键码值映射到表中一个位置来访问记录&#xff0c;从而加快查找速度。这个映射函数叫做散列函数&#xff08…

【字符串相加】

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言 字符串相加 方法一&#xff1a; 方法二&#xff1a; 总结 前言 世上有两种耀眼的光芒&#xff0c;一种是正在升起的太阳&#xff0c;一种是正在努力学习编程的…

给你N个整数,要求删除最大和最小的数之后按原顺序输出。

给你N个整数&#xff0c;要求删除最大和最小的数之后按原顺序输出。 输入输出格式 输入描述: 第一行输入一个整数N&#xff0c;N<100。 第二个输入N个整数。 输出描述: 按题意输出。#include <iostream> using namespace std; int main() {int n;cin >> n;int…

Vue2->3

Vue2->3 认识Vue31. Vue2 选项式 API vs Vue3 组合式API2. Vue3的优势 使用create-vue搭建Vue3项目1. 认识create-vue2. 使用create-vue创建项目 熟悉项目和关键文件组合式API - setup选项1. setup选项的写法和执行时机2. setup中写代码的特点3. <script setup>语法糖…

Scratch 第十六课-弹珠台游戏

第十六课-弹珠台游戏 大家好&#xff0c;今天我们一起做一款弹珠台scratch游戏&#xff0c;我们也可以叫它弹球游戏&#xff01;这款游戏在刚出来的时候非常火爆。小朋友们要认真学习下&#xff01; 这节课的学习目标 物体碰撞如何处理转向问题。复习键盘对角色的控制方式。…

ubuntu环境下docker容器详细安装使用

文章目录 一、简介二、ubuntu安装docker1.删除旧版本2.安装方法一3. 安装方法二&#xff08;推荐使用&#xff09;4.运行Docker容器5. 配置docker加速器 三、Docker镜像操作1. 拉取镜像2. 查看本地镜像3. 删除镜像4. 镜像打标签5. Dockerfile生成镜像 四、Docker容器操作1. 获取…

AtCoder ABC343 A-D题解

Problem A: 签到题。 #include <bits/stdc.h> using namespace std; int main(){int A,B;cin>>A>>B;for(int i0;i<10;i){if(i!(AB))cout<<i<<endl;//记得return} } Problem B: 依旧签到。 include <bits/stdc.h> using namespace …

实用工具:实时监控服务器CPU负载状态并邮件通知并启用开机自启

作用&#xff1a;在服务器CPU高负载时发送邮件通知 目录 一、功能代码 二、配置开机自启动该监控脚本 1&#xff0c;配置自启脚本 2&#xff0c;启动 三、功能测试 一、功能代码 功能&#xff1a;在CPU负载超过预设置的90%阈值时就发送邮件通知&#xff01;邮件内容显示…

求阶乘。。

&#xff01;&#xff01;&#xff01;答案解释摘录自蓝桥云课题解 问题描述 满足N!的末尾恰好有个0的最小的N是多少? 如果这样的N不存在输出-1。 输入格式 一个整数 K 输出格式 一个整数代表答案 样例输入 2 样例输出 10 import os import sys# 请在此输入您的代码 def coun…

懒人必备|视频号片段提取实战教程!

你是否也为如何提取视频号的视频感到困扰&#xff1f;想要留住那些美好瞬间&#xff0c;但又不知道改如何操作&#xff1f;别瞎找了&#xff01;今天就让我来教你正确的步骤&#xff0c;让你轻松成为“提取达人”&#xff01; 首先&#xff0c;打开想要提取的视频&#xff0c;找…

某u盘 对比 sd卡+读卡器

部分 u盘 性能甚至不如 读卡器SD卡 电脑 支持USB3 gen2 的 USBA接口(u盘用) 和 typec接口(读卡器用) 极致性能需求可考虑: m.2固态硬盘盒m.2固态 设备 速度对比 迅雷... 拷贝文件信息 共用格式化信息

青少年软件编程(Python)等级考试试卷(一级)2020年3月

青少年软件编程&#xff08;Python&#xff09;等级考试试卷&#xff08;一级&#xff09;2020年3月 第 1 题 【单选题】 运行下方代码段&#xff0c;输出的是( )。 print("a"*3) A :a3 B :3a C :a a a D :aaa 正确答案:D 试题解析 第 2 题 【单选题】 下…

SpringBoot整合rabbitmq-重复消费问题

说明&#xff1a;重复消费的原因大致是生产者将信息A发送到队列中&#xff0c;消费者监听到消息A后开始处理业务&#xff0c;业务处理完成后&#xff0c;监听在告知rabbitmq消息A已经被消费完成途中中断&#xff0c;也就时说我已经处理完业务&#xff0c;而队列中还存在当前消息…

【C++】类的默认成员函数(上)

&#x1f525;博客主页&#xff1a; 小羊失眠啦. &#x1f3a5;系列专栏&#xff1a;《C语言》 《数据结构》 《C》 《Linux》 《Cpolar》 ❤️感谢大家点赞&#x1f44d;收藏⭐评论✍️ 文章目录 一、默认成员函数二、构造函数构造函数的概念及特性 三、析构函数析构函数的特性…

#QT(DEMO2-登录界面)

1.IDE&#xff1a;QTCreator 2.实验&#xff1a;DEMO登录 3.记录 Line Edit输入不换行 密码框输入如下设置: 运行效果 4.代码

使用Matplotlib绘制圆环图

圆环图是饼图的修改版&#xff0c;中间区域被切掉。圆环图更关注使用弧的面积来以最有效的方式表示信息&#xff0c;而不是饼图&#xff0c;饼图更关注比较切片之间的比例面积。圆环图在空间方面更有效&#xff0c;因为圆环图内部的空白空间可用于显示有关圆环图的一些附加信息…

【C++那些事儿】深入理解C++类与对象:从概念到实践(中)| 默认构造函数 | 拷贝构造函数 | 析构函数 | 运算符重载 | const成员函数

&#x1f4f7; 江池俊&#xff1a; 个人主页 &#x1f525;个人专栏&#xff1a; ✅数据结构冒险记 ✅C那些事儿 &#x1f305; 有航道的人&#xff0c;再渺小也不会迷途。 文章目录 1. 类的6个默认成员函数2. 构造函数2.1 概念2.2 特性 3. 析构函数3.1 概念3.2 特性 4. 拷贝…

flutter 文字一行显示,超出换行

因为app有多语言&#xff0c;中文和其他语言长度不一致&#xff0c;可能导致英文会很长。 中文样式 英文样式 代码 Row(mainAxisAlignment: MainAxisAlignment.end,crossAxisAlignment: CrossAxisAlignment.end,children: [Visibility(visible: controller.info.fee ! null,ch…