关于webpack的react项目优化

这几天有空给自己写的一个react小项目的webpack配置进行优化,在这里进行一下总结。
项目地址:https://github.com/beiweiqiang/myh5

我把这个项目的开发环境和生产环境各写了一份webpack.config,用一个总的 webpack.config.js 进行switch:

// webpack.config.js
const webpackProduction = require('./webpack.prod.js');
const webpackDevelopment = require('./webpack.dev.js');


const isProduction = process.env.NODE_ENV === 'production';

process.noDeprecation = true;

module.exports = isProduction ? webpackProduction : webpackDevelopment;

对开发环境进行判断,switch合适的webpack配置。

我们先说开发环境,因为是react项目,开发环境用到了hmr,要用hmr,在项目入口处的代码还要进行一定的配置,下面是官方例子:

import React from 'react';
import ReactDOM from 'react-dom';

import { AppContainer } from 'react-hot-loader';
// AppContainer 是一个 HMR 必须的包裹(wrapper)组件

import App from './components/App';

const render = (Component) => {
  ReactDOM.render(
    <AppContainer>
      <Component/>
    </AppContainer>,
    document.getElementById('root')
  );
};

render(App);

// 模块热替换的 API
if (module.hot) {
  module.hot.accept('./components/App', () => {
    render(App)
  });
}

这里我在开发过程中有一个bug,就是console上虽然说了 app up to date,但是chrome浏览器自身不会进行自动刷新,经过google以后得到解决方案。
把最后部分改成下面这样就行:

// 模块热替换的 API
if (module.hot) {
  module.hot.accept();
}

说回开发环境下的webpack配置,因为代码有点长,所以就不贴在这里,可以参考我的项目webpack.dev.js 文件。
说几个地方:

1

output: {
    path: path.resolve(__dirname, 'client', 'dist'),
    // filename: '[name].[chunkhash].js',
    filename: '[name].js',
    publicPath: '/',
  }

这里并没有像生产环境下的webpack配置 (下面会介绍) 一样给文件加上hash值,因为计算这个hash会耗费时间,开发环境讲究构建速度要快,所以不需要这个hash值。

2
devtool: 'inline-source-map' 开启 source-map

3

devServer: {
    historyApiFallback: true,

    // Inline mode is recommended when using Hot Module Replacement.
    inline: true,
    open: true,
    hot: true,
    overlay: {
      errors: true,
      warnings: true,
    },
    watchOptions: {
      aggregateTimeout: 300,
      poll: 1000,
    },
    // contentBase: path.resolve(__dirname, 'client', 'dist'),
    contentBase: '/',
    // match the output path

    publicPath: '/',
    // match the output `publicPath`

    port: 8080,
    proxy: {
      '/**': {
        target: 'http://localhost:3000',
        secure: false,
        changeOrigin: true,
      },
    },
  },

open: true 会在命令执行时自动打开页面,contentBase: '/' 资源文件放置处,这里讲一下 proxy:
因为我的项目有自己的后台,用nodejs写的,运行在3000端口,但是webpack-dev-server也有一个自己的运行后台,运行在 port: 8080 端口,那该怎么办?
这时我们就需要进行代理proxy:
host 默认是 localhost,对8080端口下的服务器进行请求,会自动把相同的请求转到3000下,比如我们请求 localhost:8080/api,代理以后会自动请求 localhost:3000/api,我这里是把所以请求都转到端口3000下,也可以只进行 /api 代理:

'/api': {
    target: 'http://localhost:3000/api',
    secure: false,
    changeOrigin: true,
  },

4
关于 plugins:

