日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

淺談前端JS沙箱實現的幾種方式

瀏覽:191日期:2024-03-22 11:07:43
目錄前言iframe實現沙箱diff方式實現沙箱基于代理(Proxy)實現單實例沙箱基于代理(Proxy)實現多實例沙箱結束語參考前言

在微前端領域當中,沙箱是很重要的一件事情。像微前端框架single-spa沒有實現js沙箱,我們在構建大型微前端應用的時候,很容易造成一些變量的沖突,對應用的可靠性面臨巨大的風險。在微前端當中,有一些全局對象在所有的應用中需要共享,如document,location,等對象。子應用開發的過程中可能是多個團隊在做,很難約束他們使用全局變量。有些頁面可能會有多個不同的子應用,需要我們支持多沙箱,每個沙箱需要有加載,卸載,在恢復的能力。

iframe實現沙箱

在前端中,有一個比較重要的html標簽iframe,實際上,我們可以通過iframe對象,把原生瀏覽器對象通過contentWindow取出來,這個對象天然具有所有的屬性,而且與主應用的環境隔離。下面我們通過代碼看下

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

注意:只有同域的ifame才能取出對應的contentWindow, iframe的src設置為about:blank,可以保證一定是同域的,也不會發生資源加載,參考iframe src

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

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

我們運行起來,看下結果

淺談前端JS沙箱實現的幾種方式

以上我們利用iframe沙箱可以實現以下特性:

全局變量隔離,如setTimeout、location、react不同版本隔離 路由隔離,應用可以實現獨立路由,也可以共享全局路由 多實例,可以同時存在多個獨立的微應用同時運行diff方式實現沙箱

在不支持代理的瀏覽器中,我們可以通過diff的方式實習沙箱。在應用運行的時候保存一個快照window對象,將當前window對象的全部屬性都復制到快照對象上,子應用卸載的時候將window對象修改做個diff,將不同的屬性用個modifyMap保存起來,再次掛載的時候再加上這些修改的屬性。代碼如下:

