摘要:選擇性編譯是指根據(jù)打包是環(huán)境的不同,選擇性地讓特定的語句有效,讓特定的語句無效。選擇性編譯是筆者自己瞎想出來的名詞,不知道用的對不對。檢測時(shí),還沒有被替換掉。
背景
程序在開發(fā)的過程中,少不了打印調(diào)試用的日志,測試流程時(shí)偽造的數(shù)據(jù)。這些代碼是不能出現(xiàn)在生產(chǎn)環(huán)境上的。
這意味著在程序打包前,需要把相關(guān)代碼剔除掉。
這些事情用人手去做很麻煩,很容易疏漏。而且打包出來測試時(shí)遇到了bug,又得重新把測試代碼添加回去。重復(fù)整個(gè)繁瑣的過程。
既然人工做這么麻煩而且容易出錯(cuò),那能不能用程序幫我們完成這些事情呢?
當(dāng)然可以,用選擇性編譯技術(shù)就行,本文就介紹在webpack下解決這一問題的方法。
其實(shí)這個(gè)方法在webpack官網(wǎng)就有提到,本文只是提供相關(guān)示例及做一些延伸。
選擇性編譯是指根據(jù)打包是環(huán)境的不同,選擇性地讓特定的語句有效,讓特定的語句無效。實(shí)踐過程 首先,先讓我們看一個(gè)選擇性打印日志的示例。
最簡單的例子,在開發(fā)環(huán)境中,我們打印日志,但在生產(chǎn)環(huán)境中,我們讓所有打印日志的語句無效(讓程序不運(yùn)行打印的語句,甚至讓打包出來的文件根本就不包含打印日志的語句)。
選擇性編譯是筆者自己瞎想出來的名詞,不知道用的對不對。
新建一個(gè)vue項(xiàng)目
打開build/webpack.dev.conf.js文件(項(xiàng)目處于開發(fā)環(huán)境時(shí)使用到的webpack打包配置,運(yùn)行npm run dev,這個(gè)文件就會(huì)被執(zhí)行)
添加如下代碼
new webpack.DefinePlugin({ "process.env": config.dev.env, IS_DEV: JSON.stringify(true), }),
打開build/webpack.prod.conf.js文件(項(xiàng)目打包生產(chǎn)環(huán)境時(shí)使用到的webpack打包配置,運(yùn)行npm run build,這個(gè)文件就會(huì)被執(zhí)行)
添加如下代碼
new webpack.DefinePlugin({ "process.env": config.dev.env, IS_DEV: JSON.stringify(false), }),
打開src/main.js文件(項(xiàng)目入口文件,運(yùn)行項(xiàng)目時(shí)被執(zhí)行)
添加如下代碼
if (IS_DEV) { console.log("this is dev env"); } else { console.log("this is prod env"); }
分別在開發(fā)環(huán)境和生產(chǎn)環(huán)境運(yùn)行程序,查看控制臺(tái)。我們發(fā)現(xiàn),在開發(fā)環(huán)境下,打印了this is dev env。在生產(chǎn)環(huán)境下運(yùn)行打印了this is prod env。
打開打包出來的文件/dist/static/js/app.xxx.js.map,搜索this is prod env。我們發(fā)現(xiàn)IS_DEV變成了false。
同理,我們在開發(fā)環(huán)境下,查看chrome開發(fā)者工具,找到相應(yīng)的app.xxx.js文件。我們發(fā)現(xiàn)IS_DEV變成了true。
由此我們可以知道選擇性編譯本質(zhì)上是字符串的替換,先經(jīng)過DefinePlugin對代碼中的特定字符串進(jìn)行替換。再對替換后的代碼進(jìn)行編譯打包。
需要替換的變量需要分別在webpack.dev.conf.js和webpack.prod.conf.js中指定其轉(zhuǎn)換后的意義。
為什么在mian.js中使用IS_DEV,程序不會(huì)報(bào)"IS_DEV is not defined"的錯(cuò)誤?
因?yàn)闉g覽器在運(yùn)行代碼時(shí),拿到的文件里面IS_DEV已經(jīng)被替換成了true或者false,已經(jīng)不存在IS_DEV這個(gè)變量。所以不會(huì)報(bào)錯(cuò)。
DefinePlugin的使用說明除了替換成簡單的布爾值,還可以替換成字符串,數(shù)值,數(shù)組,對象等。
如TEST_DATA: JSON.stringify({name:"momo",age:18}),
為什么需要進(jìn)行JSON.stringify替換?
另外還可以替換成某個(gè)段代碼的值(代碼內(nèi)容直接使用""包裹即可)。
如:TWO: "1+1",TOW將被替換成這段代碼的結(jié)果,即2。
ESLint沖突處理ESLint是一個(gè)用來識(shí)別 ECMAScript 并且按照規(guī)則給出報(bào)告的代碼檢測工具,使用它可以避免低級(jí)錯(cuò)誤和統(tǒng)一代碼的風(fēng)格。
配置了eslint的項(xiàng)目在使用選擇性編譯功能時(shí),可能會(huì)報(bào)出這樣的錯(cuò)誤。
http://exlint.org/docs/rules/no-undef "IS_DEV" is not defined
正如報(bào)錯(cuò)信息所說的,這是由于eslint檢測代碼時(shí),發(fā)現(xiàn)IS_DEV沒有定義(這側(cè)面說明了eslint是先于條件編譯執(zhí)行的。ESLint檢測時(shí),IS_DEV還沒有被替換掉)。解決這個(gè)問題有以下三種方法:
簡單粗暴的方法,修改eslint的配置,將error改為warn唄,先打包了再說。具體操作為修改eslintrc.js文件,如果配置文件里面已有第一條語句"eslint":"recommended"(這是引用默認(rèn)配置)。則往rules里面添加一條規(guī)則"no_undef":"warn",以覆蓋掉默認(rèn)配置。
如果已經(jīng)存在"no-undef"的配置,則直接改為"warn"就行。
上面的方法影響太大,直接修改了規(guī)則。能不能局部修改規(guī)則,只在特定語句中忽略掉該規(guī)則呢?可以的,使用備注包裹需要忽略的語句就行。
但是上面的方法需要在每個(gè)用到的地方都多寫兩條語句,略麻。,有沒有更簡單的,一勞永逸的方法呢?有有有~~ 修改eslintrc.js,將IS_DEV配置成一個(gè)全局變量,之后eslint就不會(huì)認(rèn)為IS_DEV是未定義的變量了。
選擇性編輯技術(shù)的應(yīng)用場景API接口的環(huán)境替換
賬號(hào)信息的模擬,數(shù)據(jù)模擬
日志打印等
彩蛋 更優(yōu)的去除log的方法用if邏輯判斷來輸出log可能略顯繁瑣。其實(shí)對于控制臺(tái)輸出的日志,我們可以通過UglifyJs在打包時(shí)來剔除。
具體操作:
打開build/webpack.prod.conf.js文件,添加如下語句
compress: { warnings: false, // 去除warning警告 drop_debugger: true, // 發(fā)布時(shí)去除debugger語句 drop_console: true // 發(fā)布時(shí)去除console語句 },
如果想只去除console.log,同時(shí)保留console.error等錯(cuò)誤提示。可以指定去除特定的函數(shù)
compress: { warnings: false, // 去除warning警告 pure_funcs: ["console.log"], // 配置發(fā)布時(shí),不被打包的函數(shù) // drop_debugger: true, // 發(fā)布時(shí)去除debugger // drop_console: true // 發(fā)布時(shí)去除console }
注意,添加了這個(gè)配置以后,console.log在打包出來的文件就不存在了。所以前面測試用的this is dev env也會(huì)消失不見
什么叫不可達(dá)代碼?就是無論什么情況下都不會(huì)被運(yùn)行的代碼。
例如我們在條件編譯打包出來的代碼中
if (false) { console.log("this is dev env"); } else { console.log("this is prod env"); }
```console.log("this is dev env");```就是不可達(dá)代碼。 我們同樣可以使用UglifyJs的功能把這部分無用代碼去除掉。讓條件編譯不留痕跡。 具體配置如下: ``` compress: { warnings: false, // 去除warning警告 dead_code: true, // 去除不可達(dá)代碼 }, sourceMap: true ``` ![dead_code](http://upyun.luckly-mjw.cn/Assets/selective-compilation/dead_code.png)UglifyJs的更多功能
UglifyJs還用很多強(qiáng)大的功能,如代碼混淆,壓縮,重拍版等。這里附上UglifyJs官方網(wǎng)址。
英文不好的同學(xué)還可以查看對應(yīng)的中文文檔(其實(shí)這才是重點(diǎn)#機(jī)智臉)。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/89020.html
摘要:使用要給項(xiàng)目構(gòu)建接入動(dòng)態(tài)鏈接庫的思想,需要完成以下事情把網(wǎng)頁依賴的基礎(chǔ)模塊抽離出來,打包到一個(gè)個(gè)單獨(dú)的動(dòng)態(tài)鏈接庫中去。接入已經(jīng)內(nèi)置了對動(dòng)態(tài)鏈接庫的支持,需要通過個(gè)內(nèi)置的插件接入,它們分別是插件用于打包出一個(gè)個(gè)單獨(dú)的動(dòng)態(tài)鏈接庫文件。 webpack優(yōu)化 查看所有文檔頁面:全棧開發(fā),獲取更多信息。原文鏈接:webpack優(yōu)化,原文廣告模態(tài)框遮擋,閱讀體驗(yàn)不好,所以整理成本文,方便查找。 ...
摘要:從零開始搭建同構(gòu)應(yīng)用三配置這篇文章來講解來配置,我們先從最簡單的方法開始,用的方式模擬實(shí)現(xiàn)。影響生產(chǎn)環(huán)境下執(zhí)行效率。最后權(quán)衡下,還是決定使用現(xiàn)在多一套編譯配置的方案。新建,寫入以下內(nèi)容以為例,注意不能少。 從零開始搭建React同構(gòu)應(yīng)用(三):配置SSR 這篇文章來講解來配置server side render,我們先從最簡單的方法開始,用cli的方式模擬實(shí)現(xiàn)SSR。 demo在這里 ...
摘要:配置完成后就可以使用來打包代碼了。值得注意的是會(huì)刪除所有無作用代碼也就是說那些包裹在這些全局變量下的代碼塊都會(huì)被刪除這樣就能保證這些代碼不會(huì)因發(fā)布上線而泄露。默認(rèn)會(huì)從項(xiàng)目的根目錄下引入這些文件。 命令使用 npm install webpack -g 作為全局安裝, 在任意目錄使用 npm install webpack --save-dev 作為項(xiàng)目依賴安裝 np...
摘要:配置以何種方式導(dǎo)出庫。當(dāng)檢測文件不再發(fā)生變化,會(huì)先緩存起來,等等待一段時(shí)間后之后再通知監(jiān)聽者,這個(gè)等待時(shí)間通過配置。發(fā)布到線上給用戶使用的運(yùn)行環(huán)境。 一 縮小文件搜索范圍 1 include & exclude 1) action 限制編譯范圍 2) useage module: { rules: [ { test...
摘要:使用教程一,,,使用教程二版本官方的文檔上一篇講到的是被用來轉(zhuǎn)換某些類型的模塊,它則可以用來做更廣泛的任務(wù)。可看作綁定事件,打包時(shí)會(huì)觸發(fā)事件。它的作用是定義全局常量,是常量。為,啟用兩步之間的延遲。請注意,規(guī)范建議始終使用引號(hào)。 使用教程(一)--- entry,devtool,output,resolve使用教程(二)--- module.loadersWebpack版本 3.10...
閱讀 2498·2021-08-11 11:16
閱讀 2938·2019-08-30 15:55
閱讀 3337·2019-08-30 12:53
閱讀 1578·2019-08-29 13:28
閱讀 3271·2019-08-28 18:17
閱讀 944·2019-08-26 12:19
閱讀 2475·2019-08-23 18:27
閱讀 712·2019-08-23 18:17