plugins: [
  // 除了自己写的代码,将其他的modules都bundle成一个vendor文件
  new webpack.optimize.CommonsChunkPlugin({
    name: 'vendor',
    minChunks(module) {
      // 该配置假定你引入的 vendor 存在于 node_modules 目录中
      return module.context && module.context.indexOf('node_modules') !== -1;
    },
  }),
  // 生产manifest文件,用于在html模板中插入script
  new webpack.optimize.CommonsChunkPlugin({
    name: 'manifest',
    minChunks: Infinity,
  }),
  // 生成插入script后的 html 文件
  new HtmlWebpackPlugin({
    template: path.resolve(__dirname, 'server', 'static', 'index_template.html'),
    chunksSortMode: 'dependency',
    minify: {
      collapseWhitespace: true,
      removeComments: true,
    },
  }),

  // HMR
  new webpack.HotModuleReplacementPlugin(),
  // enable HMR globally

  new webpack.NamedModulesPlugin(),
  // prints more readable module names in the browser console on HMR updates
],

说到生成环境的webpack配置,可以参考我的项目webpack.prod.js 文件。

生产环境下我们就要兼顾很多方面,比如服务端发送到前端的文件要小,保证加载速度快;关于modules方面的代码我们很少会去修改,所以可以在前端进行缓存起来。

1

output: {
  path: path.resolve(__dirname, 'client', 'dist'),
  filename: '[name].[chunkhash].js',
  publicPath: '/',
},

这里对文件名进行了hash处理,浏览器会自动缓存js文件,如果收到的文件的文件名是一样的,浏览器可能不会进行自动更新,所以我们加上hash值,保证每一次更新以后的文件名不同,从而让浏览器再次请求资源。
而那些modules代码基本是不会变的,所以保证它们的hash值不变,在浏览器处缓存起来。

2
plugins

plugins: [
  // 将node_modules里的依赖整合成一个vendor
  new webpack.optimize.CommonsChunkPlugin({
    name: 'vendor',
    minChunks(module) {
      // 该配置假定你引入的 vendor 存在于 node_modules 目录中
      return module.context && module.context.indexOf('node_modules') !== -1;
    },
  }),

  new webpack.optimize.CommonsChunkPlugin({
    name: 'manifest',
  }),

  // 根据manifest生成插入script后的html
  new HtmlWebpackPlugin({
    template: path.resolve(__dirname, 'server', 'static', 'index_template.html'),
    chunksSortMode: 'dependency',
    minify: {
      collapseWhitespace: true,
      removeComments: true,
    },
  }),

  // 将js文件压缩成gz
  new CompressionPlugin(),

  // 允许创建一个在编译时可以配置的全局常量
  new webpack.DefinePlugin({
    'process.env': {
      NODE_ENV: '"production"',
    },
  }),

  // 压缩
  new webpack.optimize.UglifyJsPlugin({
    compress: {
      warnings: false,
    },
    // sourceMap: true,
  }),
  new WebpackMd5Hash(),
  new webpack.NoEmitOnErrorsPlugin(),
],

除了将文件进行 uglify 以外,这里还将文件进行gz压缩 new CompressionPlugin()
在后端部分还要加上一些代码:

// 获取js文件时选择gz压缩文件
app.get('*.js', (req, res, next) => {
  req.url = req.url + '.gz';
  res.set('Content-Encoding', 'gzip');
  next();
});

以上。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,042评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 89,996评论 2 384
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,674评论 0 345
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,340评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,404评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,749评论 1 289
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,902评论 3 405
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,662评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,110评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,451评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,577评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,258评论 4 328
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,848评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,726评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,952评论 1 264
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,271评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,452评论 2 348

推荐阅读更多精彩内容

  • 在现在的前端开发中,前后端分离、模块化开发、版本控制、文件合并与压缩、mock数据等等一些原本后端的思想开始...
    Charlot阅读 5,433评论 1 32
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,628评论 18 139
  • 构建一个小项目——FlyBird,学习webpack和react。(本文成文于2017/2/25) 从webpac...
    布蕾布蕾阅读 16,809评论 31 98
  • 最近在学习 Webpack,网上大多数入门教程都是基于 Webpack 1.x 版本的,我学习 Webpack 的...
    My_Oh_My阅读 8,173评论 40 247
  • 写在开头 先说说为什么要写这篇文章, 最初的原因是组里的小朋友们看了webpack文档后, 表情都是这样的: (摘...
    Lefter阅读 5,276评论 4 31