class DiffSandbox { constructor(name) { this.name = name; this.modifyMap = {}; // 存放修改的屬性 this.windowSnapshot = {}; } active() { // 緩存active狀態的沙箱 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);

我們運行一下,查看結果

淺談前端JS沙箱實現的幾種方式

這種方式也無法支持多實例,因為運行期間所有的屬性都是保存在window上的。

基于代理(Proxy)實現單實例沙箱

在ES6當中,我們可以通過代理(Proxy)實現對象的劫持。基本實錄也是通過window對象的修改進行記錄,在卸載時刪除這些記錄,在應用再次激活時恢復這些記錄,來達到模擬沙箱環境的目的。代碼如下

// 修改window屬性的公共方法const updateWindowProp = (prop, value, isDel) => { if (value === undefined || isDel) {delete window[prop]; } else {window[prop] = value; }}class ProxySandbox { active() {// 根據記錄還原沙箱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();// 存在新增和修改的全局變量,在沙箱激活的時候使用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)) { // 如果當前window對象有該屬性,且未更新過,則記錄該屬性在window上的初始值 const originalValue = window[prop]; modifiedPropsMap.set(prop, originalValue);}// 記錄修改屬性以及修改后的值currentUpdatedPropsValueMap.set(prop, value);// 設置值到全局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);

我們運行代碼,看下結果

淺談前端JS沙箱實現的幾種方式

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

基于代理(Proxy)實現多實例沙箱

在單實例的場景總,我們的fakeWindow是一個空的對象,其沒有任何儲存變量的功能,微應用創建的變量最終實際都是掛載在window上的,這就限制了同一時刻不能有兩個激活的微應用。

class MultipleProxySandbox { active() {this.sandboxRunning = true; } inactive() {this.sandboxRunning = false; } /** * 構造函數 * @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) => {// 優先使用共享對象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(’共享對象是否相等’, window.document === proxyWindow1.document, window.document === proxyWindow2.document);proxyWindow1.a = ’1’; // 設置代理1的值proxyWindow2.a = ’2’; // 設置代理2的值window.a = ’3’; // 設置window的值console.log(’打印輸出的值’, proxyWindow1.a, proxyWindow2.a, window.a);newSandBox1.inactive(); newSandBox2.inactive(); // 兩個沙箱都失活proxyWindow1.a = ’4’; // 設置代理1的值proxyWindow2.a = ’4’; // 設置代理2的值window.a = ’4’; // 設置window的值console.log(’失活后打印輸出的值’, proxyWindow1.a, proxyWindow2.a, window.a);newSandBox1.active(); newSandBox2.active(); // 再次激活proxyWindow1.a = ’4’; // 設置代理1的值proxyWindow2.a = ’4’; // 設置代理2的值window.a = ’4’; // 設置window的值console.log(’失活后打印輸出的值’, proxyWindow1.a, proxyWindow2.a, window.a);

運行代碼,結果如下:

淺談前端JS沙箱實現的幾種方式

這種方式同一時刻只能有一個激活的多個沙箱,從而實現多實例沙箱。

結束語

以上是微前端比較常用的沙箱實現方式,想要在生產中使用,需要我們做很多的判斷和約束。下篇我們通過源碼看下微前端框架qiankun是怎么實現沙箱的。上面的代碼在github,如需查看,請移步js-sandbox

參考

iframe src ES6 Proxy

到此這篇關于淺談前端JS沙箱實現的幾種方式的文章就介紹到這了,更多相關JS 沙箱內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: JavaScript
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
九一成人免费视频| 久久久久久久欧美精品| 麻豆亚洲精品| 亚洲综合日韩| 亚洲综合激情在线| 免费欧美日韩| 美女精品在线| 一区二区电影| 亚洲精品婷婷| 日本麻豆一区二区三区视频| 亚洲九九精品| 久久国内精品视频| 国产精品免费精品自在线观看| 国产精品a级| 首页国产精品| 91精品在线观看国产| 午夜精品网站| 亚洲免费成人av在线| 国产精品videossex| 岛国精品一区| 97国产成人高清在线观看| 三级精品视频| 亚洲综合日韩| 国产精品三级| 久久久一本精品| 国产精品美女| 青青青国产精品| 国产一区2区| 不卡一区2区| 日韩av在线播放中文字幕| 国产一区二区三区不卡av| 99久久99视频只有精品| 一二三区精品| 成人午夜毛片| 中文在线不卡| 欧美亚洲色图校园春色| 精品免费av| 好吊日精品视频| 青青伊人久久| 92国产精品| 久久先锋影音| 麻豆精品蜜桃视频网站| 国精品一区二区三区| 日本精品另类| 日韩欧美精品综合| 中文字幕av一区二区三区四区| 国产精品视频一区二区三区四蜜臂| 欧美freesex黑人又粗又大| 日韩一区精品视频| 精品三区视频| av成人国产| 免费日韩一区二区三区| 欧美精品一区二区三区精品| 91精品福利观看| 激情综合亚洲| 老司机精品视频网| 亚洲一区免费| 国产精品成久久久久| 伊人www22综合色| 亚洲伦乱视频| 国产精品欧美三级在线观看| 亚洲精品一区二区在线看| 欧美精品99| 视频一区免费在线观看| 成人在线视频中文字幕| 在线观看视频免费一区二区三区| 欧美国产美女| 国产欧美日韩在线观看视频| 午夜视频精品| 国产一区二区三区亚洲综合| 亚洲三级网址| 亚洲精品小说| 亚洲国产成人二区| 日本久久一区| 黑丝一区二区三区| 一区二区精品伦理...| 91麻豆精品激情在线观看最新 | 丝袜美腿亚洲一区| 国产精品福利在线观看播放| 日韩高清电影免费| 亚洲欧美日韩国产一区二区| 亚洲永久av| 国产精品成人一区二区网站软件| 美女日韩在线中文字幕| 久久在线免费| 中文在线资源| 久久久亚洲欧洲日产| 日本亚洲欧美天堂免费| 欧美日韩视频| 99精品网站| 欧美日韩免费观看视频| 精品国产午夜| 欧美激情精品| 亚洲不卡视频| 亚洲一级在线| 在线国产一区二区| 国产视频久久| 欧美日韩视频| 亚洲国产一区二区三区在线播放| 国产色播av在线| 精品一区二区三区中文字幕在线| 日韩av在线免费观看不卡| 亚洲另类av| 视频在线在亚洲| 午夜在线精品偷拍| 亚洲美洲欧洲综合国产一区 | 日韩高清不卡一区| 久久亚洲风情| 中文亚洲免费| 久久福利影视| 久久福利一区| 免费日韩av片| 日韩专区欧美专区| 男女激情视频一区| 蜜桃视频一区二区三区在线观看| 黄色成人91| 亚洲精品小说| 国产精品日本| 亚洲欧美日韩视频二区| 国产精品丝袜xxxxxxx| 在线视频精品| 亚洲国产日韩欧美在线| 日韩亚洲在线| 日韩中文字幕不卡| 天海翼亚洲一区二区三区| 亚洲三级在线| 欧美日韩1区| 久久久久亚洲精品中文字幕| 精品理论电影在线| 美女网站一区| 日韩午夜一区| 亚洲精品乱码日韩| 日本视频在线一区| 国产精品麻豆成人av电影艾秋| 麻豆91精品91久久久的内涵| 免费在线成人| 麻豆国产精品| 国产 日韩 欧美一区| 国产综合婷婷| 99国产精品| 日本强好片久久久久久aaa| 美日韩一区二区三区| 肉色欧美久久久久久久免费看 | 在线手机中文字幕| 99精品视频精品精品视频| 日韩一级不卡| 午夜视频一区二区在线观看| 日韩欧美美女在线观看| 国产精品久久久久久久久久白浆 | 最近高清中文在线字幕在线观看1| 日韩欧美另类一区二区| 国产午夜久久| 91av一区| 丁香六月综合| 国产日韩综合| 国产午夜久久av| 国产精品99视频| 在线一区电影| 青青国产91久久久久久| 精品国产中文字幕第一页| 久久在线视频免费观看| 一区二区亚洲视频| 麻豆91在线播放| 好吊日精品视频| 国产精品亚洲人成在99www| 黄毛片在线观看| 久久午夜精品| 久久三级中文| 亚洲在线观看| 国产乱码精品| 日韩精品水蜜桃| 亚洲精品系列| 国产资源在线观看入口av| 亚洲综合日本| 老司机精品视频网| 在线亚洲精品| 精品中文字幕一区二区三区四区| 欧美日韩激情在线一区二区三区| 日韩精品欧美精品| 秋霞国产精品| 日韩精品91亚洲二区在线观看| 精品久久福利| 久久xxxx| 日韩中文在线电影| 婷婷亚洲精品| 日韩大片免费观看| 日韩精品免费视频一区二区三区| 亚洲黄色影院| 精品亚洲成人| 蜜桃av一区二区在线观看| 国产一区2区| 日本不卡的三区四区五区| 日韩中文在线播放| 国产欧美一区二区精品久久久| 亚洲神马久久| 日韩精品91| 国产日韩视频在线| 亚洲欧洲另类| 三上悠亚国产精品一区二区三区| 中文字幕亚洲影视|