react 热替换 ([HMR])

热替换好多地方可以用到,目前比较流行的用法是搭配React和webpack实现在不刷新页面的情况下对模块的增删改。在给项目添加热替换功能的时候,可以说是踩了各种坑,webpack官方给的配置也有小问题还不得不翻墙去解决(百度出来的一个能打的也没有)。

官方的方案在这儿:https://webpack.js.org/guides/hmr-react/

我先把自己配置成功的贴出来,再说一下如果完全照搬官方配置,会产生的问题:

(只保留热替换相关配置和最基础配置,且只考虑开发不考虑生产,且假设你已安装必要的包)

  1. module.exports = {
  2. entry: [
  3. 'react-hot-loader/patch', // 1
  4. // 开启 React 代码的模块热替换(HMR)
  5. './src',
  6. ],
  7. output: {
  8. filename: 'js/bundle.js',
  9. path: path.resolve(__dirname, 'public'),
  10. publicPath:'http://localhost:7000/', // 2
  11. },
  12. module: {
  13. rules: [
  14. { test: /\.(js|jsx)$/, use: 'babel-loader', exclude: /node_modules/ },
  15. { test: /\.css$/,
  16. use:['style-loader', 'css-loader', 'postcss-loader'], // 3
  17. exclude: /node_modules/ },
  18. ],
  19. },
  20. resolve: {
  21. extensions: ['.js', '.jsx'],
  22. },
  23. devServer: {
  24. port: 7000,
  25. hot: true, // 4
  26. // 开启服务器的模块热替换(HMR)
  27. headers: {
  28. 'Access-Control-Allow-Origin': '*', // 5
  29. },
  30. },
  31. plugins: [
  32. new webpack.HotModuleReplacementPlugin(), // 6
  33. // 开启全局的模块热替换(HMR)
  34. new webpack.NamedModulesPlugin(), // +
  35. // 当模块热替换(HMR)时在浏览器控制台输出对用户更友好的模块名字信息
  36. ],
  37. }

react 和babel 部分的配置与官方相同即可。

参考官方配置你遇到第一个问题:

  1. [HMR] Update failed: SyntaxError: Unexpected token < in JSON at position 0
  2. at Object.parse (<anonymous>)
  3. at XMLHttpRequest.request.onreadystatechange

这个问题很头疼,因为你去搜索完全找不到有用的东西,而且你看不出来错在哪了,当时的心情真是。。。

最后在 react-hot-loader 的 github 库的 Issues 里翻到了有人和我一样的问题,大快人心。

原来是 output 的 publicPath 出了问题用 '/' 是不行滴,要把启动webpack服务的地址填上:'http://localhost:7000/'。其实也不能怪官方文档,人家是假设你应用程序和包都托管给 webpack 服务,但是我的只把包托给了 webpack 服务,应用程序是另外启动的 node 服务。好吧这是个比较小众的问题,你要是正好有这个问题,还搜到了这篇文章,那是好福气了。

另外一个

  1. XMLHttpRequest cannot load http://localhost:7000/4221731a75de7a449377.hot-update.json. No 'Access-Control-Allow-Origin' header is present on the requested resource. Origin 'http://localhost:8000' is therefore not allowed access. The response had HTTP status code 404

这个问题只在 webpack 服务发生变动时候有,并不影响HMR使用,所以不管它也可以。

devServer里添加

  1. headers: {
  2. 'Access-Control-Allow-Origin': '*', // 5
  3. },

即可,也是因为应用程序没有托管给 webpack 导致。

如果你的应用程序和包都是托管给 webpack 服务,那就没有这两个问题了。

(话说应用程序一般都是另启服务的吧。。。)

再说一个问题,除了上面两个,是不是这么多配置都是必须的。不配置会导致什么错误。

首先,

  1. entry里的
  2. 'react-hot-loader/patch', // 1
  3. devServer里的
  4. hot: true, // 4
  5. plugins里的
  6. new webpack.HotModuleReplacementPlugin(), // 6

从字面上看就是必须的,不用想,否则不会热。

new webpack.NamedModulesPlugin(), // +只是让控制台输出的更友好,推荐但非必需

这个比较容易忽略:

  1. { test: /\.css$/,
  2. use:['style-loader', 'css-loader', 'postcss-loader'], // 3
  3. exclude: /node_modules/ },

style-loader 是必须的,否则样式的刷新就不热了。

很多人用 ExtractTextWebpackPlugin 把 css 文件单独弄出来就不用 style-loader了,不过推荐部署应用之前弄一份就行,开发用不着 ExtractTextWebpackPlugin 。

不用 style-loader 修改样式表就不会无刷新更新页面了。不过这一点容易想到。

还有一点就是 babel 配置文件里禁用 modules 很重要,否则导致各种问题,根本别想愉快的开发。这个对应用程序本身使用 import 没有影响,毕竟 webpack2 啥标准都支持,其他地方就用 require 吧。

