چگونه از وب پک برای کوچک کردن برنامه خود تا حد امکان استفاده کنید
یکی از اولین کارهایی که هنگام بهینه سازی یک برنامه باید انجام دهید این است که آن را تا حد امکان کوچک کنید. در اینجا نحوه انجام این کار با وب پک آورده شده است.
از حالت تولید استفاده کنید (فقط بسته وب 4)
Webpack 4 پرچم mode
جدید را معرفی کرد. میتوانید این پرچم را روی 'development'
یا 'production'
تنظیم کنید تا به پک وب اشاره کنید که برنامه را برای یک محیط خاص میسازید:
// webpack.config.js
module.exports = {
mode: 'production',
};
هنگامی که برنامه خود را برای تولید میسازید، مطمئن شوید که حالت production
را فعال کنید. این باعث میشود وبپک بهینهسازیهایی مانند کوچکسازی، حذف کدهای توسعهدهنده در کتابخانهها و موارد دیگر را اعمال کند.
در ادامه مطلب
کوچک سازی را فعال کنید
کوچکسازی زمانی است که کد را با حذف فضاهای اضافی، کوتاه کردن نام متغیرها و غیره فشرده میکنید. مثل این:
// Original code
function map(array, iteratee) {
let index = -1;
const length = array == null ? 0 : array.length;
const result = new Array(length);
while (++index < length) {
result[index] = iteratee(array[index], index, array);
}
return result;
}
↓
// Minified code
function map(n,r){let t=-1;for(const a=null==n?0:n.length,l=Array(a);++t<a;)l[t]=r(n[t],t,n);return l}
Webpack از دو روش برای کوچکسازی کد پشتیبانی میکند: گزینههای کوچکسازی در سطح بسته و گزینههای خاص لودر . آنها باید به طور همزمان استفاده شوند.
کوچک سازی در سطح بسته
کوچک سازی سطح بسته نرم افزاری، کل بسته نرم افزاری را پس از کامپایل فشرده می کند. در اینجا نحوه کار آن آمده است:
شما کدهای زیر را می نویسید:
// comments.js import './comments.css'; export function render(data, target) { console.log('Rendered!'); }
Webpack آن را تقریباً به موارد زیر کامپایل می کند:
// bundle.js (part of) "use strict"; Object.defineProperty(__webpack_exports__, "__esModule", { value: true }); /* harmony export (immutable) */ __webpack_exports__["render"] = render; /* harmony import */ var __WEBPACK_IMPORTED_MODULE_0__comments_css__ = __webpack_require__(1); /* harmony import */ var __WEBPACK_IMPORTED_MODULE_0__comments_css_js___default = __webpack_require__.n(__WEBPACK_IMPORTED_MODULE_0__comments_css__); function render(data, target) { console.log('Rendered!'); }
یک مینیفایر آن را تقریباً به شکل زیر فشرده میکند:
// minified bundle.js (part of) "use strict";function t(e,n){console.log("Rendered!")} Object.defineProperty(n,"__esModule",{value:!0}),n.render=t;var o=r(1);r.n(o)
در وب پک 4، کوچکسازی سطح بستهای به طور خودکار فعال میشود - هم در حالت تولید و هم بدون آن. از مینیفایر UglifyJS در زیر کاپوت استفاده میکند. (اگر نیاز به غیرفعال کردن minification دارید، فقط از حالت توسعه استفاده کنید یا false
به گزینه optimization.minimize
ارسال کنید.)
در وب پک 3، باید مستقیماً از افزونه UglifyJS استفاده کنید. این افزونه همراه با بسته وب ارائه می شود. برای فعال کردن آن، آن را به بخش plugins
پیکربندی اضافه کنید:
// webpack.config.js
const webpack = require('webpack');
module.exports = {
plugins: [
new webpack.optimize.UglifyJsPlugin(),
],
};
گزینه های خاص لودر
راه دوم برای کوچک کردن کد، گزینه های خاص لودر است ( لودر چیست ). با گزینههای لودر، میتوانید مواردی را که مینیفایر نمیتواند کوچکسازی کند، فشرده کنید. به عنوان مثال، هنگامی که یک فایل CSS را با css-loader
وارد می کنید، فایل به یک رشته کامپایل می شود:
/* comments.css */
.comment {
color: black;
}
// minified bundle.js (part of)
exports=module.exports=__webpack_require__(1)(),
exports.push([module.i,".comment {\r\n color: black;\r\n}",""]);
Minifier نمی تواند این کد را فشرده کند زیرا یک رشته است. برای کوچک کردن محتوای فایل، باید لودر را برای انجام این کار پیکربندی کنیم:
// webpack.config.js
module.exports = {
module: {
rules: [
{
test: /\.css$/,
use: [
'style-loader',
{ loader: 'css-loader', options: { minimize: true } },
],
},
],
},
};
در ادامه مطلب
- اسناد UglifyJsPlugin
- سایر مینیفایرهای محبوب: Babel Minify ، Google Closure Compiler
NODE_ENV=production
مشخص کنید
راه دیگر برای کاهش اندازه جلویی این است که متغیر محیطی NODE_ENV
را در کد خود روی مقدار production
تنظیم کنید.
کتابخانه ها متغیر NODE_ENV
را می خوانند تا تشخیص دهند که در کدام حالت باید کار کنند - در توسعه یا تولید. برخی از کتابخانه ها بر اساس این متغیر رفتار متفاوتی دارند. به عنوان مثال، زمانی که NODE_ENV
روی production
تنظیم نشده باشد، Vue.js بررسی های اضافی انجام می دهد و هشدارها را چاپ می کند:
// vue/dist/vue.runtime.esm.js
// …
if (process.env.NODE_ENV !== 'production') {
warn('props must be strings when using array syntax.');
}
// …
React به طور مشابه کار می کند - یک ساخت توسعه را بارگیری می کند که شامل هشدارها است:
// react/index.js
if (process.env.NODE_ENV === 'production') {
module.exports = require('./cjs/react.production.min.js');
} else {
module.exports = require('./cjs/react.development.js');
}
// react/cjs/react.development.js
// …
warning$3(
componentClass.getDefaultProps.isReactClassApproved,
'getDefaultProps is only used on classic React.createClass ' +
'definitions. Use a static property named `defaultProps` instead.'
);
// …
این گونه بررسی ها و هشدارها معمولاً در تولید غیر ضروری هستند، اما در کد باقی می مانند و حجم کتابخانه را افزایش می دهند. در وب پک 4، آنها را با افزودن گزینه optimization.nodeEnv: 'production'
حذف کنید:
// webpack.config.js (for webpack 4)
module.exports = {
optimization: {
nodeEnv: 'production',
minimize: true,
},
};
در وب پک 3، به جای آن از DefinePlugin
استفاده کنید:
// webpack.config.js (for webpack 3)
const webpack = require('webpack');
module.exports = {
plugins: [
new webpack.DefinePlugin({
'process.env.NODE_ENV': '"production"'
}),
new webpack.optimize.UglifyJsPlugin()
]
};
هر دو گزینه optimization.nodeEnv
و DefinePlugin
به یک شکل کار می کنند - آنها همه رخدادهای process.env.NODE_ENV
را با مقدار مشخص شده جایگزین می کنند. با کانفیگ بالا:
Webpack همه موارد مربوط به
process.env.NODE_ENV
را با"production"
جایگزین می کند:// vue/dist/vue.runtime.esm.js if (typeof val === 'string') { name = camelize(val); res[name] = { type: null }; } else if (process.env.NODE_ENV !== 'production') { warn('props must be strings when using array syntax.'); }
↓
// vue/dist/vue.runtime.esm.js if (typeof val === 'string') { name = camelize(val); res[name] = { type: null }; } else if ("production" !== 'production') { warn('props must be strings when using array syntax.'); }
و سپس مینیفایر تمام
if
شاخهها را حذف میکند - زیرا"production" !== 'production'
همیشه نادرست است، و افزونه میداند که کد داخل این شاخهها هرگز اجرا نمیشود:// vue/dist/vue.runtime.esm.js if (typeof val === 'string') { name = camelize(val); res[name] = { type: null }; } else if ("production" !== 'production') { warn('props must be strings when using array syntax.'); }
↓
// vue/dist/vue.runtime.esm.js (without minification) if (typeof val === 'string') { name = camelize(val); res[name] = { type: null }; }
در ادامه مطلب
- "متغیرهای محیطی" چیست؟
- اسناد Webpack در مورد:
DefinePlugin
،EnvironmentPlugin
از ماژول های ES استفاده کنید
راه بعدی برای کاهش اندازه جلویی استفاده از ماژولهای ES است.
وقتی از ماژولهای ES استفاده میکنید، وبپک قادر به انجام تکان دادن درخت میشود. تکان دادن درخت زمانی است که یک باندلر کل درخت وابستگی را طی می کند، وابستگی هایی را که استفاده می شود بررسی می کند و وابستگی های استفاده نشده را حذف می کند. بنابراین، اگر از نحو ماژول ES استفاده می کنید، وب پک می تواند کدهای استفاده نشده را حذف کند:
شما یک فایل با چندین صادرات می نویسید، اما برنامه فقط از یکی از آنها استفاده می کند:
// comments.js export const render = () => { return 'Rendered!'; }; export const commentRestEndpoint = '/rest/comments'; // index.js import { render } from './comments.js'; render();
Webpack می داند که
commentRestEndpoint
استفاده نمی شود و یک نقطه صادرات جداگانه در بسته ایجاد نمی کند:// bundle.js (part that corresponds to comments.js) (function(module, __webpack_exports__, __webpack_require__) { "use strict"; const render = () => { return 'Rendered!'; }; /* harmony export (immutable) */ __webpack_exports__["a"] = render; const commentRestEndpoint = '/rest/comments'; /* unused harmony export commentRestEndpoint */ })
Minifier متغیر استفاده نشده را حذف می کند:
// bundle.js (part that corresponds to comments.js) (function(n,e){"use strict";var r=function(){return"Rendered!"};e.b=r})
اگر کتابخانه ها با ماژول های ES نوشته شده باشند، این کار حتی با کتابخانه ها نیز کار می کند.
لازم نیست دقیقاً از مینیفایر داخلی پک ( UglifyJsPlugin
) استفاده کنید. هر کوچک کننده ای که از حذف کد مرده پشتیبانی می کند (به عنوان مثال افزونه Babel Minify یا افزونه Google Closure Compiler ) این کار را انجام می دهد.
در ادامه مطلب
اسناد Webpack در مورد تکان دادن درخت
بهینه سازی تصاویر
تصاویر بیش از نیمی از اندازه صفحه را تشکیل می دهند. در حالی که آنها به اندازه جاوا اسکریپت حیاتی نیستند (مثلاً رندر را مسدود نمی کنند)، هنوز هم بخش بزرگی از پهنای باند را می خورند. از url-loader
، svg-url-loader
و image-webpack-loader
برای بهینه سازی آنها در بسته وب استفاده کنید.
url-loader
فایل های استاتیک کوچک را در برنامه قرار می دهد. بدون پیکربندی، یک فایل پاس شده را می گیرد، آن را در کنار بسته کامپایل شده قرار می دهد و URL آن فایل را برمی گرداند. اما اگر گزینه limit
را مشخص کنیم، فایل های کوچکتر از این حد را به عنوان url داده Base64 کدگذاری می کند و این url را برمی گرداند. این تصویر را در کد جاوا اسکریپت قرار می دهد و یک درخواست HTTP را ذخیره می کند:
// webpack.config.js
module.exports = {
module: {
rules: [
{
test: /\.(jpe?g|png|gif)$/,
loader: 'url-loader',
options: {
// Inline files smaller than 10 kB (10240 bytes)
limit: 10 * 1024,
},
},
],
}
};
// index.js
import imageUrl from './image.png';
// → If image.png is smaller than 10 kB, `imageUrl` will include
// the encoded image: '…'
// → If image.png is larger than 10 kB, the loader will create a new file,
// and `imageUrl` will include its url: `/2fcd56a1920be.png`
svg-url-loader
درست مانند url-loader
کار می کند - با این تفاوت که فایل ها را به جای Base64 با کدگذاری URL کدگذاری می کند. این برای تصاویر SVG مفید است - چون فایل های SVG فقط یک متن ساده هستند، این رمزگذاری در اندازه موثرتر است.
module.exports = {
module: {
rules: [
{
test: /\.svg$/,
loader: "svg-url-loader",
options: {
limit: 10 * 1024,
noquotes: true
}
}
]
}
};
image-webpack-loader
تصاویری را که از آن عبور می کنند فشرده می کند. از تصاویر JPG، PNG، GIF و SVG پشتیبانی می کند، بنابراین ما قصد داریم از آن برای همه این انواع استفاده کنیم.
این لودر تصاویر را در برنامه جاسازی نمی کند، بنابراین باید به صورت جفت با url-loader
و svg-url-loader
کار کند. برای جلوگیری از کپی پیست کردن آن در هر دو قانون (یکی برای تصاویر JPG/PNG/GIF و دیگری برای تصاویر SVG)، این بارگذار را به عنوان یک قانون جداگانه با enforce: 'pre'
:
// webpack.config.js
module.exports = {
module: {
rules: [
{
test: /\.(jpe?g|png|gif|svg)$/,
loader: 'image-webpack-loader',
// This will apply the loader before the other ones
enforce: 'pre'
}
]
}
};
تنظیمات پیشفرض لودر در حال حاضر خوب است – اما اگر میخواهید آن را بیشتر پیکربندی کنید، گزینههای افزونه را ببینید. برای انتخاب گزینه هایی که باید مشخص کنید، راهنمای عالی Addy Osmani در مورد بهینه سازی تصویر را بررسی کنید.
در ادامه مطلب
بهینه سازی وابستگی ها
بیش از نیمی از اندازه متوسط جاوا اسکریپت از وابستگی ها ناشی می شود و بخشی از آن اندازه ممکن است غیر ضروری باشد.
به عنوان مثال، Lodash (در نسخه 4.17.4) 72 کیلوبایت کد کوچک شده را به بسته اضافه می کند. اما اگر فقط از 20 روش آن استفاده کنید، تقریباً 65 کیلوبایت کد کوچک شده هیچ کاری انجام نمی دهد.
مثال دیگر Moment.js است. نسخه 2.19.1 آن 223 کیلوبایت کد کوچک می گیرد که بسیار زیاد است – متوسط اندازه جاوا اسکریپت در یک صفحه در اکتبر 2017 452 کیلوبایت بود . با این حال، 170 کیلوبایت از آن اندازه فایل های محلی سازی است. اگر از Moment.js با چندین زبان استفاده نمیکنید، این فایلها بدون هدف، بسته را پر میکنند.
همه این وابستگی ها را می توان به راحتی بهینه کرد. ما رویکردهای بهینهسازی را در یک مخزن GitHub جمعآوری کردهایم – آن را بررسی کنید !
فعال کردن الحاق ماژول برای ماژولهای ES (با نام مستعار scope hoisting)
هنگامی که در حال ساخت یک بسته نرم افزاری هستید، وب پک هر ماژول را در یک تابع قرار می دهد:
// index.js
import {render} from './comments.js';
render();
// comments.js
export function render(data, target) {
console.log('Rendered!');
}
↓
// bundle.js (part of)
/* 0 */
(function(module, __webpack_exports__, __webpack_require__) {
"use strict";
Object.defineProperty(__webpack_exports__, "__esModule", { value: true });
var __WEBPACK_IMPORTED_MODULE_0__comments_js__ = __webpack_require__(1);
Object(__WEBPACK_IMPORTED_MODULE_0__comments_js__["a" /* render */])();
}),
/* 1 */
(function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_exports__["a"] = render;
function render(data, target) {
console.log('Rendered!');
}
})
در گذشته، این مورد برای جداسازی ماژول های CommonJS/AMD از یکدیگر ضروری بود. با این حال، این یک سربار اندازه و عملکرد برای هر ماژول اضافه کرد.
Webpack 2 پشتیبانی از ماژولهای ES را معرفی کرد که بر خلاف ماژولهای CommonJS و AMD، میتوانند بدون بستهبندی هر کدام با یک عملکرد همراه شوند. و webpack 3 چنین بستهبندی را امکانپذیر کرد - با الحاق ماژول . این چیزی است که الحاق ماژول انجام می دهد:
// index.js
import {render} from './comments.js';
render();
// comments.js
export function render(data, target) {
console.log('Rendered!');
}
↓
// Unlike the previous snippet, this bundle has only one module
// which includes the code from both files
// bundle.js (part of; compiled with ModuleConcatenationPlugin)
/* 0 */
(function(module, __webpack_exports__, __webpack_require__) {
"use strict";
Object.defineProperty(__webpack_exports__, "__esModule", { value: true });
// CONCATENATED MODULE: ./comments.js
function render(data, target) {
console.log('Rendered!');
}
// CONCATENATED MODULE: ./index.js
render();
})
تفاوت را می بینید؟ در بسته ساده، ماژول 0 به render
از ماژول 1 نیاز داشت. با الحاق ماژول، require
به سادگی با عملکرد مورد نیاز جایگزین می شود و ماژول 1 حذف می شود. بسته نرم افزاری ماژول های کمتری دارد - و سربار ماژول کمتری دارد!
برای روشن کردن این رفتار، در بسته وب 4 ، گزینه optimization.concatenateModules
را فعال کنید:
// webpack.config.js (for webpack 4)
module.exports = {
optimization: {
concatenateModules: true
}
};
در بسته وب 3، از ModuleConcatenationPlugin
استفاده کنید:
// webpack.config.js (for webpack 3)
const webpack = require('webpack');
module.exports = {
plugins: [
new webpack.optimize.ModuleConcatenationPlugin()
]
};
در ادامه مطلب
- اسناد Webpack برای ModuleConcatenationPlugin
- "معرفی مختصر بر scope hoisting"
- شرح مفصلی از کاری که این افزونه انجام می دهد
اگر هم کد وب بسته و هم غیر بسته وب دارید از externals
استفاده کنید
ممکن است پروژه بزرگی داشته باشید که در آن برخی از کدها با وب پک کامپایل می شوند و برخی کدها نه. مانند یک سایت میزبانی ویدیو، که ویجت پخش کننده ممکن است با بسته وب ساخته شود، و صفحه اطراف ممکن است نباشد:
اگر هر دو قطعه کد وابستگی های مشترکی دارند، می توانید آنها را به اشتراک بگذارید تا از دانلود چندباره کد آنها جلوگیری کنید. این کار با گزینه externals
بسته وب انجام می شود - ماژول ها را با متغیرها یا سایر واردات خارجی جایگزین می کند.
اگر وابستگی ها در window
موجود باشد
اگر کد غیر بسته وب شما متکی به وابستگی هایی است که به عنوان متغیر در window
موجود است، نام مستعار وابستگی به نام متغیرها:
// webpack.config.js
module.exports = {
externals: {
'react': 'React',
'react-dom': 'ReactDOM'
}
};
با این پیکربندی، webpack بستههای react
و react-dom
را جمع نمیکند. در عوض، آنها با چیزی شبیه به این جایگزین می شوند:
// bundle.js (part of)
(function(module, exports) {
// A module that exports `window.React`. Without `externals`,
// this module would include the whole React bundle
module.exports = React;
}),
(function(module, exports) {
// A module that exports `window.ReactDOM`. Without `externals`,
// this module would include the whole ReactDOM bundle
module.exports = ReactDOM;
})
اگر وابستگی ها به عنوان بسته های AMD بارگذاری شوند
اگر کد غیر بسته وب شما وابستگی ها را در window
نمایش نمی دهد، همه چیز پیچیده تر می شود. با این حال، اگر کد غیر بسته وب این وابستگیها را به عنوان بستههای AMD مصرف میکند، همچنان میتوانید از بارگیری یک کد دو بار اجتناب کنید.
برای انجام این کار، کد وب بسته را به عنوان یک بسته AMD و ماژول های مستعار در URL های کتابخانه کامپایل کنید:
// webpack.config.js
module.exports = {
output: {
libraryTarget: 'amd'
},
externals: {
'react': {
amd: '/libraries/react.min.js'
},
'react-dom': {
amd: '/libraries/react-dom.min.js'
}
}
};
Webpack بسته را در define()
قرار می دهد و آن را به این URL ها وابسته می کند:
// bundle.js (beginning)
define(["/libraries/react.min.js", "/libraries/react-dom.min.js"], function () { … });
اگر کد غیر بسته وب از همان URL ها برای بارگیری وابستگی های خود استفاده کند، این فایل ها فقط یک بار بارگیری می شوند - درخواست های اضافی از حافظه پنهان بارگذار استفاده می کنند.
در ادامه مطلب
- اسناد Webpack در
externals
جمع بندی
- اگر از webpack 4 استفاده می کنید، حالت تولید را فعال کنید
- کد خود را با گزینههای مینیفایر و لودر در سطح بسته به حداقل برسانید
- با جایگزین کردن
NODE_ENV
باproduction
کد فقط توسعه را حذف کنید - از ماژول های ES برای فعال کردن تکان دادن درخت استفاده کنید
- فشرده سازی تصاویر
- بهینه سازی های وابستگی خاص را اعمال کنید
- الحاق ماژول را فعال کنید
- اگر این برای شما منطقی است
externals
استفاده کنید