js三座大山之异步六实现微任务的N种方式

异步编程

js三座大山之异步一单线程,event loope,宏任务&微任务
js三座大山之异步二异步方案
js三座大山之异步三promise本质
js三座大山之异步四-Promise的同步调用消除异步的传染性
js三座大山之异步五基于异步的js性能优化
js三座大山之异步六实现微任务的N种方式

js中现在有两类任务

一类是天然存在的在当前执行栈中按顺序依次执行的同步任务

一类是为了解决单线程带来的问题引入的异步任务

宏任务和微任务主要的区别体现在执行时机上,当事件循环每次执行完一个宏任务时,它会检查是否有微任务队列。如果有,那么在下一个宏任务开始之前,事件循环会一直执行微任务队列中的所有任务。也就是说,微任务的优先级高于宏任务,微任务会在任何新的宏任务开始之前执行,并且直至微任务被清空后才会执行下一个宏任务

宏任务和微任务根据不同的触发方式 在执行时机上有所不同 其他并无区别。并不是说微任务执行性能好于宏任务。

微任务一般由JavaScript引擎发起。

宏任务一般由宿主环境(如浏览器或Node.js)发起。

实现异步微任务的N种方式

promise

scss 复制代码
Promise.resolve().then(callback);

这个大家都知道而且是最常用的 在之前的文章中也有讲解过promise的本质 这里不再解释。

queueMicrotask

queueMicrotask api在浏览器和node环境都支持。是一个全局的方法,它接受一个回调函数作为参数,并将这个回调函数放入微任务队列中。

javascript 复制代码
queueMicrotask(()=>{
    console.log('queueMicrotask')
})

MutationObserver 浏览器环境

MutationObserver是一个可以用来监视DOM变化的API。当你使用MutationObserver创建一个观察者并开始观察某个DOM元素时,每当这个元素或其子元素发生变化,MutationObserver就会将你提供的回调函数添加到微任务队列中。

ini 复制代码
        const callback = () => {
            console.log('test MutationObserver')
        }
        const observer = new MutationObserver(callback);
        const element = document.createTextNode("");
        observer.observe(element, {
            characterData: true,
        });
        const change = () => {
            element.data = `${Date.now()}`;
        };
        change();

process.nextTick node环境

process.nextTick是Node.js特有的API,它的回调函数会在当前操作完成后,下一个事件循环开始前立即执行,也是微任务的一种。

arduino 复制代码
process.nextTick(() => {
    console.log('nextTick');
});

微任务代码总结

javascript 复制代码
const micros = [
    {
        name: 'queueMicrotask',
        test: () => {
            return typeof queueMicrotask === "function";
        },
        run: (callback: Task) => {
            queueMicrotask(callback);
        },
    },
    {
        name: 'Promise',
        test: () => {
            return typeof Promise !== "undefined";
        },
        run: (callback: Task) => {
            Promise.resolve().then(callback);
        },
    },
    {
        name: 'process.nextTick',
        test: () => {
            return (
                typeof process === "object" && typeof process.nextTick === "function"
            );
        },
        run: (callback: Task) => {
            process.nextTick(callback);
        },
    },
    {
        name: 'MutationObserver',
        test: () => {
            return (
                typeof MutationObserver !== "undefined" && typeof window !== "undefined"
            );
        },
        run: (callback: Task) => {
            const observer = new MutationObserver(callback);
            const element = document.createTextNode("");
            observer.observe(element, {
                characterData: true,
            });
            const change = () => {
                element.data = `${Date.now()}`;
            };
            change();
        },
    },
];

export const microTask = (callback: Task) => {
    const runner = [...micros].find((item) => item.test());
    runner && runner.run(callback);
};

github源码

相关推荐
excel1 分钟前
一文读懂 Vue 组件间通信机制(含 Vue2 / Vue3 区别)
前端·javascript·vue.js
JarvanMo5 分钟前
Flutter 应用生命周期:使用 AppLifecycleListener 阻止应用崩溃
前端
我的xiaodoujiao1 小时前
从 0 到 1 搭建 Python 语言 Web UI自动化测试学习系列 9--基础知识 5--常用函数 3
前端·python·测试工具·ui
聪明的笨猪猪2 小时前
Java “并发容器框架(Fork/Join)”面试清单(含超通俗生活案例与深度理解)
java·经验分享·笔记·面试
李鸿耀3 小时前
Flex 布局下文字省略不生效?原因其实很简单
前端
皮蛋瘦肉粥_1214 小时前
pink老师html5+css3day06
前端·css3·html5
华仔啊8 小时前
前端必看!12个JS神级简写技巧,代码效率直接飙升80%,告别加班!
前端·javascript
excel8 小时前
dep.ts 逐行解读
前端·javascript·vue.js
爱上妖精的尾巴8 小时前
5-20 WPS JS宏 every与some数组的[与或]迭代(数组的逻辑判断)
开发语言·前端·javascript·wps·js宏·jsa
excel8 小时前
Vue3 响应式核心源码全解析:Dep、Link 与 track/trigger 完整执行机制详解
前端