Skip to content

starunaway/BuildReactWithWebpack

Repository files navigation

从零开始构建 react。使用 webpack+react 全家桶

从无到有的构建 react 应用,再到 webpack 的配置优化

npm init

现在前端的基础设施已经离不开 npm这个大的基础环境了。本文也是在此基础上进行讲解。欲食用,请确保已经成功安装了nodenpm

找一个自己心怡的风水宝地,在命令行中按下 npm init,按照提示随便输入一些简介,就生成了如下一个文件。文件内容未必一样,长的像就可以了

{
  "name": "buildreactwithwebpack",
  "version": "1.0.0",
  "description": "从零开始构建react。使用webpack+react全家桶",
  "main": "index.js",
  "scripts": {
    "build": "cross-env NODE_ENV=production webpack --config webpack/webpack.config.js ",
    "test": "echo \"Error: no test specified\" && exit 1"
  },
  "repository": {
    "type": "git",
    "url": "git+https://github.com/starunaway/BuildReactWithWebpack.git"
  },
  "author": "starunaway",
  "license": "MIT",
  "bugs": {
    "url": "https://github.com/starunaway/BuildReactWithWebpack/issues"
  },
  "homepage": "https://github.com/starunaway/BuildReactWithWebpack#readme"
}

为什么使用 webpack

首先来看一下原有的前端开发过程

在项目目录下创建一个 html 文件。内容如下:

<!DOCTYPE html>
<html>
  <head>
    <title>起步</title>
    <script src="https://unpkg.com/[email protected]"></script>
  </head>
  <body>
    <script src="./src/index.js"></script>
  </body>
</html>

可以看到在 index.html文件中,引入了 moment库和本地编写的 index.js文件

在项目目录下创建 src文件夹,在 src 下面创建一个 index.js文件。index.js文件内容如下,功能是显示当前的时间。(不要问为什么显示个时间还得引入一个库,如果问就是我懒)

function component() {
  var element = document.createElement('div');
  element.innerHTML = 'hello webpack';
  element.innerHTML += moment().format('YYYY-MM-DD HH:mm:ss');
  return element;
}

document.body.appendChild(component());

ps:这部分内容是webpack 官网上的,简单无脑通俗易懂,就直接借用了

如果需要引入一个公共的包,需要在script标签中引入网络上的文件路径。如果是本地文件,需要引入本地的相对路径。 这么做在文件数量较少的情况下还好,但是如果有几十上百个文件呢?

  1. 如何知道每个依赖包何时被使用?
  2. 如何知道哪个文件是否真正的被使用了?
  3. 如果保证文件的引入顺序不会出错呢?

初次使用 webpack 进行打包

接下来就来尝试一下 webpack吧。相信webpack,它会帮我们管理好这一切。
在项目目录下创建dist文件夹,在dist下面创建 index.html文件

<!DOCTYPE html>
<html>
  <head>
    <title>hello</title>
  </head>
  <body>
    <script src="index.js"></script>
  </body>
</html>

script 内引入的资源路径修改一下,接下来使用 npm i -S moment 安装一下 moment 包。并修改一下src/index.js的内容 唔,其实也没啥好修改的,就是在顶部引入一下 moment

import moment from 'moment';

// 以下还是原来的

接下来就要开始使用 webpack 了,是不是有点小激动?

首先运行 npm i -D webpack webpack-cli,这俩货缺一不可灰常重要。其中 webpack-cliwebpack4.0之后才需要的

接下来在package.json中添加一行start命令:npx webpack

ps:npx是调用项目内部安装的模块,主要是在命令行下。这里直接写成webpack也是可以的。

//.....
    "start": "npx webpack",


//.....
  "devDependencies": {

    "webpack": "^4.43.0",
    "webpack-cli": "^3.3.11",

  }


//.....

在命令行中运行 npm start ,将 dist/index.html 文件在浏览器中打开瞅一眼。真神奇,和刚才是一样一样的(除了日期,别抬杠)

使用 webpack 的配置文件

webpack 的配置项有很多很多很多,虽然可以通过命令行的方式一个个传入进去,但是每次都得写一堆配置,怎么能对得起天生懒惰的程序员!所以 webpack 提供了通过配置文件写入配置的方法。来试一下

在项目目录下创建 webpack.config.js文件

const path = require('path');

module.exports = {
  entry: './src/index.js',
  output: {
    filename: 'bundle.js',
    path: path.resolve(__dirname, 'dist'),
  },
};

