每日前端手写题--day16

以下题目来自掘金等其它博客,但是问题的答案都是根据笔者自己的理解做出的。如果你最近想要换工作或者巩固一下自己的前端知识基础,不妨和我一起参与到每日刷题的过程中来,如何?

第16天要刷的手写题如下:

  1. 实现一个函数,能够将传入的函数promisify
  2. 多种方式实现对象私有化属性
  3. 封装异步函数以防止await操作符作用的时候其出错溢出

下面是我的一些理解:

1. 实现一个函数,能够将传入的函数promise化

原理:将输入的函数A映射称为函数B, B函数的返回值是一个Promise对象,在此Promise对象内部的同步代码中会执行原来的函数A并为其构造一个临时的回调函数,在此临时回调函数内部改变包裹的Promise对象的状态,从而实现数据的传递。

下面就以fs.readFile演化成fs.readFileSync为例解释上面的原理:

1.1 fs.readFile的使用

js 复制代码
fs.readFile('data.json', 'utf8', function(err, data){
    if (err) {
        handleError(err);
    } else {
        handleData(data);
    }
});

!上述代码中的handleErrorhandleData是外部提供的将错误/数据传递出去的接口。

1.2 使用一个函数包裹这个过程,将其中使用到的量参数化

js 复制代码
const _f = (...args) => {
    fs.readFile(...args, function(err, data){
        if (err) {
            handleError(err);
        } else {
            handleData(data);
        }
    });
}

_f('data.json', 'utf8');

1.3 使用柯里化的思想将fs.readFile也作为参数

js 复制代码
const _c = (exec) => {
    return (...args) => {
        exec(...args, function(err, data){
            if (err) {
                handleError(err);
            } else {
                handleData(data);
            }
        });
    }
}

_c(fs.readFile)('data.json', 'utf8');

1.4 使用Promise的resolve和reject作为handleError和handleData

js 复制代码
const _c = (exec) => {
    return new Promise ((handleData, handleError)=>{
        (...args) => {
            exec(...args, function(err, data){
                if (err) {
                    handleError(err);
                } else {
                    handleData(data);
                }
            });
        }
    })
}

const _p = _c(fs.readFile)('data.json', 'utf8');

1.5 使用void、箭头函数和三目运算对上面的函数进行简化

js 复制代码
const _c = exec => new Promise ((handleData, handleError)=>void (...args) => void exec(...args, (err, data) => void err ? handleError(err) : handleData(data)));

最后的答案就是:exec => new Promise ((handleData, handleError)=>void (...args) => void exec(...args, (err, data) => void err ? handleError(err) : handleData(data)));

如果想要继续装一下,写成e => new Promise((a, b)=> (...c) => e(...c, (n, d) => n ? b(n) : a(d)))

1.6 总结

所谓promisify,其本质就是使用promise两种修改状态的函数作为处理失败和成功的函数。

2. 使用多种方式实现对象私有化属性

常见的实现方法有三种:

  • 使用代理,即Proxy
  • 使用闭包和Symbol
  • 使用闭包和哈希表

2.1 使用代理,对带有特殊标记的属性做特殊的处理

需要劫持的是get和ownKeys两种操作:

js 复制代码
const getProxyObj = obj => new Proxy ( obj, {
    get (target, key) {
        if(key.startsWith('_')) throw new Error();
        return Reflect.get(target, key);
    },
    ownKeys (target) {
        return Reflect.ownKeys(target).filter(v=>!v.startsWith('_'));
    }
}) 

2.2 使用闭包提供一个独一无二的属性名

由于在闭包的外部拿不到这个独一无二的属性名,也无法将其生产出来,所以可以在一定程度上保证私有性;但是如果使用键的遍历方法还是可以找到的。

js 复制代码
const MC = (
    function () {
        const _age = Symbol('age');
        return class MC {
            constructor(name){
                this._name = name;
            }
        }
    }
)();

2.3 使用闭包提供一个哈希表,哈希表中存放的是实例和实例的私有属性的对应关系

  • 在闭包外面拿不到这个哈希表所以可以在一定程度上保证私有性;
  • 使用WeakMap而不是Map解决内存泄漏的问题
js 复制代码
const MC = (
    function () {
        const _wm = new WeakMap();
        return class MC {
            constructor(name, age){
                _wm.set(this, {name, age});
            }
            get(key){
                return _wm.get(this)[key];
            }
        }
    }
)();

此外还可以在编译阶段解决此问题,比如使用ts中的private修饰符。

3. 封装异步函数以防止await操作符作用的时候其出错溢出

原理:将异步函数A通过此函数映射成为异步函数B,在B中执行A,同时对错误进行捕获并输出

js 复制代码
const useWrapper = async originFunc => {
    try {
        const rst = await originFunc();
        return [rst, null];
    } catch (err) {
        return [null, err];
    }
};


// 原来的调用方式
let data;
try {
    data = await originFunc();
} catch (e) {
    data = null;
}

// 包装之后的调用方式
const [data, err] =  await useWrapper(originFunc);
相关推荐
excel8 分钟前
HLS TS 文件损坏的元凶:Git 提交与拉取
前端
Aphasia31120 分钟前
https连接传输流程
前端·面试
徐小夕20 分钟前
万字长文!千万级文档 RAG 知识库系统落地实践
前端·算法·github
threelab31 分钟前
Three.js 物理模拟着色器 | 三维可视化 / AI 提示词
开发语言·前端·javascript·人工智能·3d·着色器
武器大师7233 分钟前
lv_binding_js 代码解读
开发语言·javascript·ecmascript
kyriewen41 分钟前
CSS Container Queries:彻底告别 @media 写到手软,附 5 个真实布局案例
前端·css·面试
Patrick_Wilson2 小时前
router.replace 之后紧跟 reload,页面为什么无限刷新?
javascript·react.js·浏览器
小小小小宇2 小时前
OpenMemory MCP
前端
和平宇宙3 小时前
AI笔记005. hermes-DeepSeek V4 Pro, 128K上下文引发的探索
前端·人工智能·笔记
IT_陈寒3 小时前
Redis持久化这个坑,我爬了一整天才出来
前端·人工智能·后端