微前端的设计思想,面试官问的那些前端原理你都懂吗

27 阅读6分钟

打开全栈工匠技能包-1小时轻松掌握SSR

两小时精通jq+bs插件开发

生产环境下如歌部署Node.js

网易内部VUE自定义插件库NPM集成

谁说前端不用懂安全,XSS跨站脚本的危害

webpack的loader到底是什么样的?两小时带你写一个自己loader

开源分享:docs.qq.com/doc/DSmRnRG…

registerApps(routers: Router[]) {

(routers || []).forEach((r) => {

this.Apps.push({

entry: r.path,

activeRule: (location) => (location.href.indexOf(r.activeWhen) !== -1)

});

});

}

3.1.2 拦截

我们需要通过拦截注册路由事件以保证主/子应用的逻辑处理时机。

import Hailuo from ".";

// 需要拦截的实践

const EVENTS_NAME = ['hashchange', 'popstate'];

// 实践收集

const EVENTS_STACKS = {

hashchange: [],

popstate: []

};

// 基座切换路由后的逻辑

const handleUrlRoute = (...args) => {

// 加载对应的子应用

Hailuo.loadApp();

// 执行子应用路由的方法

callAllEventListeners(...args);

};

export const patch = () => {

// 1. 先保证基座的事件监听路由的变化

window.addEventListener('hashchange', handleUrlRoute);

window.addEventListener('popstate', handleUrlRoute);

// 2. 重写addEventListener和removeEventListener

// 当遇到路由事件后:收集到stack中

// 如果是其他事件:执行original事件监听方法

const originalAddEventListener = window.addEventListener;

const originalRemoveEventListener = window.removeEventListener;

window.addEventListener = (name, handler) => {

if(name && EVENTS_NAME.includes(name) && typeof handler === "function") {

EVENTS_STACKS[name].indexOf(handler) === -1 && EVENTS_STACKS[name].push(handler);

return;

}

return originalAddEventListener.call(this, name, handler);

};

window.removeEventListener = (name, handler) => {

if(name && EVENTS_NAME.includes(name) && typeof handler === "function") {

EVENTS_STACKS[name].indexOf(handler) === -1 && 

(EVENTS_STACKS[name] = EVENTS_STACKS[name].filter((fn) => (fn !== handler)));

return;

return originalRemoveEventListener.call(this, name, handler);

};

// 手动给pushState和replaceState添加上监听路由变化的能力

// 有点像vue2中数组的变异方法

const createPopStateEvent = (state: any, name: string) => {

const evt = new PopStateEvent("popstate", { state });

evt['trigger'] = name;

return evt;

};

const patchUpdateState = (updateState: (data: any, title: string, url?: string)=>void, name: string) => {

return function() {

const before = window.location.href;

updateState.apply(this, arguments);

const after = window.location.href;

if(before !== after) {

handleUrlRoute(createPopStateEvent(window.history.state, name));

}

};

}

window.history.pushState = patchUpdateState(

window.history.pushState,

"pushState"

);

window.history.replaceState = patchUpdateState(

window.history.replaceState,

"replaceState"

);

}

3.1.3 加载

通过路由可以匹配到符合的子应用后,那么该如何将它加载到页面呢?

我们知道SPA的html文件只是一个空模板,实质是通过js驱动的页面渲染,那么我们把某一个页面的js文件,全都剪切到另一个html的<script>标签中执行,就实现了A页面加载B的页面。

async loadApp() {

// 加载对应的子应用

const shouldMountApp = this.Apps.filter(this.isActive);

const app = shouldMountApp[0];

const subapp = document.getElementById('submodule');

await fetchUrl(app.entry)

// 将html渲染到主应用里

.then((text) => {

subapp.innerHTML = text;

});

// 执行 fetch到的js

const res = await fetchScripts(subapp, app.entry);

if(res.length) {

execScript(res.reduce((t, c) => (t+c), ''));

}

Better实践 ——html-entry

它是一个加载并处理html、js、css的库。

它不是去加载一个个的js、css资源,而是去加载微应用的入口html。

  • 第一步 :发送请求,获取子应用入口HTML。

  • 第二步 :处理该html文档,去掉html、head标签,处理静态资源。

  • 第三步 :处理sourceMap;处理js沙箱;找到入口js。

  • 第四步 :获取子应用provider内容

同时,约束了子应用提供加载和销毁函数(这个结构是不是很眼熟):

export function provider({ dom, basename, globalData }) {

return {

render() {

ReactDOM.render(

<App basename={basename} globalData={globalData} />,

dom ? dom.querySelector('#root') : document.querySelector('#root')

);

},

destroy({ dom }) {

if (dom) {

ReactDOM.unmountComponentAtNode(dom);

}

},

};

}

3.2 沙箱(Sandbox)


沙箱是什么:你可以理解为对作用域的一种比喻,在一个沙箱内,我的任何操作不会对外界产生影响。

Why we need sandbox?

当我们集成了很多子应用到一起后,势必会出现冲突,如全局变量冲突样式冲突,这些冲突可能会导致应用样式异常,甚至功能不可用。所以想让微前端达到生产可用的程度,让每个子应用之间达到一定程度隔离的沙箱机制是必不可少的。

实现沙箱,最重要的是:控制沙箱的开启和关闭。

3.2.1 快照沙箱

原理就是运行在某一环境A时,打一个快照,当从别的环境B切换回来的时候,我们通过这个快照就可以立即恢复之前环境A时的情况,比如:

// 切换到环境A

window.a = 2;

// 切换到环境B

window.a = 3;

// 切换到环境A

console.log(a);    // 2

实现思路,我们假设有Sandbox这个类:

class Sandbox {

private original;

private mutated;

sandBoxActive: () => void;

sandBoxDeactivate: () => void;

}

const sandbox = new Sandbox();

const code = "...";

sandbox.activate();

execScript(code);

sandbox.sandBoxDeactivate();

来理一下这个逻辑:

  1. 在sandBoxActive的时候,把变量存到original里;

  2. 在sandBoxDeactivate的时候,把当前变量和original对比,不同的存到mutated(保存了快照),然后把变量的状态恢复到original;

  3. 当该沙箱再次触发sandBoxActive,就可以把mutated的变量恢复到window上,实现沙箱的切换。

3.2.2 VM沙箱

类似于node中的vm模块(可在 V8 虚拟机上下文中编译和运行代码):nodejs.cn/api/vm.html…

快照沙箱的缺点是无法同时支持多个实例。 但是vm沙箱利用proxy就可以解决这个问题。

class SandBox {

execScript(code: string) {

const varBox = {};

const fakeWindow = new Proxy(window, {

get(target, key) {

return varBox[key] || window[key];

},

set(target, key, value) {

varBox[key] = value;

return true;

}

})

const fn = new Function('window', code);

fn(fakeWindow);

}

}

export default SandBox;

// 实现了隔离

const sandbox = new Sandbox();

sandbox.execScript(code);

const sandbox2 = new Sandbox();

sandbox2.execScript(code2);

// map

varBox = {

'aWindow': '...',

'bWindow': '...'

}

我们把各个子应用的window放到map中,通过proxy代理,当访问时,直接就是访问到的各个子应用的window对象;如果没有,比如使用window.addEventListener,就会去真正的window中寻找。

3.2.3 CSS沙箱

  • 前提:webpack在构建的时候,最终是通过appendChild去添加style标签到html里的

解决方案:劫持appendChild,增加namespace。

❤️ 谢谢支持


以上便是本次分享的全部内容,希望对你有所帮助^_^

文末

js前端的重头戏,值得花大部分时间学习。

JavaScript知识

推荐通过书籍学习,《 JavaScript 高级程序设计(第 4 版)》你值得拥有。整本书内容质量都很高,尤其是前十章语言基础部分,建议多读几遍。

前端电子书

开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】

另外,大推一个网上教程 现代 JavaScript 教程 ,文章深入浅出,很容易理解,上面的内容几乎都是重点,而且充分发挥了网上教程的时效性和资料链接。

学习资料在精不在多,二者结合,定能构建你的 JavaScript 知识体系。

面试本质也是考试,面试题就起到很好的考纲作用。想要取得优秀的面试成绩,刷面试题是必须的,除非你样样精通。

这是288页的前端面试题

288页面试题