电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

淺談前端JS沙箱實(shí)現(xiàn)的幾種方式

瀏覽:108日期:2024-03-22 11:07:43
目錄前言iframe實(shí)現(xiàn)沙箱diff方式實(shí)現(xiàn)沙箱基于代理(Proxy)實(shí)現(xiàn)單實(shí)例沙箱基于代理(Proxy)實(shí)現(xiàn)多實(shí)例沙箱結(jié)束語(yǔ)參考前言

在微前端領(lǐng)域當(dāng)中,沙箱是很重要的一件事情。像微前端框架single-spa沒有實(shí)現(xiàn)js沙箱,我們?cè)跇?gòu)建大型微前端應(yīng)用的時(shí)候,很容易造成一些變量的沖突,對(duì)應(yīng)用的可靠性面臨巨大的風(fēng)險(xiǎn)。在微前端當(dāng)中,有一些全局對(duì)象在所有的應(yīng)用中需要共享,如document,location,等對(duì)象。子應(yīng)用開發(fā)的過程中可能是多個(gè)團(tuán)隊(duì)在做,很難約束他們使用全局變量。有些頁(yè)面可能會(huì)有多個(gè)不同的子應(yīng)用,需要我們支持多沙箱,每個(gè)沙箱需要有加載,卸載,在恢復(fù)的能力。

iframe實(shí)現(xiàn)沙箱

在前端中,有一個(gè)比較重要的html標(biāo)簽iframe,實(shí)際上,我們可以通過iframe對(duì)象,把原生瀏覽器對(duì)象通過contentWindow取出來,這個(gè)對(duì)象天然具有所有的屬性,而且與主應(yīng)用的環(huán)境隔離。下面我們通過代碼看下

let iframe = document.createElement(’iframe’,{src:’about:blank’});document.body.appendChild(iframe);const sandboxGlobal = iframe.contentWindow;

注意:只有同域的ifame才能取出對(duì)應(yīng)的contentWindow, iframe的src設(shè)置為about:blank,可以保證一定是同域的,也不會(huì)發(fā)生資源加載,參考iframe src

在前言中我們提到,微前端除了有一個(gè)隔離的window環(huán)境外,其實(shí)還需要共享一些全局對(duì)象,這時(shí)候我們可以用代理去實(shí)現(xiàn)。下面我們通過代碼看下

