react 热替换 ([HMR])
react 热替换 ([HMR])
热替换好多地方可以用到,目前比较流行的用法是搭配React和webpack实现在不刷新页面的情况下对模块的增删改。在给项目添加热替换功能的时候,可以说是踩了各种坑,webpack官方给的配置也有小问题还不得不翻墙去解决(百度出来的一个能打的也没有)。
官方的方案在这儿:https://webpack.js.org/guides/hmr-react/
我先把自己配置成功的贴出来,再说一下如果完全照搬官方配置,会产生的问题:
(只保留热替换相关配置和最基础配置,且只考虑开发不考虑生产,且假设你已安装必要的包)
module.exports = {
entry: [
'react-hot-loader/patch', // 1
// 开启 React 代码的模块热替换(HMR)
'./src',
],
output: {
filename: 'js/bundle.js',
path: path.resolve(__dirname, 'public'),
publicPath:'http://localhost:7000/', // 2
},
module: {
rules: [
{ test: /\.(js|jsx)$/, use: 'babel-loader', exclude: /node_modules/ },
{ test: /\.css$/,
use:['style-loader', 'css-loader', 'postcss-loader'], // 3
exclude: /node_modules/ },
],
},
resolve: {
extensions: ['.js', '.jsx'],
},
devServer: {
port: 7000,
hot: true, // 4
// 开启服务器的模块热替换(HMR)
headers: {
'Access-Control-Allow-Origin': '*', // 5
},
},
plugins: [
new webpack.HotModuleReplacementPlugin(), // 6
// 开启全局的模块热替换(HMR)
new webpack.NamedModulesPlugin(), // +
// 当模块热替换(HMR)时在浏览器控制台输出对用户更友好的模块名字信息
],
}
react 和babel 部分的配置与官方相同即可。
参考官方配置你遇到第一个问题:
[HMR] Update failed: SyntaxError: Unexpected token < in JSON at position 0
at Object.parse (<anonymous>)
at XMLHttpRequest.request.onreadystatechange
这个问题很头疼,因为你去搜索完全找不到有用的东西,而且你看不出来错在哪了,当时的心情真是。。。
最后在 react-hot-loader 的 github 库的 Issues 里翻到了有人和我一样的问题,大快人心。
原来是 output 的 publicPath 出了问题用 '/' 是不行滴,要把启动webpack服务的地址填上:'http://localhost:7000/'。其实也不能怪官方文档,人家是假设你应用程序和包都托管给 webpack 服务,但是我的只把包托给了 webpack 服务,应用程序是另外启动的 node 服务。好吧这是个比较小众的问题,你要是正好有这个问题,还搜到了这篇文章,那是好福气了。
另外一个
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里添加
headers: {
'Access-Control-Allow-Origin': '*', // 5
},
即可,也是因为应用程序没有托管给 webpack 导致。
如果你的应用程序和包都是托管给 webpack 服务,那就没有这两个问题了。
(话说应用程序一般都是另启服务的吧。。。)
再说一个问题,除了上面两个,是不是这么多配置都是必须的。不配置会导致什么错误。
首先,
entry里的
'react-hot-loader/patch', // 1
devServer里的
hot: true, // 4
plugins里的
new webpack.HotModuleReplacementPlugin(), // 6
从字面上看就是必须的,不用想,否则不会热。
new webpack.NamedModulesPlugin(), // +只是让控制台输出的更友好,推荐但非必需
这个比较容易忽略:
{ test: /\.css$/,
use:['style-loader', 'css-loader', 'postcss-loader'], // 3
exclude: /node_modules/ },
style-loader 是必须的,否则样式的刷新就不热了。
很多人用 ExtractTextWebpackPlugin 把 css 文件单独弄出来就不用 style-loader了,不过推荐部署应用之前弄一份就行,开发用不着 ExtractTextWebpackPlugin 。
不用 style-loader 修改样式表就不会无刷新更新页面了。不过这一点容易想到。
还有一点就是 babel 配置文件里禁用 modules 很重要,否则导致各种问题,根本别想愉快的开发。这个对应用程序本身使用 import 没有影响,毕竟 webpack2 啥标准都支持,其他地方就用 require 吧。
react 热替换 ([HMR])的更多相关文章
- 模块热替换 HMR
devserver:{hot:true},既及时更新代码,样式(需配合loader)变化,自动重编译,只适用于开发环境. 入口文件中,添加监视: + if (module.hot) {+ module ...
- webpack学习之—— 模块热替换(Hot Module Replacement)
模块热替换(HMR - Hot Module Replacement)功能会在应用程序运行过程中替换.添加或删除模块,而无需重新加载整个页面.主要是通过以下几种方式,来显著加快开发速度: 保留在完全重 ...
- 【webpack】-- 模块热替换
全称是Hot Module ReplaceMent(HMR),理解成热模块替换或者模块热替换都可以吧,和.net中的热插拔一个意思,就是在运行中对程序的模块进行更新.这个功能主要是用于开发过程中,对生 ...
- 【webpack学习笔记】a05-模块热替换
什么是模块热替换? 这个功能会在程序运行过程中替换.添加或删除模块,而无需重新加载整个页面 有什么用呢? 保留在完全重新加载页面时丢失的应用程序状态. 只更新变更内容,以节省宝贵的开发时间. 调整样式 ...
- 启用 webpack 的模块热替换特性
启用 webpack 的模块热替换特性: module.exports = { //... devServer: { hot: true } } 注意,必须有 webpack.HotModuleRep ...
- webpack学习_模块热替换(Hot Module Peaplacement)
模块热替换(Hot Module Replacement 或 HMR) 是webpack提供的最有用的功能之一.允许在u女性是更新各种模块,而无需进行完全刷新. 启用HMR 承接之前的代码 webpa ...
- webpack 模块热替换的理解和使用
模块热替换(webpack文档上也叫 Hot Module Replacement 或 HMR)是 webpack 提供的最有用的功能之一.它允许在运行时更新各种模块,而无需进行完全刷新. 这句话其实 ...
- Java_类的热替换
转自:http://www.ibm.com/developerworks/cn/java/j-lo-hotswapcls/#ibm-pcon Java ClassLoader 技术剖析 在本文中,我们 ...
- 使用自己的ClassLoader实现热替换
首先实现一个自己的ClassLoader,该ClassLoader重写findClass方法. 从classpath中加载类资源. 注意,不要重写loadClass方法. 因为在使用自定义的MyCl ...
随机推荐
- Java设计模式:工厂模式
问题提出 Java的工厂模式与现实生活中的工厂的模型是很相似的.工厂是用来做什么?当然是用来生成产品.因此在Java的工厂模式的关键点就是如何描述好产品和工厂这2个角色之间的关系. 下面来仔细描述一下 ...
- [Android]Gradle 插件 DiscardFilePlugin(class注入&清空类和方法)
以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/6732128.html Android Gradle 插件 Di ...
- java复习(2)---java基础杂记
java命名规范: 参考:http://www.cnblogs.com/maowang1991/archive/2013/06/29/3162366.html 1.项目名小写 2.包名小写 3.类名每 ...
- Eclipse实现图形化界面插件-vs4e
vs4e插件下载地址:http://visualswing4eclipse.googlecode.com/files/vs4e_0.9.12.I20090527-2200.zip 下载完成后,解压,然 ...
- Jmeter如何将上一个请求的结果作为下一个请求的参数——使用正则表达式提取器
首先在线程组下添加两个HTTP请求, 添加好两个HTTP请求后,在每个HTTP请求下添加一个查看结果数 在第一个HTTP请求下添加正则表达式提取器 在第一个HTTP请求添加好IP地址,路径,端口号,协 ...
- Linux-进程描述(5)之进程环境
main函数和启动例程 当内核使用一个exec函数执行C程序时,在调用main函数之前先调用一个特殊的启动例程,可执行程序将此例程指定为程序的起始地址.启动例程从内核获取命令行参数和环境变量,然后为调 ...
- [进程管理]Linux进程状态解析之R、S、D
Linux是一个分时操作系统,能够在一个cpu上运行多个程序,每个被运行的程序实例对应一个或多个进程,这里介绍一下Linux进程状态. Linux是一个多用户,多任务的系统,可以同时运行多个用户的多个 ...
- java.net.SocketException: Broken pipe 异常可能的原因
org.apache.catalina.connector.ClientAbortException: java.net.SocketException: Broken pipe at org.apa ...
- php中表单提交复选框与下拉列表项
在赶项目中,抽出半个小时来写篇博客吧,这个功能说实话不难,为什么要写呢,因为在复选框那里有小小的难点,我试了好多遍才试成功的,希望能为以后需要帮助的同学提供点思路. 先看一下我做的效果吧 就是给每个业 ...
- 前端模块化——seaJS
1.seaJS手记 一:Bower获取 要安装bower Npm install -g bower Bower install seajs 二:Use方法是整个项目的入口方法,通常一个项目中只调用一次 ...