简单明了,知道单词的意思就知道是干啥的了。如上。entry 就是打包的入口,output就是文件输出的路径

现在,把 dist/index.html 文件下 script 引入的路径由index.js 替换为 bundle.js。重新运行 npm start 试一下

webpack 打包资源

webpack 只有这点功能吗?作为前端三剑客,css 就不支持吗?当然不是!webpack 还可以打包各种资源,包括但不限于 css/less/scss,各种图片,字体,数据。但是有一个前提,需要先配置一下 loader。来试一下

首先在 src 目录下创建一个 style.less 文件(用 less 习惯了,关键是 less 不需要下其他的 loader,和css通用,省事)

.time {
  color: blue;
}

src/index.js 中给元素加上time样式,并引入 style.less

import moment from 'moment';
// 引入文件
import './style.less';

function component() {
  var element = document.createElement('div');
  element.innerHTML = 'hello webpack';
  element.innerHTML += moment().format('YYYY-MM-DD HH:mm:ss');

  // 添加类名
  element.classList.add('time');
  return element;
}
document.body.appendChild(component());

接下来修改一下 webpack.config.js,和 entry 平级,创建以下内容

  entry:{},
  output:{},
  module: {
    rules: [
      {
        test: /\.(less|css)$/,
        use: ['style-loader', 'css-loader'],
      },
    ],
  },

顾名思义,模块的规则是什么____?填空题。更高级的用法,以后慢慢讲

使用npm i -D style-loader css-loader,安装一下这俩 loader。安装完成后重新运行npm start。再次打开 dist/index.html,看看文字颜色

使用多入口

现在,我们在 html 文件里只引入了一个 js 文件。如果我们需要有多个 js 文件需要引入呢?我们需要手动去添加每一个文件吗?显然,这是不现实的。作为一个懒惰的程序员,怎么可能在这种事情上动手动脚呢。

那么怎么自动添加文件入口呢?来搞一下

嗯,现在src目录下创建一个新的js文件 src/print.js

export default function printMe() {
  console.log('I get called from print.js!');
}

修改src/index.js如下:

import moment from 'moment';
import printMe from './print';
import './style.less';

function component() {
  var element = document.createElement('div');
  var btn = document.createElement('button');

  btn.innerHTML = 'Click me and check the console!';
  btn.onclick = printMe;

  element.innerHTML = 'hello webpack';
  element.innerHTML += moment().format('YYYY-MM-DD HH:mm:ss');

  element.classList.add('time');
  element.appendChild(btn);

  return element;
}
document.body.appendChild(component());

什么叫多入口?就是在 index.html 里面同时引入多个文件,而不是把这些文件打包到一起。为什么要引入多个文件而不是打包成一个?一个简单的场景,首屏加载速度,一开始的时候加载很多暂时用不到的数据,浪费

言归正传,怎么使用 webpack 来进行操作呢?

module.exports = {
  entry: {
    app: './src/index.js',
    print: './src/print.js',
  },
  output: {
    filename: '[name].bundle.js',
    path: path.resolve(__dirname, 'dist'),
  },
  module: {
    //....
  },
};

很简单,在配置文件里修改 entryoutput。其中 output.filename[name]name,就是 entry 下的 key 值。至于怎么优化,接下来用到的时候再讨论。

现在,修改dist/index.html 文件

<!DOCTYPE html>
<html>
  <head>
    <title>hello</title>
    <script src="./print.bundle.js"></script>
  </head>
  <body>
    <script src="./app.bundle.js"></script>
  </body>
</html>

重新运行 npm start,在浏览器打开,瞅一眼

使用插件来管理入口

有没有发现每次修改完文件,需要重新打包的时候,都得修改一下 index.html?真的麻烦,怎么才能变得更懒呢?那就得借助插件了。插件是什么?不同于loader用来处理各式各样的资源,插件目的在于解决 loader 无法实现的其他事。webpack 提供了很多插件。具体可以查看 这里

来来来,打开命令行,跟我一起输入 npm i -D html-webpack-plugin
然后修改一下webpack.config.js

module.exports = {
  entry: {
    app: './src/index.js',
    print: './src/print.js',
  },
  output: {
    filename: '[name].bundle.js',
    path: path.resolve(__dirname, 'dist'),
  },
  module: {
    rules: [
      {
        test: /\.(less|css)$/,
        use: ['style-loader', 'css-loader'],
      },
    ],
  },
  //    新增这里
  plugins: [
    new HtmlWebpackPlugin({
      title: 'HtmlWebpackPlugin',
    }),
  ],
};

