摘要:此文已由作者張磊授權(quán)網(wǎng)易云社區(qū)發(fā)布。歡迎訪問(wèn)網(wǎng)易云社區(qū),了解更多網(wǎng)易技術(shù)產(chǎn)品運(yùn)營(yíng)經(jīng)驗(yàn)。前言在對(duì)蜂巢項(xiàng)目從遷移到的過(guò)程中,遇到的問(wèn)題,以及在此過(guò)程中所使用的解決方案。三個(gè)頭是一致的。文章來(lái)源網(wǎng)易云社區(qū)
此文已由作者張磊授權(quán)網(wǎng)易云社區(qū)發(fā)布。
歡迎訪問(wèn)網(wǎng)易云社區(qū),了解更多網(wǎng)易技術(shù)產(chǎn)品運(yùn)營(yíng)經(jīng)驗(yàn)。
前言
在對(duì)蜂巢項(xiàng)目從 nej + regularjs 遷移到 vue 的過(guò)程中,遇到的問(wèn)題,以及在此過(guò)程中所使用的解決方案。
遇到的問(wèn)題
父子頁(yè)面通信
項(xiàng)目分為待重構(gòu)的模塊和已重構(gòu)的模塊,待重構(gòu)的模塊是使用 nej 和 regular ,重構(gòu)的模塊是 vue。頁(yè)面是通過(guò) iframe 引用重構(gòu)的模塊。
這里會(huì)涉及到幾個(gè)問(wèn)題。iframe 和父窗口數(shù)據(jù)交換問(wèn)題、模態(tài)框、以及路由同步的問(wèn)題。 這三個(gè)問(wèn)題的解決方案都是使用了一個(gè)通信機(jī)制。這個(gè)通信機(jī)制對(duì)數(shù)據(jù)進(jìn)行了序列化(在 ie 下,不序列化會(huì)遇到暗坑),所以函數(shù)是無(wú)法進(jìn)行傳遞的,只可以傳遞值,然后通過(guò) JSON.stringify 序列化成字符串,傳遞過(guò)去后再反序列化成相應(yīng)類(lèi)型。采用這種形式,在項(xiàng)目后期,對(duì)代碼進(jìn)行批量修改的時(shí)候查找也很方便,甚至可以將通信機(jī)制二次改造,而不需要改業(yè)務(wù)代碼。
問(wèn)題需要特殊說(shuō)明的有:
模態(tài)框
由于 iframe 并不是鋪滿(mǎn)整個(gè)頁(yè)面,在 iframe 內(nèi)部實(shí)現(xiàn)模態(tài)框的時(shí)候,導(dǎo)致導(dǎo)航欄不會(huì)被覆蓋掉。于是就可以看到頁(yè)面的一部分變灰,但導(dǎo)航欄還是可以點(diǎn)擊的。同時(shí)模態(tài)框的居中是相對(duì)于 iframe 的,所以看起來(lái)也不是特別居中。暫時(shí)解決方案,是使用通信機(jī)制傳參調(diào)用父頁(yè)面的模態(tài)框的邏輯。2. 路由同步
nej 和 vue 都有一套路由方案,但是路由的格式是不一致的,同時(shí)模塊的命名方案也會(huì)不一致,再者 iframe 和 父頁(yè)面路由的變更都需要通知到對(duì)方。
接下來(lái)講通信機(jī)制如何解決問(wèn)題的。
API
/**
* 發(fā)送信息 * @param {string} receiver 收件人 * @param {string} action 描述 * @param {*} [msg] 內(nèi)容 * @param {function} [cb] 回執(zhí)函數(shù) * @param {boolean} [isTemp] 如果收件人不存在,通過(guò)設(shè)置這個(gè)參數(shù)來(lái)指定該消息是丟棄還是保存,當(dāng)未來(lái)某時(shí)刻收件人存在的時(shí)候,會(huì)依次讀取保存的信息,一般不需要指定。 */
send(receiver, action, msg, cb, isTemp) {
},
// Bridge.send("parent", "urlchange", "/module/list");
// Bridge.send("parent", "error", "網(wǎng)絡(luò)出錯(cuò)");
action 可以在父子頁(yè)面的 handles 對(duì)象里注冊(cè)。eg:
// 父頁(yè)面const handles = {
show() { toggle(true); }, hide() { toggle(false); }, hideModal: _u._$hideModal, alert(options) { _u._$alert(options); }, error(msg) { CloudUI.Toast.error(msg); }, confirm(options, cb) { _u._$confirm(Object.assign({ onok() { cb("sub", { msg: true, }); }, oncancel() { cb("sub", { msg: false, }); }, }, options)); },
}
// 子頁(yè)面handles.urlchange = function urlchange(path) {
router.replace(path);
};
再來(lái)一些復(fù)雜的例子
同步調(diào)用
子頁(yè)面向父頁(yè)面?zhèn)鬟f數(shù)據(jù)
Bridge.send("parent", "urlchange", "/module/list");// 這里注意一點(diǎn),為了以后方便,在 vue 模塊內(nèi)部使用的路由均是 vue 的,vue 路由向 nej 路由的轉(zhuǎn)換在 /src/html/module/vue/map.js 進(jìn)行配置,配置信息如下:// {// "/m/module/": "/module/list",// }
父頁(yè)面向子頁(yè)面?zhèn)鬟f數(shù)據(jù)
Bridge.send("sub", "urlchange", "/module/list");// 而在 nej 模塊,寫(xiě)路由就可以隨意點(diǎn),可以直接寫(xiě) vue 的路由,也可以讓其進(jìn)行轉(zhuǎn)換,nej 的模塊在后面均會(huì)丟棄,所以允許隨意一點(diǎn)
異步調(diào)用
// send(receiver, action, msg, cb, isTemp)Bridge.send("parent", "confirm", { content: "所選快照正在維護(hù)中,創(chuàng)建可能需要等待較長(zhǎng)時(shí)間,建議稍后再試。", okButton: "繼續(xù)創(chuàng)建", cancelButton: "稍后再試", primaryButton: "cancelButton",
}, (err, status) => { if (status) { this.create();
} else { this.submitting = false; }
});// 其中第四個(gè)參數(shù)是回調(diào)函數(shù) callback,模仿 nodejs 的實(shí)現(xiàn),err 存在的時(shí)候就是失敗,第二個(gè)參數(shù)是調(diào)用返回的 message??梢詤⒁?jiàn)上面 handles.confirm。
路由
vue 以及 vue-router 支持的異步加載僅僅是組件級(jí)別的,而不是路由級(jí)別的,所以實(shí)現(xiàn)路由級(jí)別的異步加載就會(huì)繞一些。eg:
// 一般實(shí)現(xiàn)const Create = () => import("./create.vue");const List = () => import("./list.vue");
這種方案下,webpack 會(huì)對(duì)每一個(gè)路由進(jìn)行打包,導(dǎo)致一個(gè)路由一個(gè) chunk 的模式,前端加載負(fù)擔(dān)過(guò)大。實(shí)際上,我們需要的粒度可能沒(méi)有這么細(xì),在這里使用一種 vue-router 官方的方案(滑到頁(yè)面底部)。
// 優(yōu)化實(shí)現(xiàn)const Create = () => import(/ webpackChunkName: "a" / "./create.vue");const List = () => import(/ webpackChunkName: "a" / "./list.vue");
這里是使用注釋 / webpackChunkName: "a" / 來(lái)標(biāo)明打入同一個(gè) chunk a 中。唯一的坑點(diǎn)是使用注釋。當(dāng)然還有一種方案進(jìn)行處理。eg:
// index.jsexport { default as create } from "./create.vue";export { default as list } from "./list.vue";// route.jsconst Create = () => import("./index.js").then((modules) => modules.create);
路由寫(xiě)在每個(gè)模塊的下面,只存在一個(gè)文件。
eg:
// 建議- modules
- moduleA - routes.js
不建議在子目錄放置路由,不清晰,完整的路由,可能需要打開(kāi)多個(gè)文件,才能看到
// 不建議- modules
- moduleA - detail - routes.js - routes.js
建議方案會(huì)產(chǎn)生 routes.js 文件變的很龐大的問(wèn)題,不方便查看??蓞⒖既缦聦?xiě)法,可以緩解此問(wèn)題:
const routesA = [];const routesB = [];export default [
...routesA, ...routesB,
]
路由的劃分問(wèn)題
eg:
// 不建議const router = [
{ path: "/", component: () => import(/* webpackChunkName: "module" */ "./list.vue"), children: [ { path: "tab1", name: "module.list.tab1", // ... }, { path: "tab2", name: "module.list.tab2", // ... }, { path: "tab3", name: "module.list.tab3", // ... }, ], }, { path: "tab1/edit", name: "module.edit.tab1", // ... }
];
module 模塊頁(yè),有三個(gè) tab。三個(gè) tab 頭是一致的。 list.vue 的代碼僅僅是實(shí)現(xiàn)了 3個(gè) tab 一致的部分即頭部。觀察 tab1/edit 和 tab1,在邏輯層面上它們應(yīng)該被放到一起,但是 path: "/" 所在的組件的 dom 中含有 3個(gè) tab 的頭,導(dǎo)致沒(méi)有辦法寫(xiě)在一起(寫(xiě)在一起的話(huà)同時(shí)會(huì)繼承頭部),權(quán)限控制更顯麻煩。更好的做法是 path: "/" 這一層級(jí)不做任何 dom 相關(guān)的東西,寫(xiě)到每個(gè) tab 內(nèi)部。
// 建議const tab1 = { path: "tab1", // 權(quán)限控制 tab1 的準(zhǔn)入 children: [
{ path: "list", name: "module.tab1.list", // ... }, { path: "edit", name: "module.tab1.edit", // ... }, ],
};
const router = [
{ path: "/", // 權(quán)限控制 module 的準(zhǔn)入 children: [ tab1, { path: "tab2", name: "module.tab1.list", // ... }, { path: "tab3", name: "module.tab3.list", // ... }, ], }
];
當(dāng)然可以根據(jù)權(quán)限控制進(jìn)行調(diào)整,寫(xiě)法不是很固定。交互可能不太喜歡定義 path: "list", 但是第一種寫(xiě)法,相當(dāng)于污染了整個(gè)路由的頂層,那后面必須定義多個(gè)頂層進(jìn)行覆蓋,由模塊單入口路由變成了模塊多入口路由。
openapi 和 webapi 數(shù)據(jù)轉(zhuǎn)換
舉例說(shuō)明:在模塊從 webapi 遷移到 openapi 的時(shí)候使用了一種方案,在數(shù)據(jù)獲取層面對(duì)數(shù)據(jù)進(jìn)行轉(zhuǎn)換。即:
// openapiconst result1 = { Id: 1, Name: 2,
};// webapiconst result2 = { id: 1, name: 2,
};// transform(result1) 后的數(shù)據(jù)結(jié)構(gòu)包含 result2 中有用的數(shù)據(jù)結(jié)構(gòu)// 這個(gè) transform 函數(shù)會(huì)將 openapi 的數(shù)據(jù)轉(zhuǎn)換成 webapi 的,這樣只需要改數(shù)據(jù)結(jié)構(gòu),讓新老保持一致,再修改少量的業(yè)務(wù)邏輯即可完成接口遷移工作。
這個(gè)問(wèn)題本身屬于后端接口變更,與框架遷移屬于并行任務(wù),多帶帶拿來(lái)看并無(wú)關(guān)聯(lián),問(wèn)題放在一起的時(shí)候,就變得棘手了。
在對(duì) win 模塊進(jìn)行遷移的時(shí)候,在使用 vue 的時(shí)候希望接口方面使用 opeanpi 的數(shù)據(jù),不進(jìn)行數(shù)據(jù)轉(zhuǎn)換。但是在使用老模塊的時(shí)候,為了盡量少的動(dòng)業(yè)務(wù)代碼,對(duì) opeanpi 的數(shù)據(jù)進(jìn)行了轉(zhuǎn)換,那就意味著兩者的數(shù)據(jù)的并不一致,在使用上面提到的通信機(jī)制(調(diào)用父頁(yè)面的模態(tài)框,需要傳遞數(shù)據(jù))的時(shí)候,這就很致命了,意味著一方需要再做一次數(shù)據(jù)轉(zhuǎn)換。目前代碼是 vue 模塊手工硬編碼轉(zhuǎn)換的,后面可以把這部分放到 nej ,可以借用其已有的接口的數(shù)據(jù)轉(zhuǎn)換函數(shù),對(duì) vue 傳遞的數(shù)據(jù)進(jìn)行二次轉(zhuǎn)換。
另外還有一個(gè)問(wèn)題,如果模塊先進(jìn)行框架遷移,后進(jìn)行接口遷移,此時(shí)就面臨兩個(gè)方案。一個(gè)是使用 transform 函數(shù)對(duì)數(shù)據(jù)進(jìn)行轉(zhuǎn)換,另一種,推到重寫(xiě)。此時(shí)肯定更傾向于第一種方案,那么就需要對(duì) transform 函數(shù)進(jìn)行設(shè)計(jì),讓其更方便使用。這里簡(jiǎn)單設(shè)計(jì)了一種。
const source = { standard: { bandwidth: 1, ipChargeType: 2
}, instanceId: 6,
};
const rules = { standard: "NewStandard", instanceId: "InstanceId", "standard.bandwidth": ["InternetMaxBandwidth", "BizParam.InternetMaxBandwidth"], "standard.ipChargeType": "BizParam.NetworkChargeType",
};
const out = { NewStandard: { bandwidth: 1, ipChargeType: 2
}, InstanceId: 6, BizParam: { InternetMaxBandwidth: 1, NetworkChargeType: 2, }, InternetMaxBandwidth: 1,
};
it(transform(source, rules) is correct, () => { assert.deepEqual(transform(source, rules), Object.assign({}, source, out));
});
it(transform(source, rules, true) is correct, () => { assert.deepEqual(transform(out, rules, true), Object.assign({}, out, source));
});
靜態(tài)資源
這里主要是 js 文件內(nèi)引用的靜態(tài)資源,該靜態(tài)資源的路徑需要用此語(yǔ)法進(jìn)行設(shè)置
default: { logo: require(@/assets/images/logos/logo.png), // @是項(xiàng)目根路徑},
這樣這個(gè)靜態(tài)文件就可以享受到 webpack 的處理,算出正確的路徑,不然有可能出現(xiàn)顯示不出來(lái)的情況。 另外靜態(tài)資源不推薦寫(xiě)相對(duì)路徑。eg: ../../../assets/images/logos/logo.png
接口
nej + regular 在代碼里寫(xiě)了接口,在 vue 需要再次找到接口,重新寫(xiě)一遍,基本不可復(fù)用。但如果之前放置接口的地方稍顯混亂,那么在找接口的時(shí)候,就需要一個(gè)個(gè)業(yè)務(wù)邏輯的看過(guò)去。試想可不可以
將接口按照模塊放置在一起,稱(chēng)為 api 層,同時(shí)再劃分出來(lái) service 層。api 層通過(guò) json 來(lái)描述一個(gè)接口的方方面面, service 層是從 api 層生成出來(lái)的,外加上對(duì)接口進(jìn)行二次處理和對(duì)多個(gè)接口拼接。
然后就很好的抽象出一個(gè)獨(dú)立的 api 層,和業(yè)務(wù)邏輯無(wú)關(guān),僅和后端文檔輸出有關(guān),同時(shí) service 層又很好的保持一定的業(yè)務(wù)相關(guān)性。那么在換框架的時(shí)候,api 直接拿走即可,service 層僅需要稍許改動(dòng)。另有文章介紹具體的實(shí)現(xiàn)。
這里可能會(huì)有接口數(shù)據(jù)緩存以及一定時(shí)間內(nèi)只發(fā)一次請(qǐng)求的需求,那么想一下,需要在該層實(shí)現(xiàn)嗎?還是需要更高的一層對(duì)數(shù)據(jù)處理的抽象,而不受限于僅對(duì)接口數(shù)據(jù)?又或者對(duì) service 層的定義進(jìn)行擴(kuò)充,包含對(duì)數(shù)據(jù)處理的抽象?
網(wǎng)易云計(jì)算基礎(chǔ)服務(wù)深度整合了 IaaS、PaaS 及容器技術(shù),提供彈性計(jì)算、DevOps 工具鏈及微服務(wù)基礎(chǔ)設(shè)施等服務(wù),幫助企業(yè)解決 IT、架構(gòu)及運(yùn)維等問(wèn)題,使企業(yè)更聚焦于業(yè)務(wù),是新一代的云計(jì)算平臺(tái),點(diǎn)擊可免費(fèi)試用。
文章來(lái)源: 網(wǎng)易云社區(qū)
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/25269.html
摘要:本文將介紹網(wǎng)易云基礎(chǔ)服務(wù)蜂巢實(shí)例遷移功能的實(shí)現(xiàn),并探討如何高效完成實(shí)例遷移任務(wù)。網(wǎng)易云基礎(chǔ)服務(wù)蜂巢提供了負(fù)載監(jiān)控閾值選項(xiàng),在業(yè)務(wù)負(fù)載超過(guò)該閾值時(shí),會(huì)暫停遷移操作,直到負(fù)載重新低于閾值。 歡迎訪問(wèn)網(wǎng)易云社區(qū),了解更多網(wǎng)易技術(shù)產(chǎn)品運(yùn)營(yíng)經(jīng)驗(yàn)。 我們把數(shù)據(jù)庫(kù)里部分或全部 Schema和數(shù)據(jù)遷移到另一個(gè)實(shí)例的行為稱(chēng)為實(shí)例遷移,將導(dǎo)出數(shù)據(jù)的實(shí)例稱(chēng)為源實(shí)例,導(dǎo)入數(shù)據(jù)的實(shí)例稱(chēng)為目標(biāo)實(shí)例。 根據(jù)遷移數(shù)...
摘要:最后,張曉龍透露未來(lái)網(wǎng)易云會(huì)在以下三個(gè)方面繼續(xù)深耕研發(fā)高性能容器,跟進(jìn)開(kāi)源社區(qū)最新版本并適配,加大參與社區(qū)力度并反饋社區(qū)。文章來(lái)源網(wǎng)易云社區(qū) 歡迎訪問(wèn)網(wǎng)易云社區(qū),了解更多網(wǎng)易技術(shù)產(chǎn)品運(yùn)營(yíng)經(jīng)驗(yàn)。 10 月 15 日,聚焦 Kubernetes 中國(guó)行業(yè)應(yīng)用與技術(shù)落地的首屆中國(guó) Kubernetes 用戶(hù)大會(huì)(KEUC)在杭州成功舉辦。本次大會(huì)吸引了來(lái)自全球各地的技術(shù)精英齊聚一堂,共同探...
摘要:作為為應(yīng)用平臺(tái)即時(shí)通訊技術(shù)開(kāi)發(fā)和部署提供便捷途徑的云平臺(tái),網(wǎng)易云聚焦穩(wěn)定性和易用性,通過(guò)強(qiáng)化私有云定制化的服務(wù)來(lái)滿(mǎn)足不同用戶(hù)的需求,而這正是網(wǎng)易云作為即時(shí)通訊領(lǐng)域的老兵的競(jìng)爭(zhēng)優(yōu)勢(shì)所在。9月22日,2017中國(guó)移動(dòng)互聯(lián)網(wǎng)大會(huì)暨中國(guó)首席技術(shù)官大會(huì)在杭州舉行。眾多業(yè)界技術(shù)領(lǐng)袖參與了此次盛會(huì),就互聯(lián)網(wǎng)時(shí)代的IT技術(shù)研發(fā)挑戰(zhàn)和應(yīng)對(duì)策略進(jìn)行了探討。網(wǎng)易技術(shù)委員會(huì)資深專(zhuān)家委員徐杭生在會(huì)議上發(fā)表演講,認(rèn)為移...
摘要:在互聯(lián)網(wǎng)的第三個(gè)時(shí)代,也就是下一個(gè)互聯(lián)網(wǎng)十年里,云計(jì)算將成為這場(chǎng)大戰(zhàn)的制勝關(guān)鍵所在。就在前不久,亞馬遜旗下的云計(jì)算服務(wù)平臺(tái)宣布正式在中國(guó)商用。的殺入對(duì)于中國(guó)的云計(jì)算市場(chǎng)也是一大不可忽視的力量?! 』ヂ?lián)網(wǎng)的第一個(gè)時(shí)代我們定義為PC互聯(lián)網(wǎng),互聯(lián)網(wǎng)的第二個(gè)時(shí)代我們定義為移動(dòng)互聯(lián)網(wǎng),而互聯(lián)網(wǎng)的第三個(gè)時(shí)代我們則定義為萬(wàn)物聯(lián)網(wǎng)。當(dāng)前國(guó)內(nèi)的互聯(lián)網(wǎng)正處于第二個(gè)時(shí)代向第三個(gè)時(shí)代過(guò)渡期,而云計(jì)算則是支撐起萬(wàn)物聯(lián)...
摘要:網(wǎng)易云信即時(shí)通訊云服務(wù)的產(chǎn)品優(yōu)勢(shì)網(wǎng)易云信涉足多元行業(yè)提升用戶(hù)體驗(yàn)滿(mǎn)足內(nèi)外協(xié)同今年月,網(wǎng)易云信正式宣布通信與視頻業(yè)務(wù)實(shí)現(xiàn)戰(zhàn)略升級(jí)?! №n寒導(dǎo)演的影片《乘風(fēng)破浪》中,有這樣一個(gè)片段,在六一還堅(jiān)持囤BB機(jī)有前途的時(shí)候,從事電腦編程和軟件開(kāi)發(fā)的小馬則認(rèn)為即時(shí)通訊才是未來(lái)的發(fā)展方向??吹竭@里,很多觀影者都不謀而合地笑了,小馬口中的即時(shí)通訊就是在1999年誕生的OICQ。影片中阿浪說(shuō)的那句話(huà)沒(méi)錯(cuò),這個(gè)世...
閱讀 1785·2021-11-15 11:37
閱讀 3056·2021-11-04 16:05
閱讀 1922·2021-10-27 14:18
閱讀 2755·2021-08-12 13:30
閱讀 2500·2019-08-29 14:18
閱讀 2086·2019-08-29 13:07
閱讀 2024·2019-08-27 10:54
閱讀 2726·2019-08-26 12:15