javascript基础修炼(13)——记一道有趣的JS脑洞练习题【华为云技术分享】
【摘要】 开脑洞,也巩固基础知识
示例代码托管在:http://www.github.com/dashnowords/blogs
博客园地址:《大史住在大前端》原创博文目录
一. 题目
改造下面的代码,使之输出0 - 9,写出你能想到的所有解法。
首先作为前端开发者,你起码得知道下面的代码会输出什么,强烈建议自己动手试试能写出多少种解法。
for (var i = ; i< ; i++){
setTimeout(() => {
console.log(i);
}, )
}
二. 解法风暴
console.log(i)
在执行时,会按照词法作用域来取得循环条件中的变量 i
的值,本题的基本思路实际上就是如何在console.log语句
和for循环条件
之间添加(或修改)代码来隔离变量 i
的词法作用域。
解法一:最容易想到的方法——ES6块级作用域
//最容易想到的就是使用let实现的局部作用域
for (let i = ; i< ; i++){
setTimeout(() => {
console.log(i);
}, )
}
//变式
for (var i = ; i< ; i++){
let a = i;
setTimeout(() => {
console.log(a);
}, )
}
解法二:大多数前端曾接触过的第一种方法——IIFE(立即执行函数)
for(var i = ; i < ; i++){
(function(i){
setTimeout(() => {
console.log(i);
},)
})(i);
}
解法三:比较优雅的做法——setTimeout可以接收多个参数
//setTimeout的函数签名是setTimeout(fn, delay, ...params),params会作为fn执行时的实参传入
for (var i = ; i< ; i++){
setTimeout((i) => {
console.log(i);
}, , i);
}
解法四:利用函数方法bind为setTimeout传入预设参数
/*Function.prototype.bind(thisArg, ...args)
* 会得到一个新函数,新函数执行时预先设置了this和一部分参数,相当于把setTimeout改造成了偏函数
* bind执行后,setTimeout的第一个参数仍然是一个函数。
*/
for (var i = ; i < ; i++){
setTimeout(((i) => {
console.log(i);
}).bind(null,i), );
}
解法五:利用禁术with
with
的作用是延长作用域链会在词法作用域末端继续添加参数定义,在正式开发中通常是禁用的。下图右侧的scope
一栏中就可以看到local
作用域之上又多了一个with
引入的作用域,其中就包含传入的i
值。
for(var i = ; i < ; i++){
with({i}){
setTimeout(() => { console.log(i); },)
}
}
解法六:利用Promise传递决议结果来隔离作用域
//在每一轮循环中的i作为实参传递给promise的onFinished函数实现作用域隔离
for(var i = ; i < ; i++){
new Promise((resolve,reject)=>{
resolve(i);
}).then((i)=>{
setTimeout(() => { console.log(i); },)
}).catch(err=>{
console.log(err);
})
}
解法七:利用try...catch来隔离作用域
for(var i = ; i < ; i++){
try{
throw i;
}catch(i){
setTimeout(() => { console.log(i); },)
}
}
解法八:浏览器环境下setTimeout第一个参数可以为undefined(node.js中会报错)
//console.log相当于同步运行,跟setTimeout实际没什么关系了
for (var i = ; i< ; i++){
setTimeout(
console.log(i)
, )
}
解法九:篡改console.log
let result = [];
let consoleLog = console.log;
console.log = (n)=> {
result.push(n);
if(result.length === ) result.map((i,id)=>consoleLog(id));
} for(var i = ; i < ; i++){
setTimeout(() => {
console.log(i);
},)
} //变式——稍微有点欠扁
console.log = (function(){
let consoleLog = console.log;
let i = ;
return n => i++ === && consoleLog('0,1,2,3,4,5,6,7,8,9');
})(); for(var i = ; i < ; i++){
setTimeout(() => {
console.log(i);
},)
}
解法十:不按套路出牌的骚操作
for (var i = ; i < ; i++){
setTimeout(() => {
console.log(i++ % );
}, );
} //变式
for (var i = ; i < ; i++){
setTimeout(() => {
console.log(i++);
}, );
}
i = ;
作者:大史不说话
HDC.Cloud 华为开发者大会2020 即将于2020年2月11日-12日在深圳举办,是一线开发者学习实践鲲鹏通用计算、昇腾AI计算、数据库、区块链、云原生、5G等ICT开放能力的最佳舞台。
javascript基础修炼(13)——记一道有趣的JS脑洞练习题【华为云技术分享】的更多相关文章
- javascript基础修炼(13)——记一道有趣的JS脑洞练习题
目录 一. 题目 二. 解法风暴 示例代码托管在:http://www.github.com/dashnowords/blogs 博客园地址:<大史住在大前端>原创博文目录 华为云社区地址 ...
- 【我的物联网成长记6】由浅入深了解NB-IoT【华为云技术分享】
版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https://blog.csdn.net/devcloud/article/detai ...
- javascript基础修炼(5)—Event Loop(Node.js)
开发者的javascript造诣取决于对[动态]和[异步]这两个词的理解水平. 一. 一道考察异步知识的面试题 题目是这样的,要求写出下面代码的输出: setTimeout(() => { co ...
- 【我的物联网成长记8】超速入门AT指令集【华为云技术分享】
[摘要] 在物联网中,AT命令集可用于控制&调测设备.通信模块入网等.本文为您介绍NB-IoT常用的AT命令集及其调测工具. 什么是AT指令集 AT命令,用来控制TE(Terminal Equ ...
- 搞清楚一道关于Integer的面试题【华为云技术分享】
请看题1: public class IntegerDemo { public static void main(String[] args) { Integer a = ; Integer b = ...
- javascript基础修炼(2)——What's this(上)
目录 一.this是什么 二.近距离看this 三. this的一般指向规则 四. 基本规则示例 五. 后记 开发者的javascript造诣取决于对[动态]和[异步]这两个词的理解水平. 一.thi ...
- javascript基础修炼(4)——UMD规范的代码推演
javascript基础修炼(4)--UMD规范的代码推演 1. UMD规范 地址:https://github.com/umdjs/umd UMD规范,就是所有规范里长得最丑的那个,没有之一!!!它 ...
- javascript基础修炼(7)——Promise,异步,可靠性
开发者的javascript造诣取决于对[动态]和[异步]这两个词的理解水平. 一. 别人是开发者,你也是 Promise技术是[javascript异步编程]这个话题中非常重要的,它一度让我感到熟悉 ...
- javascript基础修炼(8)——指向FP世界的箭头函数
一. 箭头函数 箭头函数是ES6语法中加入的新特性,而它也是许多开发者对ES6仅有的了解,每当面试里被问到关于"ES6里添加了哪些新特性?"这种问题的时候,几乎总是会拿箭头函数来应 ...
随机推荐
- 7.22 NOIP模拟7
又是炸掉的一次考试 T1.方程的解 本次考试最容易骗分的一道题,但是由于T2花的时间太多,我竟然连a+b=c都没判..暴力掉了40分. 首先a+b=c,只有一组解. 然后是a=1,b=1,答案是c-1 ...
- CSPS模拟 82
还是Z哥的题,不过据说最后一题是D哥的 怪不得D2T3突然良心 (其实是突然毒瘤) 总分260大概是省三水平 不过和迪神并列了感觉还是很荣niu幸bi的- T1 考场上我弃了 因为是数学题,还要推柿子 ...
- 一、EditPlus 的安装 - Java软件的安装
EditPlus:该软件的功能类似于windows的文本编辑器,可处理文本.HTML和程序语言的Windows编辑器. 1.安装包的下载:http://pan.baidu.com/s/1qW1akZq ...
- NOIP模拟14-16
最近事情有些多,先咕了! 鸽了,时间太久远了,写了话坑太大,太费时间了!
- SpringBoot 源码解析 (一)----- SpringBoot核心原理入门
Spring Boot 概述 Build Anything with Spring Boot:Spring Boot is the starting point for building all Sp ...
- freeradius简单配置
1.安装freeradius sudo apt-get update sudo apt-get install freeradius 2.配置client.conf vim /etc/freeradi ...
- Java自学基础、进阶、项目实战网站推荐
推荐一个自学的好平台,有Java基础,前端,后端,基础的内容都有讲解,还有框架的讲解和实战项目,特别适合自学 JAVA 自学网站 JAVA 练习题 Spring 教程 Mybatis 教程 Sprin ...
- 第五天、vim,重定向,用户和组管理
第五天.vim,重定向,用户和组管理 vim vi:Visual editor,文本编辑器 行编辑器:sed 全屏编辑器:vim,vi,nano 其他编辑器gedit,gvim 定义别名让vi等于vi ...
- [Office] 显示Office 2013中的Developer Tab和启用Macro
显示Developer Tab是对Office 2013进行开发的前提条件. 在Excel 2013中,打开Excel Option: 只要勾选对应的Developer选项即可显示该tab. 随着Of ...
- [LC]141题 Intersection of Two Linked Lists (相交链表)(链表)
①中文题目 编写一个程序,找到两个单链表相交的起始节点. 如下面的两个链表: 在节点 c1 开始相交. 注意: 如果两个链表没有交点,返回 null.在返回结果后,两个链表仍须保持原有的结构.可假定整 ...