webpack 快速构建 React 学习环境(2)-- 热更新

简介: 上一篇文章 《webpack 快速构建 React 学习环境(1)》中介绍了构建一个最简单开发环境,这里接着完善这个开发环境,让它用起来更加的趁手。

上一篇文章 《webpack 快速构建 React 学习环境(1)》中介绍了构建一个最简单开发环境,这里接着完善这个开发环境,让它用起来更加的趁手。

看着篇文章前请先看 《webpack 快速构建 React 学习环境(1)》

本小结内容对应构建的项目源码:github.com/wewin11235/…,仓库的 stage2 分支

文章同步发布在个人博客站点

上一篇中构建的开发环境存在的问题

上一节搭建的开发环境不能热加载,每次文件改动后都需要重新编译,手动刷新页面。虽然使用 webpack --watch 命令在文件变化后能重新编译,但是仍需要手动刷新页面。webpack --watch 的方式还有个缺点,每次都是重新编译生成新的文件到 build 目录下, 文件多的时候这个编译过程就会慢。

webpack-dev-server 配合 HRM 可以构建一个完美的开发环境,改动保存后自动编译,无需手动刷新页面。

使用 webpack-dev-server

webpack-dev-server 主要是启动了一个使用 express 的 Http 服务器。它的作用主要是用来伺服资源文件。此外这个Http服务器和client使用了websocket通讯协议,原始文件作出改动后,webpack-dev-server 会实时的编译. webpack-dev-server 的时时编译并不会输出到 webpack.config.js 中指定的出口,编译生成的文件在内存中,也有效的提高了编译效率。

安装与配置:

npm in webpack-dev-server -D
复制代码

安装完成后: ./node_modules/.bin/webpack-dev-server 便可启动 server


ℹ 「wds」: Project is running at http://localhost:8081/
ℹ 「wds」: webpack output is served from /./
 「wdm」: Hash: b2af4145bdae26f19266
复制代码

dev server 被启动在了 8081 端口, server 会默认找项目主目录下的 index.html 文件作为服务的根页面。 由于我们编译后的服务入口文件 index.html 放在 build 目录下,所以要对 webpack-dev-server 做一下配置:

webpack.config.js 增加如下配置:

devServer: {
    contentBase: path.join(__dirname, 'build')
}
复制代码

此时 webpack.config.js :

const path = require('path');
var HtmlWebpackPlugin = require('html-webpack-plugin');

module.exports = {
  entry: './src/index.js',
  output: {
    path: path.resolve(__dirname, 'build'),
    filename: '[name].[hash:8].js',
    publicPath: '/',
  },
  module: {
    rules: [
      {
        test: /\.js$/,
        exclude: /(node_modules)/,
        use: {
          loader: 'babel-loader'
        },
      },
    ],
  },
  plugins: [new HtmlWebpackPlugin({
    filename: 'index.html',
    template: 'src/index.html'
  })],
  devServer: {
    contentBase: path.join(__dirname, 'build'),
  }
};
复制代码

这样就可以访问 http://localhost:8081 查看启动的服务。

此时 dev server 虽然启动,但尝试修改 index.js 内容可以发现页面并不能自动刷新,webpack dever server 的自动刷新有两种模式 Iframe 和 inline 两种模式,这里用 inline 模式,修改配置如下:

const path = require('path');
var HtmlWebpackPlugin = require('html-webpack-plugin');

module.exports = {
  entry: './src/index.js',
  output: {
    path: path.resolve(__dirname, 'build'),
    filename: '[name].[hash:8].js',
    publicPath: '/',
  },
  module: {
    rules: [
      {
        test: /\.js$/,
        exclude: /(node_modules)/,
        use: {
          loader: 'babel-loader'
        },
      },
    ],
  },
  plugins: [
    new HtmlWebpackPlugin({
      filename: 'index.html',
      template: 'src/index.html'
    })
  ],
  devServer: {
    contentBase: path.join(__dirname, 'build'),
    port: 9000,   // 指定服务启动在 9000 端口
     inline: true,  // inline 模式启动
     open: true  // 执行webpack-dev-server 后自动打开浏览器
  }
};
复制代码