react 热替换 ([HMR])的更多相关文章

  1. 模块热替换 HMR

    devserver:{hot:true},既及时更新代码,样式(需配合loader)变化,自动重编译,只适用于开发环境. 入口文件中,添加监视: + if (module.hot) {+ module ...

  2. webpack学习之—— 模块热替换(Hot Module Replacement)

    模块热替换(HMR - Hot Module Replacement)功能会在应用程序运行过程中替换.添加或删除模块,而无需重新加载整个页面.主要是通过以下几种方式,来显著加快开发速度: 保留在完全重 ...

  3. 【webpack】-- 模块热替换

    全称是Hot Module ReplaceMent(HMR),理解成热模块替换或者模块热替换都可以吧,和.net中的热插拔一个意思,就是在运行中对程序的模块进行更新.这个功能主要是用于开发过程中,对生 ...

  4. 【webpack学习笔记】a05-模块热替换

    什么是模块热替换? 这个功能会在程序运行过程中替换.添加或删除模块,而无需重新加载整个页面 有什么用呢? 保留在完全重新加载页面时丢失的应用程序状态. 只更新变更内容,以节省宝贵的开发时间. 调整样式 ...

  5. 启用 webpack 的模块热替换特性

    启用 webpack 的模块热替换特性: module.exports = { //... devServer: { hot: true } } 注意,必须有 webpack.HotModuleRep ...

  6. webpack学习_模块热替换(Hot Module Peaplacement)

    模块热替换(Hot Module Replacement 或 HMR) 是webpack提供的最有用的功能之一.允许在u女性是更新各种模块,而无需进行完全刷新. 启用HMR 承接之前的代码 webpa ...

  7. webpack 模块热替换的理解和使用

    模块热替换(webpack文档上也叫 Hot Module Replacement 或 HMR)是 webpack 提供的最有用的功能之一.它允许在运行时更新各种模块,而无需进行完全刷新. 这句话其实 ...

  8. Java_类的热替换

    转自:http://www.ibm.com/developerworks/cn/java/j-lo-hotswapcls/#ibm-pcon Java ClassLoader 技术剖析 在本文中,我们 ...

  9. 使用自己的ClassLoader实现热替换

    首先实现一个自己的ClassLoader,该ClassLoader重写findClass方法. 从classpath中加载类资源. 注意,不要重写loadClass方法.  因为在使用自定义的MyCl ...

随机推荐

  1. 关于php 高并发解决的一点思路

    涉及抢购.秒杀.抽奖.抢票等活动时,为了避免超卖,那么库存数量是有限的,但是如果同时下单人数超过了库存数量,就会导致商品超卖问题.那么我们怎么来解决这个问题呢,我的思路如下(伪代码): sql1:查询 ...

  2. LINUX ON AZURE 安全建议(全)

    本文为个人原创,可以自由转载,转载请注明出处,多谢! 本文地址:http://www.cnblogs.com/taosha/p/6399554.html 1.网络与安全规划 Azure 虚拟网络 (V ...

  3. jQuery / zepto ajax 全局默认设置

    jQuery / zepto 的 $.ajax 方法需要配置很多选项, 有些是很常用的每个 ajax 请求都要用到的, 可以全局设置, 避免每次都写. 注意: 此处用的 jQuery 版本是 1.8. ...

  4. tr069开源协议EasyCwmp移植

    1.平台MT7628 2.交叉编译器及版本信息mipsel-linux + buildroot-gcc463_32bits.tar.bz2 3.创建工作目录lancer@ubuntu:~$ mkdir ...

  5. 详解Executor框架

    在Java中,使用线程来异步执行任务.Java线程的创建与销毁需要一定的开销,如果我们为每一个任务创建一个新线程来执行,这些线程的创建与销毁将消耗大量的计算资源.同时,为每一个任务创建一个新线程来执行 ...

  6. T_SQL编程赋值、分支语句、循环

    咱们在C#中会常用到赋值.循环.分支语句什么的 今天咱们来看下当初在C#用到的一点东西放到SQL中是怎么使用的 创建变量 在C#中创建一个值类型变量很简单 int a:这就可以了 SQL: decla ...

  7. jQuery修炼心得-DOM节点的删除

    要移除页面上节点是开发者常见的操作,jQuery提供了几种不同的方法用来处理这个问题. 1.empty empty 顾名思义,清空方法,但是与删除又有点不一样,因为它只移除了 指定元素中的所有子节点. ...

  8. 原创-angularjs2不同组件间的通信

    AngualrJs2官方方法是以@Input,@Output来实现组件间的相互传值,而且组件之间必须父子关系,下面给大家提供一个简单的方法,实现组件间的传值,不仅仅是父子组件,跨模块的组件也可以实现传 ...

  9. 0基础搭建Hadoop大数据处理-环境

    由于Hadoop需要运行在Linux环境中,而且是分布式的,因此个人学习只能装虚拟机,本文都以VMware Workstation为准,安装CentOS7,具体的安装此处不作过多介绍,只作需要用到的知 ...

  10. string___assign

    #include <iostream> #include <iterator> #include <string> int main() { std::string ...