本篇文件主要讲结合 Webpack 和 Express 实现先后端热更新开发,若是你还不太了解webpack推荐阅读javascript
webpack 官网文档html
Webpack dev server 是一个轻量的node.js express服务器,实现了 webpack 编译代码实时输出更新。在先后端分离的前端项目开发中常常用到。不过这篇文章应该不会讲到它。前端
Webpack dev middleware 是 WebPack 的一个中间件。它用于在 Express 中分发须要经过 WebPack 编译的文件。单独使用它就能够完成代码的热重载(hot reloading)功能。java
特性:node
Webpack hot middleware 它经过订阅 Webpack 的编译更新,以后经过执行 webpack 的 HMR api 将这些代码模块的更新推送给浏览器端。webpack
HMR 即 Hot Module Replacement 是 Webpack 一个重要的功能。它可使咱们不用经过手动地刷新浏览器页面实现将咱们的更新代码实时应用到当前页面中。git
HMR 的实现原理是在咱们的开发中的应用代码中加入了 HMR Runtime,它是 HMR 的客户端(浏览器端 client)用于和开发服务器通讯,接收更新的模块。服务端工做就是前面提到的 Webpack hot middleware 的,它会在代码更新编译完成以后经过以 json 格式输出给HMR Runtime 就会更具 json 中描述来动态更新相应的代码。github
先来在webpack配置文件中引入web
var webpack = require('webpack');
var HotMiddleWareConfig = 'webpack-hot-middleware/client?path=/__webpack_hmr&timeout=20000'
module.exports = {
context: __dirname,
entry: [
// 添加一个和HotMiddleWare通讯的客户端
HotMiddleWareConfig,
// 添加web应用入口文件
'./client.js'
],
output: {
path: __dirname,
publicPath: '/',
filename: 'bundle.js'
},
devtool: '#source-map',
plugins: [
new webpack.optimize.OccurenceOrderPlugin(),
// 在 webpack 插件中引入 webpack.HotModuleReplacementPlugin
new webpack.HotModuleReplacementPlugin(),
new webpack.NoErrorsPlugin()
],
};
复制代码
webpack-hot-middleware example webpack.config.jsexpress
在咱们的开发环境中是这样配置的。getEntries 是自动根据咱们规则获取到入口文件并加上 webpack hot middle 配置。
var webpack = require('webpack');
var path = require('path')
var merge = require('webpack-merge')
var baseConfig = require('./webpack.base')
var getEntries = require('./getEntries')
var publicPath = 'http://0.0.0.0:7799/dist/';
var hotMiddlewareScript = 'webpack-hot-middleware/client?reload=true';
var assetsInsert = require('./assetsInsert')
module.exports = merge(baseConfig, {
entry: getEntries(hotMiddlewareScript),
devtool: '#eval-source-map',
output: {
filename: './[name].[hash].js',
path: path.resolve('./public/dist'),
publicPath: publicPath
},
plugins: [
new webpack.DefinePlugin({
'process.env': {
NODE_ENV: '"development"'
}
}),
new webpack.optimize.OccurenceOrderPlugin(),
new webpack.HotModuleReplacementPlugin(),
new webpack.NoErrorsPlugin(),
new assetsInsert()
]
})
复制代码
在 Express 的配置主要就4个步骤:
var http = require('http');
var express = require('express');
var app = express();
app.use(require('morgan')('short'));
// ************************************
// This is the real meat of the example
// ************************************
(function() {
// Step 1: 引入 webpack 的配置文件和 生成 webpack 的编译器
var webpack = require('webpack');
var webpackConfig = require(process.env.WEBPACK_CONFIG ? process.env.WEBPACK_CONFIG : './webpack.config');
var compiler = webpack(webpackConfig);
// Step 2: 将编译器挂载给 webpack dev middleware
app.use(require("webpack-dev-middleware")(compiler, {
noInfo: true, publicPath: webpackConfig.output.publicPath
}));
// Step 3: 将编译器挂载给 webpack hot middleware
app.use(require("webpack-hot-middleware")(compiler, {
log: console.log, path: '/__webpack_hmr', heartbeat: 10 * 1000
}));
})();
// 定义 express 配置
app.get("/", function(req, res) {
res.sendFile(__dirname + '/index.html');
});
app.get("/multientry", function(req, res) {
res.sendFile(__dirname + '/index-multientry.html');
});
if (require.main === module) {
var server = http.createServer(app);
server.listen(process.env.PORT || 1616, function() {
console.log("Listening on %j", server.address());
});
}
复制代码
webpack-hot-middleware example server.js
要注意的是必定要在定义 express router 前定义 webpack 相关的中间件。还有一点是这里server.js 只是开发环境中使用,在生成环境中咱们就不须要再用到它们了。 因此在咱们实际的使用中须要经过定义环境变量来区分开发和生产环境
var NODE_ENV = process.env.NODE_ENV || 'production';
var isDev = NODE_ENV === 'development';
if (isDev) {
var webpack = require('webpack'),
webpackDevMiddleware = require('webpack-dev-middleware'),
webpackHotMiddleware = require('webpack-hot-middleware'),
webpackDevConfig = require('./build/webpack.config.js');
var compiler = webpack(webpackDevConfig);
app.use(webpackDevMiddleware(compiler, {
publicPath: webpackDevConfig.output.publicPath,
noInfo: true,
stats: {
colors: true
}
}));
app.use(webpackHotMiddleware(compiler));
routerConfig(app, {
dirPath: __dirname + '/server/routes/',
map: {
'index': '/',
'api': '/api/*',
'proxy': '/proxy/*'
}
});
var reload = require('reload');
var http = require('http');
var server = http.createServer(app);
reload(server, app);
app.use(express.static(path.join(__dirname, 'public')));
server.listen(port, function(){
console.log('App (dev) is now running on port ' + port + '!');
});
} else {
routerConfig(app, {
dirPath: __dirname + '/server/routes/',
map: {
'index': '/',
'api': '/api/*',
'proxy': '/proxy/*'
}
});
app.use(express.static(path.join(__dirname, 'public')));
app.listen(port, function () {
console.log('App (Production) is now running on port ' + port + '!');
});
}
复制代码
以上在前端咱们实现了前端文件的热更新,可是咱们在修改服务端文件的时候,并不会使Node自动重启,因此咱们使用 supervisor 来做为监听文件修改事件来自动重启 Node服务。
supervisor 须要 全局安装
npm install supervisor -g
复制代码
安装完成以后咱们就能够在命令行中使用
咱们在 package.json 的 scripts 中写好经常使用的命令,以后只用 npm run xxx 便可使用
"scripts": {
"dev": "export NODE_ENV=development && supervisor -w server,app.js app",
"build": "node build/build.js",
"start": "node app"
},
复制代码
supervisor [options] <program>
supervisor -w server,app.js app
复制代码
-w 就是一个 options 配置项,它用于监听指定目录或者文件的变动,可使用,
分隔,监听多个目录或者文件,这就是监听了 server 目录和根目录的 app.js 到变动以后就会重启咱们的 Express 入口文件 app。