上面的配置,当文件有更新,你会发现浏览器刷新了,配置 HMR 可以实现局部热替换,不用整个浏览器刷新。HMR 的配置请参考 [hot-module-replacement] (https://webpack.docschina.org/guides/hot-module-replacement)。 react 框架下还需要用到 react-hot-loader

开启 HotModuleReplace

改动有以下几处: 添加 src/print.js 文件

export default function printMe() {
  console.log("Updating print.js...");
}
复制代码

修改 webpack.config.js 文件,添加 hot 配置:

const path = require('path');
const webpack = require("webpack");
var HtmlWebpackPlugin = require('html-webpack-plugin');

module.exports = {
  entry: './src/index.js',
  output: {
    path: path.resolve(__dirname, 'build'),
    filename: '[name].[hash:8].js',
    publicPath: '/',
  },
  mode: 'development',
  module: {
    rules: [
      {
        test: /\.js$/,
        exclude: /(node_modules)/,
        use: {
          loader: 'babel-loader'
        },
      },
    ],
  },
  plugins: [
    new HtmlWebpackPlugin({
      filename: 'index.html',
      template: 'src/index.html'
    }),
    new webpack.NamedModulesPlugin(),
    new webpack.HotModuleReplacementPlugin()
  ],
  devServer: {
    contentBase: path.join(__dirname, 'build'),
    port: 9000,
    open: true,  // HMR 支持
    hot: true
  }
};
复制代码

src/index.js 文件如下:

import React from 'react';
import ReactDOM from 'react-dom';
import { hot } from 'react-hot-loader'

class HelloReact extends React.Component{
  constructor(props) {
    super(props);
  }

  render(){
    return( <div>Hello  React</div>);
  }
}

export default hot(module)(HelloReact);

ReactDOM.render(<HelloReact />, document.getElementById('root'));

if (module.hot) {
  module.hot.accept('./print.js', function(){
      console.log("Accepting the updated printMe module!");
      printMe();
  })
}
复制代码

React 框架下热更新支持(react-hot-loader)

安装 react-hot-loader

npm i react-hot-loader -D
复制代码

修改 .babelrc 如下

{
  "presets": [
    ["@babel/preset-env", {
      "targets": {
        "node": "current"
      }
    }],
    "@babel/preset-react"
  ],
  "plugins": ["react-hot-loader/babel"]  //添加 ‘react-hot-loader/babel’ 这个插件支持
}
复制代码

在使用的时候改变下 React 组件的 export 的方式,如我们的 src/index.js 文件修改为如下:

import React from 'react';
import ReactDOM from 'react-dom';
import { hot } from 'react-hot-loader';  // 引入 hot 模块

class HelloReact extends React.Component{
  constructor(props) {
    super(props);
  }

  render(){
    return( <div>Hello  React</div>);
  }
}

export default hot(module)(HelloReact);  //修改 export 时候的方式

ReactDOM.render(<HelloReact />, document.getElementById('root'));

if (module.hot) {
  module.hot.accept('./print.js', function(){
      console.log("Accepting the updated printMe module!");
      printMe();
  })
}
复制代码

此时运行 ./node_modules/.bin/webpack-dev-server --mode=development ,再尝试修改 index.js 内容你会发现页面更新了,但是浏览器并没有刷新。

这样我们就构建了一个自动编译,自动更新的 React 的开发环境。

重构示例 react-demo

这个重构和开发环境搭建无关,只是在引入了 react-hot-loader 后,会发现现在的 src/index.js 此时显得很不优雅,这给因为 ReactHello 作为一个组件本就应该抽取为一个独立的文件:

新建文件:src/ReactHello.js:

import React from 'react';
import { hot } from 'react-hot-loader'

class HelloReact extends React.Component{
  constructor(props) {
    super(props);
  }

  render(){
    return( <div>Hello React</div>);
  }
}

export default hot(module)(HelloReact);
复制代码

调整 src/index.js:

import React from 'react';
import ReactDOM from 'react-dom';
import HelloReact from './HelloReact.js';

ReactDOM.render(<HelloReact />, document.getElementById('root'));

if (module.hot) {
  module.hot.accept('./print.js', function(){
      console.log("Accepting the updated printMe module!");
      printMe();
  })
}
复制代码

这样看着就舒服多了。

文章配合使用的 Demo 地址:github.com/wewin11235/… 仓库的 stage2 分支



原文发布时间为:2018年06月29日

作者:蚂蚁哈哈哈

本文来源: 掘金  如需转载请联系原作者
相关文章
|
2月前
|
存储 监控 安全
如何确保 React Native 热更新的安全性?
确保React Native热更新的安全性至关重要
|
3月前
|
前端开发 JavaScript
React学习之——条件渲染
【10月更文挑战第16天】React 中没有像Vue中v-if这种指令。React 中的条件渲染和 JavaScript 中的一样,使用 JavaScript 运算符 if 或者条件运算符去创建元素来表现当前的状态,然后让 React 根据它们来更新 UI。
|
4月前
|
前端开发 JavaScript
学习react基础(3)_setState、state、jsx、使用ref的几种形式
本文探讨了React中this.setState和this.state的区别,以及React的核心概念,包括核心库的使用、JSX语法、类与函数组件的区别、事件处理和ref的使用。
92 3
学习react基础(3)_setState、state、jsx、使用ref的几种形式
|
4月前
|
前端开发
学习react基础(2)_props、state和style的使用
本文介绍了React中组件间数据传递的方式,包括props和state的使用,以及如何在React组件中使用style样式。
46 0
|
2月前
|
前端开发 JavaScript 安全
如何在 React Native 中实现热更新?
如何在 React Native 中实现热更新?
171 8
|
3月前
|
JavaScript
webpack学习三:webpack初始化整合配置vue,一步一步的抽离代码块整合vue。
这篇文章是关于如何在webpack环境中配置Vue.js,包括安装Vue.js、解决报错、理解el与template的区别、使用SPA模式、抽离模板为对象、封装为单独的js文件、安装vue-loader时遇到的问题及解决方案,以及整个过程的总结。
106 2
webpack学习三:webpack初始化整合配置vue,一步一步的抽离代码块整合vue。
|
3月前
|
JavaScript
webpack学习五:webpack的配置文件webpack.config.js分离,分离成开发环境配置文件和生产环境配置文件
这篇文章介绍了如何将webpack的配置文件分离成开发环境和生产环境的配置文件,以提高打包效率。
60 1
webpack学习五:webpack的配置文件webpack.config.js分离,分离成开发环境配置文件和生产环境配置文件
|
3月前
|
JavaScript 前端开发 Java
webpack学习一:什么是模块化开发,什么是webpack,以及二者之间的关系。
这篇文章介绍了模块化开发的概念、历史和实现方式,以及webpack作为一个现代JavaScript应用的静态模块打包工具,它如何帮助我们将ES6等高级语法打包成浏览器可以识别的低级语法,并解释了npm在webpack安装和使用中的作用。
48 1
webpack学习一:什么是模块化开发,什么是webpack,以及二者之间的关系。
|
2月前
|
前端开发 JavaScript 安全
学习如何为 React 组件编写测试:
学习如何为 React 组件编写测试:
45 2
|
2月前
|
缓存 监控
webpack 提高构建速度的方式
【10月更文挑战第23天】需要根据项目的具体情况和需求,综合运用这些方法,不断进行优化和改进,以达到最佳的构建速度和效果。同时,随着项目的发展和变化,还需要持续关注和调整构建速度的相关措施,以适应不断变化的需求。