class SandboxWindow { /** * 構(gòu)造函數(shù) * @param {*} context 需要共享的對(duì)象 * @param {*} frameWindow iframe的window */ constructor(context, frameWindow) {return new Proxy(frameWindow, { get(target, name) {if (name in context) { // 優(yōu)先使用共享對(duì)象 return context[name];}return target[name]; }, set(target, name, value) {if (name in context) { // 修改共享對(duì)象的值 return context[name] = value;}target[name] = value; }}) }}// 需要全局共享的變量const context = { document:window.document, history: window.history }// 創(chuàng)建沙箱const newSandboxWindow = new SandboxWindow(context, sandboxGlobal); // 判斷沙箱上的對(duì)象和全局對(duì)象是否相等console.log(’equal’,newSandboxWindow.document === window.document)newSandboxWindow.abc = ’1’; //在沙箱上添加屬性console.log(window.abc); // 在全局上查看屬性console.log(newSandboxWindow.abc) //在沙箱上查看屬性

我們運(yùn)行起來,看下結(jié)果

淺談前端JS沙箱實(shí)現(xiàn)的幾種方式

以上我們利用iframe沙箱可以實(shí)現(xiàn)以下特性:

全局變量隔離,如setTimeout、location、react不同版本隔離 路由隔離,應(yīng)用可以實(shí)現(xiàn)獨(dú)立路由,也可以共享全局路由 多實(shí)例,可以同時(shí)存在多個(gè)獨(dú)立的微應(yīng)用同時(shí)運(yùn)行diff方式實(shí)現(xiàn)沙箱

在不支持代理的瀏覽器中,我們可以通過diff的方式實(shí)習(xí)沙箱。在應(yīng)用運(yùn)行的時(shí)候保存一個(gè)快照window對(duì)象,將當(dāng)前window對(duì)象的全部屬性都復(fù)制到快照對(duì)象上,子應(yīng)用卸載的時(shí)候?qū)indow對(duì)象修改做個(gè)diff,將不同的屬性用個(gè)modifyMap保存起來,再次掛載的時(shí)候再加上這些修改的屬性。代碼如下:

class DiffSandbox { constructor(name) { this.name = name; this.modifyMap = {}; // 存放修改的屬性 this.windowSnapshot = {}; } active() { // 緩存active狀態(tài)的沙箱 this.windowSnapshot = {}; for (const item in window) { this.windowSnapshot[item] = window[item]; } Object.keys(this.modifyMap).forEach(p => { window[p] = this.modifyMap[p]; }) } inactive() { for (const item in window) { if (this.windowSnapshot[item] !== window[item]) {// 記錄變更this.modifyMap[item] = window[item];// 還原windowwindow[item] = this.windowSnapshot[item]; } } }}const diffSandbox = new DiffSandbox(’diff沙箱’);diffSandbox.active(); // 激活沙箱window.a = ’1’console.log(’開啟沙箱:’,window.a);diffSandbox.inactive(); //失活沙箱console.log(’失活沙箱:’, window.a);diffSandbox.active(); // 重新激活console.log(’再次激活’, window.a);

我們運(yùn)行一下,查看結(jié)果

淺談前端JS沙箱實(shí)現(xiàn)的幾種方式

這種方式也無法支持多實(shí)例,因?yàn)檫\(yùn)行期間所有的屬性都是保存在window上的。

基于代理(Proxy)實(shí)現(xiàn)單實(shí)例沙箱

在ES6當(dāng)中,我們可以通過代理(Proxy)實(shí)現(xiàn)對(duì)象的劫持。基本實(shí)錄也是通過window對(duì)象的修改進(jìn)行記錄,在卸載時(shí)刪除這些記錄,在應(yīng)用再次激活時(shí)恢復(fù)這些記錄,來達(dá)到模擬沙箱環(huán)境的目的。代碼如下

// 修改window屬性的公共方法const updateWindowProp = (prop, value, isDel) => { if (value === undefined || isDel) {delete window[prop]; } else {window[prop] = value; }}class ProxySandbox { active() {// 根據(jù)記錄還原沙箱this.currentUpdatedPropsValueMap.forEach((v, p) => updateWindowProp(p, v)); } inactive() {// 1 將沙箱期間修改的屬性還原為原先的屬性this.modifiedPropsMap.forEach((v, p) => updateWindowProp(p, v));// 2 將沙箱期間新增的全局變量消除this.addedPropsMap.forEach((_, p) => updateWindowProp(p, undefined, true)); } constructor(name) {this.name = name;this.proxy = null;// 存放新增的全局變量this.addedPropsMap = new Map(); // 存放沙箱期間更新的全局變量this.modifiedPropsMap = new Map();// 存在新增和修改的全局變量,在沙箱激活的時(shí)候使用this.currentUpdatedPropsValueMap = new Map();const { addedPropsMap, currentUpdatedPropsValueMap, modifiedPropsMap } = this;const fakeWindow = Object.create(null);const proxy = new Proxy(fakeWindow, { set(target, prop, value) {if (!window.hasOwnProperty(prop)) { // 如果window上沒有的屬性,記錄到新增屬性里 // debugger; addedPropsMap.set(prop, value);} else if (!modifiedPropsMap.has(prop)) { // 如果當(dāng)前window對(duì)象有該屬性,且未更新過,則記錄該屬性在window上的初始值 const originalValue = window[prop]; modifiedPropsMap.set(prop, originalValue);}// 記錄修改屬性以及修改后的值currentUpdatedPropsValueMap.set(prop, value);// 設(shè)置值到全局window上updateWindowProp(prop, value);return true; }, get(target, prop) {return window[prop]; },});this.proxy = proxy; }}const newSandBox = new ProxySandbox(’代理沙箱’);const proxyWindow = newSandBox.proxy;proxyWindow.a = ’1’console.log(’開啟沙箱:’, proxyWindow.a, window.a);newSandBox.inactive(); //失活沙箱console.log(’失活沙箱:’, proxyWindow.a, window.a);newSandBox.active(); //失活沙箱console.log(’重新激活沙箱:’, proxyWindow.a, window.a);

我們運(yùn)行代碼,看下結(jié)果

淺談前端JS沙箱實(shí)現(xiàn)的幾種方式

這種方式同一時(shí)刻只能有一個(gè)激活的沙箱,否則全局對(duì)象上的變量會(huì)有兩個(gè)以上的沙箱更新,造成全局變量沖突。

基于代理(Proxy)實(shí)現(xiàn)多實(shí)例沙箱

在單實(shí)例的場(chǎng)景總,我們的fakeWindow是一個(gè)空的對(duì)象,其沒有任何儲(chǔ)存變量的功能,微應(yīng)用創(chuàng)建的變量最終實(shí)際都是掛載在window上的,這就限制了同一時(shí)刻不能有兩個(gè)激活的微應(yīng)用。

class MultipleProxySandbox { active() {this.sandboxRunning = true; } inactive() {this.sandboxRunning = false; } /** * 構(gòu)造函數(shù) * @param {*} name 沙箱名稱 * @param {*} context 共享的上下文 * @returns */ constructor(name, context = {}) {this.name = name;this.proxy = null;const fakeWindow = Object.create({});const proxy = new Proxy(fakeWindow, { set: (target, name, value) => {if (this.sandboxRunning) { if (Object.keys(context).includes(name)) {context[name] = value; } target[name] = value;} }, get: (target, name) => {// 優(yōu)先使用共享對(duì)象if (Object.keys(context).includes(name)) { return context[name];}return target[name]; }})this.proxy = proxy; }}const context = { document: window.document };const newSandBox1 = new MultipleProxySandbox(’代理沙箱1’, context);newSandBox1.active();const proxyWindow1 = newSandBox1.proxy;const newSandBox2 = new MultipleProxySandbox(’代理沙箱2’, context);newSandBox2.active();const proxyWindow2 = newSandBox2.proxy;console.log(’共享對(duì)象是否相等’, window.document === proxyWindow1.document, window.document === proxyWindow2.document);proxyWindow1.a = ’1’; // 設(shè)置代理1的值proxyWindow2.a = ’2’; // 設(shè)置代理2的值window.a = ’3’; // 設(shè)置window的值console.log(’打印輸出的值’, proxyWindow1.a, proxyWindow2.a, window.a);newSandBox1.inactive(); newSandBox2.inactive(); // 兩個(gè)沙箱都失活proxyWindow1.a = ’4’; // 設(shè)置代理1的值proxyWindow2.a = ’4’; // 設(shè)置代理2的值window.a = ’4’; // 設(shè)置window的值console.log(’失活后打印輸出的值’, proxyWindow1.a, proxyWindow2.a, window.a);newSandBox1.active(); newSandBox2.active(); // 再次激活proxyWindow1.a = ’4’; // 設(shè)置代理1的值proxyWindow2.a = ’4’; // 設(shè)置代理2的值window.a = ’4’; // 設(shè)置window的值console.log(’失活后打印輸出的值’, proxyWindow1.a, proxyWindow2.a, window.a);

運(yùn)行代碼,結(jié)果如下:

淺談前端JS沙箱實(shí)現(xiàn)的幾種方式

這種方式同一時(shí)刻只能有一個(gè)激活的多個(gè)沙箱,從而實(shí)現(xiàn)多實(shí)例沙箱。

結(jié)束語(yǔ)

以上是微前端比較常用的沙箱實(shí)現(xiàn)方式,想要在生產(chǎn)中使用,需要我們做很多的判斷和約束。下篇我們通過源碼看下微前端框架qiankun是怎么實(shí)現(xiàn)沙箱的。上面的代碼在github,如需查看,請(qǐng)移步j(luò)s-sandbox

參考

iframe src ES6 Proxy

到此這篇關(guān)于淺談前端JS沙箱實(shí)現(xiàn)的幾種方式的文章就介紹到這了,更多相關(guān)JS 沙箱內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: JavaScript
相關(guān)文章:
主站蜘蛛池模板: 培训无忧网-教育培训咨询招生第三方平台| 山东艾德实业有限公司 | 税筹星_灵活用工平台_企业财务顾问_财税法薪综合服务平台 | 欧版反击式破碎机-欧版反击破-矿山石料破碎生产线-青州奥凯诺机械 | 外观设计_设备外观设计_外观设计公司_产品外观设计_机械设备外观设计_东莞工业设计公司-意品深蓝 | 活性炭-蜂窝-椰壳-柱状-粉状活性炭-河南唐达净水材料有限公司 | 河南凯邦机械制造有限公司 | 电动葫芦|环链电动葫芦-北京凌鹰名优起重葫芦 | 淘气堡_室内儿童乐园_户外无动力儿童游乐设备-高乐迪(北京) | YT保温材料_YT无机保温砂浆_外墙保温材料_南阳银通节能建材高新技术开发有限公司 | 语料库-提供经典范文,文案句子,常用文书,您的写作得力助手 | 太阳能发电系统-太阳能逆变器,控制器-河北沐天太阳能科技首页 | 定坤静电科技静电消除器厂家-除静电设备| MTK核心板|MTK开发板|MTK模块|4G核心板|4G模块|5G核心板|5G模块|安卓核心板|安卓模块|高通核心板-深圳市新移科技有限公司 | 真空泵维修保养,普发,阿尔卡特,荏原,卡西亚玛,莱宝,爱德华干式螺杆真空泵维修-东莞比其尔真空机电设备有限公司 | 横河变送器-横河压力变送器-EJA变送器-EJA压力变送器-「泉蕴仪表」 | 列管冷凝器,刮板蒸发器,外盘管反应釜厂家-无锡曼旺化工设备有限公司 | 找果网 | 苹果手机找回方法,苹果iPhone手机丢了找回,认准找果网! | 称重传感器,测力传感器,拉压力传感器,压力变送器,扭矩传感器,南京凯基特电气有限公司 | 激光内雕_led玻璃_发光玻璃_内雕玻璃_导光玻璃-石家庄明晨三维科技有限公司 激光内雕-内雕玻璃-发光玻璃 | 首页_欧瑞传动官方网站--主营变频器、伺服系统、新能源、软起动器、PLC、HMI | 假肢-假肢价格-假肢厂家-河南假肢-郑州市力康假肢矫形器有限公司 | 深圳天际源广告-形象堆头,企业文化墙,喷绘,门头招牌设计制作专家 | 注浆压力变送器-高温熔体传感器-矿用压力传感器|ZHYQ朝辉 | 知网论文检测系统入口_论文查重免费查重_中国知网论文查询_学术不端检测系统 | 半自动预灌装机,卡式瓶灌装机,注射器灌装机,给药器灌装机,大输液灌装机,西林瓶灌装机-长沙一星制药机械有限公司 | 拉力测试机|材料拉伸试验机|电子拉力机价格|万能试验机厂家|苏州皖仪实验仪器有限公司 | 变频器维修公司_plc维修_伺服驱动器维修_工控机维修 - 夫唯科技 变位机,焊接变位机,焊接变位器,小型变位机,小型焊接变位机-济南上弘机电设备有限公司 | 爱佩恒温恒湿测试箱|高低温实验箱|高低温冲击试验箱|冷热冲击试验箱-您身边的模拟环境试验设备技术专家-合作热线:400-6727-800-广东爱佩试验设备有限公司 | 粉末冶金注射成型厂家|MIM厂家|粉末冶金齿轮|MIM零件-深圳市新泰兴精密科技 | 液压扳手-高品质液压扳手供应商 - 液压扳手, 液压扳手供应商, 德国进口液压拉马 | 政府园区专业委托招商平台_助力企业选址项目快速落地_东方龙商务集团 | 明渠式紫外线杀菌器-紫外线消毒器厂家-定州市优威环保 | 京港视通报道-质量走进大江南北-京港视通传媒[北京]有限公司 | 青岛球场围网,青岛车间隔离网,青岛机器人围栏,青岛水源地围网,青岛围网,青岛隔离栅-青岛晟腾金属制品有限公司 | 一礼通 (www.yilitong.com)-企业礼品解决方案一站式服务平台 | 威廉希尔WilliamHill·足球(中国)体育官方网站 | 24位ADC|8位MCU-芯易德科技有限公司 | 地源热泵一体机,地源热泵厂家-淄博汇能环保设备有限公司 | 点焊机-缝焊机-闪光对焊机-电阻焊设备生产厂家-上海骏腾发智能设备有限公司 | 亿立分板机_曲线_锯片式_走刀_在线式全自动_铣刀_在线V槽分板机-杭州亿协智能装备有限公司 |