vue多頁面開發(fā)和打包正確處理方法

前段時(shí)間做項(xiàng)目,技術(shù)棧是vue+webpack,主要就是官網(wǎng)首頁加后臺管理系統(tǒng) 根據(jù)當(dāng)時(shí)情況,分析出三種方案
- 一個(gè)項(xiàng)目代碼里面嵌兩個(gè)spa應(yīng)用(官網(wǎng)和后臺系統(tǒng))
- 分開兩套項(xiàng)目源碼
- 一套項(xiàng)目源碼里面就一個(gè)spa應(yīng)用
思考:
- 直接否定了一套項(xiàng)目源碼里一個(gè)spa應(yīng)用(ui樣式會相互覆蓋,如果沒有代碼規(guī)范后期比較難維護(hù))
- 兩套源碼的話,后臺可能開兩個(gè)端口,然后需要用nginx反向代理可能比較麻煩,而且前端開發(fā)也比較麻煩麻煩,畢竟需要維護(hù)兩個(gè)git倉庫,兩套git上線流程,可能會損耗很多時(shí)間。
- 對自己的技術(shù)(盲目)自信,也想嘗嘗鮮,分析出需求也不算很復(fù)雜。選了第一種方案,就是多個(gè)單頁面應(yīng)用在一套源碼里面
上一張多頁面的結(jié)構(gòu)圖
下載vue spa模板
npm install vue-cli -g vue init webpack multiple-vue-amazing
改造多頁面應(yīng)用
npm install glob --save-dev
修改src文件夾下面的目錄結(jié)構(gòu)
在util.js里面加入
/* 這里是添加的部分 ---------------------------- 開始 */
// glob是webpack安裝時(shí)依賴的一個(gè)第三方模塊,還模塊允許你使用 *等符號, 例如lib/*.js就是獲取lib文件夾下的所有js后綴名的文件
var glob = require('glob')
// 頁面模板
var HtmlWebpackPlugin = require('html-webpack-plugin')
// 取得相應(yīng)的頁面路徑,因?yàn)橹暗呐渲?,所以是src文件夾下的pages文件夾
var PAGE_PATH = path.resolve(__dirname, '../src/pages')
// 用于做相應(yīng)的merge處理
var merge = require('webpack-merge')
//多入口配置
// 通過glob模塊讀取pages文件夾下的所有對應(yīng)文件夾下的js后綴文件,如果該文件存在
// 那么就作為入口處理
exports.entries = function () {
var entryFiles = glob.sync(PAGE_PATH + '/*/*.js')
var map = {}
entryFiles.forEach((filePath) => {
var filename = filePath.substring(filePath.lastIndexOf('\/') + 1, filePath.lastIndexOf('.'))
map[filename] = filePath
})
return map
}
//多頁面輸出配置
// 與上面的多頁面入口配置相同,讀取pages文件夾下的對應(yīng)的html后綴文件,然后放入數(shù)組中
exports.htmlPlugin = function () {
let entryHtml = glob.sync(PAGE_PATH + '/*/*.html')
let arr = []
entryHtml.forEach((filePath) => {
let filename = filePath.substring(filePath.lastIndexOf('\/') + 1, filePath.lastIndexOf('.'))
let conf = {
// 模板來源
template: filePath,
// 文件名稱
filename: filename + '.html',
// 頁面模板需要加對應(yīng)的js腳本,如果不加這行則每個(gè)頁面都會引入所有的js腳本
chunks: ['manifest', 'vendor', filename],
inject: true
}
if (process.env.NODE_ENV === 'production') {
conf = merge(conf, {
minify: {
removeComments: true,
collapseWhitespace: true,
removeAttributeQuotes: true
},
chunksSortMode: 'dependency'
})
}
arr.push(new HtmlWebpackPlugin(conf))
})
return arr
}
/* 這里是添加的部分 ---------------------------- 結(jié)束 */
webpack.base.conf.js 文件
/* 修改部分 ---------------- 開始 */
entry: utils.entries(),
/* 修改部分 ---------------- 結(jié)束 */
webpack.dev.conf.js 文件
/* 注釋這個(gè)區(qū)域的文件 ------------- 開始 */
// new HtmlWebpackPlugin({
// filename: 'index.html',
// template: 'index.html',
// inject: true
// }),
/* 注釋這個(gè)區(qū)域的文件 ------------- 結(jié)束 */
new FriendlyErrorsPlugin()
/* 添加 .concat(utils.htmlPlugin()) ------------------ */
].concat(utils.htmlPlugin())
webpack.prod.conf.js 文件
/* 注釋這個(gè)區(qū)域的內(nèi)容 ---------------------- 開始 */
// new HtmlWebpackPlugin({
// filename: config.build.index,
// template: 'index.html',
// inject: true,
// minify: {
// removeComments: true,
// collapseWhitespace: true,
// removeAttributeQuotes: true
// // more options:
// // https://github.com/kangax/html-minifier#options-quick-reference
// },
// // necessary to consistently work with multiple chunks via CommonsChunkPlugin
// chunksSortMode: 'dependency'
// }),
/* 注釋這個(gè)區(qū)域的內(nèi)容 ---------------------- 結(jié)束 */
// copy custom static assets
new CopyWebpackPlugin([
{
from: path.resolve(__dirname, '../static'),
to: config.build.assetsSubDirectory,
ignore: ['.*']
}
])
/* 該位置添加 .concat(utils.htmlPlugin()) ------------------- */
].concat(utils.htmlPlugin())
引入第三方ui庫
npm install element-ui bootstrap-vue --save
分別在不同的頁面引入不同的ui index.js
import BootstrapVue from 'bootstrap-vue' Vue.use(BootstrapVue)
admin.js
import ElementUI from 'element-ui' import 'element-ui/lib/theme-chalk/index.css' Vue.use(ElementUI)
上面多頁面的配置是參考網(wǎng)上的,而且網(wǎng)上的思路大都很相似,核心就是改多個(gè)entry,配置完成了之后,開發(fā)的時(shí)候也是發(fā)現(xiàn)不了問題的,然后大概就開發(fā)了一個(gè)月,開發(fā)完之后對官網(wǎng)進(jìn)行性能分析時(shí)發(fā)現(xiàn),webpack打包的vendor.js網(wǎng)絡(luò)加載時(shí)間特別長,導(dǎo)致首屏的白屏?xí)r間非常長,最終通過-webpack-bundle-analyzer分析得到了結(jié)論
npm run build --report
你會發(fā)現(xiàn)vendor.js包含了index.html和admin.html的共同部分,所以這個(gè)vendor包注定會很大很冗余
解決思路
既然是vendor過大引起加載速度慢,那就分離這個(gè)vendor就好了。我是這樣想的,把各個(gè)頁面中都使用到的第三方代碼提取至vendor.js中,然后各個(gè)頁面中用到的第三方代碼再打包成各自的vendor-x.js,例如現(xiàn)有頁面index.html、admin.html,則最終會打包出vendor.js、vendor-index.js、vendor-admin.js。
webpack.prod.conf.js 文件
new webpack.optimize.CommonsChunkPlugin({
name: 'vendor-admin',
chunks: ['vendor'],
minChunks: function (module, count) {
return (
module.resource &&
/\.js$/.test(module.resource) &&
module.resource.indexOf(path.join(__dirname, '../node_modules')) === 0 &&
module.resource.indexOf('element-ui') != -1
)
}
}),
new webpack.optimize.CommonsChunkPlugin({
name: 'vendor-index',
chunks: ['vendor'],
minChunks: function (module, count) {
return (
module.resource &&
/\.js$/.test(module.resource) &&
module.resource.indexOf(path.join(__dirname, '../node_modules')) === 0 &&
module.resource.indexOf('bootstrap-vue') != -1
)
}
}),
再次分析,一切都很ok,vendor.js被分離成了vendor.js、vendor-index、vendor-admin.js
本來以為解決了CommonsChunkPlugin的分離vendor.js的問題,就可以了,然后打包出來發(fā)現(xiàn)index.html和admin.html都少了一個(gè)引入(各自對應(yīng)的那個(gè)vendor-xx.js)
解決方案
這個(gè)問題其實(shí)就是HtmlWebpackPlugin的問題 把原來的 chunksSortMode: 'dependency'改成自定義函數(shù)的配置,如下
util.js文件
chunksSortMode: function (chunk1, chunk2) {
var order1 = chunks.indexOf(chunk1.names[0])
var order2 = chunks.indexOf(chunk2.names[0])
return order1 - order2
},
最終實(shí)現(xiàn)
- 每個(gè)頁面加載各自的chunk
- 每個(gè)頁面有不同的參數(shù)
- 每個(gè)頁面能共享公共chunk
- 瀏覽器緩存,性能更好
- 如果還嫌慢的話,開啟gzip
感想
大功告成了,雖然配置看起來很簡單,不過我當(dāng)時(shí)開發(fā)的時(shí)候,思考了很久,所以假如你CommonsChunkPlugin和HtmlWebpackPlugin不熟悉或者只會用別人第三方的配置表,估計(jì)會踩大坑,比如說,CommonsChunkPlugin不指定chunks,默認(rèn)是什么?minChunks大多數(shù)人只會寫一個(gè)數(shù)值,然而自定義一個(gè)函數(shù)的寫法其實(shí)才是最強(qiáng)大的,根據(jù)我個(gè)人的經(jīng)驗(yàn)chunks結(jié)合minChunks自定義函數(shù)的寫法,能解決幾乎所有CommonsChunkPlugin靈異的事件。
總結(jié)
以上所述是小編給大家介紹的vue多頁面開發(fā)和打包正確處理方法,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時(shí)回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
相關(guān)文章
vant/vue手機(jī)端長按事件以及禁止長按彈出菜單實(shí)現(xiàn)方法詳解
這篇文章主要介紹了vant/vue手機(jī)端長按事件以及禁止長按彈出菜單實(shí)現(xiàn)方法詳解,需要的朋友可以參考下2022-12-12
vue動(dòng)態(tài)綁定class選中當(dāng)前列表變色的方法示例
這篇文章主要介紹了vue動(dòng)態(tài)綁定class選中當(dāng)前列表變色的方法示例,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-12-12
vue3循環(huán)設(shè)置ref并獲取的解決方案
我們在平時(shí)做業(yè)務(wù)的時(shí)候,父子組件通信會經(jīng)常用到ref,這篇文章主要給大家介紹了關(guān)于vue3循環(huán)設(shè)置ref并獲取的解決方案,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-02-02
vue實(shí)現(xiàn)動(dòng)態(tài)路由的方法及路由原理解析
這篇文章主要介紹了路由原理及vue實(shí)現(xiàn)動(dòng)態(tài)路由,Vue Router 提供了豐富的 API,可以輕松地實(shí)現(xiàn)路由功能,并支持路由參數(shù)、查詢參數(shù)、命名路由、嵌套路由等功能,可以滿足不同應(yīng)用程序的需求,需要的朋友可以參考下2023-06-06
使用vite創(chuàng)建vue3之vite.config.js的配置方式
這篇文章主要介紹了使用vite創(chuàng)建vue3之vite.config.js的配置方式,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-08-08
Vue?warn:Property?"state"?was?accessed?during
這篇文章主要為大家介紹了Vue?warn:Property?"state"?was?accessed?during?render的報(bào)錯(cuò)解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06