接下来可以删除一下dist文件夹,运行一下npm start,走你~

在每次构建前清理文件

如果你细心的话,会发现 dist 文件夹下有一堆东西,每次构建时其实有多文件是占那啥不那啥的,并且如果上生产环境,万一不小心添加一些没用但还产生影响的文件咋办?难道每次构建前都要手动去删除一下吗?这也太不懒了!

想想上一节对插件的定义?是不是想找一个插件来帮你做这件事?别说,还真有

命令行,npm i -D clean-webpack-plugin 走起

修改webpack.config.js

// ... require 其他的一堆包
const {CleanWebpackPlugin} = require('clean-webpack-plugin');

module.exports = {
  entry: {},
  //   ...其他的配置
  plugins: [
    new HtmlWebpackPlugin({
      title: 'HtmlWebpackPlugin',
    }),
    //  新增这个插件
    new CleanWebpackPlugin(),
  ],
};

再次运行npm start,有木有发现 dist 每次都只有这次打包的内容 !

开发时加入代码原始位置的映射

webpack 打包源代码时,把文件都打包到一起了,很难追踪到 error(错误) 和 warning(警告) 在源代码中的原始位置。
例如,如果将三个源文件(a.js,b.jsc.js)打包到一个 bundle(bundle.js)中,而其中一个源文件包含一个错误,那么堆栈跟踪就会直接指向到 bundle.js。 毕竟咱不是神,只是个懒惰的程序员,bug 是肯定存在的,又不想一行行翻代码,所以,就需要开启新的配置项了.

在配置文件中新增一个新的字段

module.exports = {
  //  ...
  devtool: 'inline-source-map',
};

随便写个 bug,现在会准确的告诉你是哪一行

使用 webpack-dev-server 实时重新加载

人一直在懒的路上越走越远。话说我每次修改文件,还得打包一次,刷新网页,烦不烦?怎么才能让它自己刷新呢?这就用到了新的webpack工具 : webpack-dev-serverwebpack-dev-server 在本地启动了一个 node 服务,用来监听文件变化。每次文件修改后,会重新 ”打包“一次,不过”打包“后的文件是放在内存中,并且 server 创建了一个socket,每次文件变动后,会向前端发送一次通知,前端在接到通知后重新加载页面,达到自动刷新的效果.

lets go,随我一起越来越懒吧

首先,需要npm i -D webpack-dev-server

修改 package.json 启动命令

{
  "scripts": {
    "start": "webpack-dev-server --open"
  }
}

webpack.config.js中配置一下devServer

module.exports = {
  entry: {
    app: './src/index.js',
    print: './src/print.js',
  },
  //...
  devtool: 'inline-source-map',
  devServer: {
    contentBase: './dist',
  },
  //...
};

重新执行 npm start,瞅一眼你的浏览器,随便改改代码,保存一下瞅瞅

开启模块热替换

模块热替换是 webpack 提供的最有用的功能之一。它允许在运行时更新所有类型的模块,而无需完全刷新。 在 webpack 4.x 版本后,已经不需要插件就可以支持了,省心省力省代码
webpack.config.js中配置一下devServer

module.exports = {
  //...

  devServer: {
    contentBase: './dist',
    hot: true,
  },
  //...
};

引入 react

重头戏来了,终于开始配置 react 了。话不多说,直接开始。主要是我也累了

想要使用 react,必须安装 react 的依赖 npm i -S react react-dom

react 使用了 JSX 语法,所以需要 babel 的支持。babel 是什么?告诉你一个好消息,本文作者和百度谷歌等成为了战略合作伙伴关系,各位读者可免费在以上网站搜索 所以嘞,还得再安装一下 babel

npm i -D @babel/core @babel/preset-react babel-loader

还记得之前讲过的吧,需要使用 loader 对资源进行打包,为了怕出错,把完整的 webpack 配置贴一下

const HtmlWebpackPlugin = require('html-webpack-plugin');
const {CleanWebpackPlugin} = require('clean-webpack-plugin');
const path = require('path');

module.exports = {
  entry: './src/index.js',
  output: {
    filename: '[name].bundle.js',
    path: path.resolve(__dirname, 'dist'),
  },
  module: {
    rules: [
      {
        test: /\.(less|css)$/,
        use: ['style-loader', 'css-loader'],
      },
      {
        test: /\.(js|jsx)$/,
        use: ['babel-loader'],
        exclude: /node_modules/,
      },
    ],
  },
  plugins: [
    new HtmlWebpackPlugin({
      template: './src/index.html',
    }),
    new CleanWebpackPlugin(),
  ],
  devtool: 'inline-source-map',
  devServer: {
    contentBase: './dist',
    hot: true,
  },
};

大部分配置和原来的一样,变换的部分:

  1. entry 文件现在改成了单入口,原因是教程到这,只需要一个入口就可以满足
  2. module.rules 新加了对 jsxloader
  3. react 要求必须有一个根元素,HtmlWebpackPlugin 默认的模板不能满足,所以我们自己创建了一个模板

看看模板长啥样 : src/index.html

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <meta http-equiv="X-UA-Compatible" content="ie=edge" />
    <title>React-Manage</title>
  </head>
  <body>
    <div id="root"></div>
  </body>
</html>

修改一下 src/index.js,作为 react 的起点

import React from 'react';
import ReactDom from 'react-dom';
import App from './App.js';

ReactDom.render(<App />, document.getElementById('root'));

src 下创建一个 App,走一个 src/App.js

import React from 'react';

function App() {
  return <div>this is a react app</div>;
}

export default App;

现在启动项目,发现 babel 是不起作用的,原因是 babel 的配置缺少了,补充一下 在项目目录下创建.babelrc,和 src 平级

{
  "presets": ["@babel/preset-react"]
}

再次启动项目,叮叮叮叮~我们的 react 终于跑起来了,接下来就可以进行开发了

对 webpack 按环境进行配置

现在我们的配置只有一个,但实际上我们需要在不同环境有不同的配置,比如在生产环境需要代码压缩,在开发环境需要一些开发工具帮我们提升开发效率。so,我们需要针对不同的环境进行不同的配置。

遵循不重复原则,保留一个通用配置。为了将这些配置合并在一起,我们将使用一个名为 webpack-merge 的工具

首先需要安装一下相关工具 npm i -D webpack-merge cross-env

在项目根目录下创建 webpack 文件夹,在此处存放相关配置。 首先创建 webpack.base.config.js,主要存放通用的基础配置

const HtmlWebpackPlugin = require('html-webpack-plugin');
const path = require('path');
module.exports = {
  entry: './src/index.js',
  output: {
    filename: '[name].bundle.js',
    path: path.resolve(__dirname, '../dist'),
  },
  resolve: {
    extensions: ['.js', 'jsx'],
    mainFiles: ['index'],
    alias: {
      '@src': path.resolve(__dirname, '../src'),
    },
  },
  module: {
    rules: [
      {
        test: /\.(less|css)$/,
        use: ['style-loader', 'css-loader'],
      },
      {
        test: /\.(js|jsx)$/,
        use: ['babel-loader'],
        exclude: /node_modules/,
      },
    ],
  },
  plugins: [
    new HtmlWebpackPlugin({
      template: './src/index.html',
    }),
  ],
};

接下来创建开发环境的配置。注意,开发环境的配置是在基础配置上做增量 webpack.dev.config.js

module.exports = {
  devtool: 'inline-source-map',
  devServer: {
    contentBase: './dist',
    port: 10086,
    hot: true,
    open: true,
  },
};

可以看到,dev 环境的配置只有 dev 相关的配置属性。同理,我们创建生产环境的配置 webpack.prd.config.js

const {CleanWebpackPlugin} = require('clean-webpack-plugin');

module.exports = {
  plugins: [new CleanWebpackPlugin()],
};

最后,使用 webpack-merge 工具将配置文件按照环境合并起来 webpack.config.js

const merge = require('webpack-merge');
const baseConfig = require('./webpack.base.config');
const devConfig = require('./webpack.dev.config');
const prdConfig = require('./webpack.prd.config');

module.exports = (env, argv) => {
  const config = process.env.NODE_ENV === 'development' ? devConfig : prdConfig;
  return merge(baseConfig, config);
};

文件结构如下

|-webpack
| |- webpack.config.js
| |- webpack.dev.config.js
| |- webpack.base.config.js
| |- webpack.prd.config.js
|
|-src
| |-- index.js
| |-- index.html
| |-- App.js
|
|- package.json
|- .babelrc

About

从零开始构建react。使用webpack+react全家桶

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages