webpack入门

一、webpack五大核心概念

1、entry

打包的入口,可以是字符串、数组或对象

2、output

打包的输出

3、mode

模式,可以分为开发模式(development)与生产模式(production),默认为生产模式

4、loader

默认只能处理js、json格式的文件,loader的作用就是将其他格式的文件,转换成webpack能够处理的文件

5、plugin

webpack插件,每一个插件都有一个特定的功能,能处理loader无法处理的事情

二、HtmlWebpackPlugin与CleanWebpackPlugin

  • HtmlWebpackPlugin:生成一个html文件,自动打包后的js文件
  • CleanWebpackPlugin:每次打包时自动清除旧的文件,默认清除output.path目录
const HtmlWebpackPlugin = require('html-webpack-plugin')
const { CleanWebpackPlugin } = require('clean-webpack-plugin')
module.export = {
    entry: {
    page1: './src/page1.js',
    page2: './src/page2.js',
    common: './src/common.js',
  },
  output: {
    path: './dist',
    filename: '[chunkhash].js'
  },
  plugin: [
    new HtmlWebpackPlugin({
        filename: 'index.html',  // 生成html的文件名
      template: './src/template.html',  // 指定模版
      title: 'this is title',  // 设置html的title,可在指定模版中通过ejs语法引入
      inject: true,  // 默认值,script标签位于body底部,可选值为body、header、false(表示不自动引入)
      hash: false,  // true表示引入js文件后面添加hash值作为参数
      chunks:L [page1, common], //  多入口打包会有多个文件,默认引入全部,此配置表示只引入page1,common
      minify: {
        collapseWhitespace: true,  // 去除空格
        minifyCSS: true,  // 压缩html内联的css
        minifyJS: true,  // 压缩html内联的js
        removeComments: true  // 移除注释
      }
    }),
    new CleanWebpackPlugin(),
  ]
}

三、处理项目中的样式文件

webpack默认无法处理css、less、scss等文件,需要借助插件来处理项目中的样式文件

  • npm i css-loader mini-css-extract-plugin sass-loader node-sass -D
  • npm i optimize-css-assets-webpack-plugin -D
const MiniCssExtractPlugin = require('mini-css-extract-plugin')
const OptimizeCssAssetsPlugin = require('optimize-css-assets-webpack-plugin')
module.export = {
    ...
  module: {
    rules: [
      {
        test: /\.(css|scss)$/,
        use: [
            MiniCssExtractPlugin,
          'css-loader',
          'sass-loader',
        ]
      }
    ]
  },
  plugins: [
    // 将css单独插入html中
    new MiniCssExtractPlugin(),
    // 压缩css大小
    new OptimizeCssAssetsPlugin,
    // 每次打包先清除dist文件夹
    new CleanWebpackPlugin(),
    new HtmlWebpackPlugin()
  ]
}

四、处理项目中的图片

npm i file-loader html-loader url-loader image-webpack-loader -D

  • file-loader 处理图片
  • html-loader 处理html中src引入的图片
  • url-loader 将图片转为base64,压缩图片体积
  • image-webpack-loader,压缩图片体积,不能与url-loader一起使用
module.exports = {
    ...
  output: {
    publicPath: './'
  },
  plugins: [
    new MiniCssExtractPlugin({
        filename: 'style/[name].css'
    })
  ]
  module: {
    rules: [
      {
        test: /\.(scss|css)$/,
        use: [
          {
            loader: MiniCssExtractPlugin.loader,
            options: {
              publicPath: '../'
            },
          },
          'css-loader',
          'sass-loader',
        ],
      },
      {
        test: /\.(png|jpeg|gif|svg|jpg)/,
        use: {
          loader: 'url-loader',
          options: {
            name: '[name]-[contenthash:10].[ext]', // ext表示文件的后缀名
            esModule: false, // 不使用es6的模块化语法
            outputPath: 'images',
            limit: 10 * 1024, // 表示小于10kb的图片会被base64编码
          },
        },
      },
      {
        test: '/\.html$/',
        use: 'html-loader',
      },
    ],
  },
}

五、理解name、hash、chunkhash、contenthash

  • name:chunk name,chunk的名称,多入口chunkname为其key值,单入口不指定默认为main,异步加载的模块默认为数字
  • hash:每次打包生成的hash,项目中任意与打包相关的文件内容改变,此hash就会改变
  • chunkhash:根据chunk的内容生成的hash
  • contenthash:单个输出文件内容的hash

六、devServer的使用

webpack-dev-server是webpack官方提供的一个小型express服务器,它可以为webpack打包生成的资源文件提供web服务

首先安装:npm i webpack-dev-server -D

在package.json中配置

"script": {
    ...
  "start": "webpack-dev-server"
}
// webpack.config.js
module.exports = {
    ...
  devServer: {
    contentBase: path.resolve(__dirname, 'public'),  // 告诉服务器从哪里提供内容,默认为当前工作目录
    watchContentBase: true,  // 监视contentBase里面的内容,一旦变化就reload
    watchOptions: {
        ignore: ''  // 忽略哪些文件的变化
    },
    historyApiFallback: true,  // 请求的资源不存在时返回index.html,比如vue-router的history模式
    clientLogLevel: 'none',  // 不要显示启动日志信息
    overlay: false,  // 如果出错,不要全屏提示
    progress: true,  // 控制台输出运行进度
    compress: true,  // 启用gzip压缩
    open: true,  // 自动打开浏览器
    host: '0.0.0.0',  // 如果希望被局域网访问,设置为0.0.0.0,默认为localhost
    port: '8888',  // 端口,默认8080
    useLocalIp: true,  // 使用本地ip,如果host设置0.0.0.0,这里需要设置为true
    hot: true,  // 开启热更新
    proxy: {  // 设置代理
        'api': {  // 当url中含有/api时就会使用这里设置的代理
        target: 'http://xxx.com',  // 目标服务器地址
        changeOrigin: true,  // 跨域
        ws: true,  // 代理websocket
        pathRewrite: {
            '^/api': ''  // url重写,将url里面的/api去掉
        }
      }
    }
  }
}

七、js兼容性处理

由于ie浏览器无法识别高阶语法,因此需要使用babel来转换这些语法

  • babel-loader
  • @babel/core (babel核心库、核心api)
  • @babel/preset-env (babel预设,babel是插件化的,转换不同的语法,需要不同的插件,预设的作用就是按需引入)
    npm i babel-loader @babel/core @babel/preset-env -D
// webpack.config.js
module.exports = {
    module: {
    rules: [
      {
        test: /\.js$/,
        exclude: /node_modules/,  // 不处理node_modules中的文件
        use: {
            loader: 'babel-loader',
          options: {
            presets: ['@babel/preset-env']
          }
        }
      }
    ]
  }
}

还要安装@babel/polyfill来处理ES6中的一些语法

@babel/polyfill 其实是 core-js2 与 regenerator-runtime 组成的一个集成包,使用 core-js2,则安装 @babel/polyfill,使用 core-js3 则安装 core-js 与 regenerator-runtime ,以 core-js3 为例

npm i core-js regenerator-runtime -S

// webpack.config.js
module.exports = {
    module: {
    rules: [
      {
        test: /\.js$/,
        exclude: /node_modules/,  // 不处理node_modules中的文件
        use: [
          {
            loader: 'babel-loader',
            options: {
                presets: [
                ['@bebel/preset-env', {
                    modules: false,  // 对ES6模块文件不做转化,以便使用 tree shaking
                  useBuiltIns: usage,  // 取值可以是false,'entry','usage'
                  corejs: 3,  // corejs版本号
                  targets: {}  // 需要兼容的浏览器,未配置,取 browserslist中的值
                }]
              ]
            }
          }
        ]
      }
    ]
  }
}

关于useBuiltIns取值的说明

  • false:需要在js文件顶部引入,不需要指定corejs版本号
// <--- core-js2 --->
// import "@babel/polyfill";
// <--- core-js3 --->
import "core-js/stable";
import "regenerator-runtime/runtime";
  • entry:需要在js文件顶部引入,需要指定corejs版本号,根据配置的浏览器,打包浏览器不兼容的内容
  • usage:不需要在js文件顶部引入,需要指定corejs版本号,根据配置的浏览器,以及代码中用到的API来按需打包
    当项目中的js越来越多,babel转换耗时越来越长,可以使用babel缓存及多进程打包来提高速度,安装thred-loader
// webpack.config.js
module.exports = {
    module: {
    rules: [
      {
        test: /\.js$/,
        exclude: /node_modules/,  // 不处理node_modules中的文件
        use: [
          {
            loader: 'thred-loader',  // 耗时比较长的 loader 才需要多进程,否则只会更慢
            options: {
                workers: 2  // 进程数
            }
          },
          {
            loader: 'babel-loader',
            options: {
                presets: [
                ['@bebel/preset-env', {
                    modules: false,  // 对ES6模块文件不做转化,以便使用 tree shaking
                  useBuiltIns: usage,  // 取值可以是false,'entry','usage'
                  corejs: 3,  // corejs版本号
                  targets: {}  // 需要兼容的浏览器,未配置,取 browserslist中的值
                }]
              ]
            }
          }
        ]
      }
    ]
  }
}

八、其他配置

1、resolve:设置模块如何被解析

alias:为路径设置别名,让引入变得更简单

// webpack.config.js
module.exports = {
    resolve: {
    alias: {
        '@': path.resolve(__dirname, 'src'),
      // $ 匹配结尾,不会影响vue/xxx的路径,表示项目中引入的vue为运行时版本
      'vue$': 'vue/dist/vue.runtime.js'
    }
  }
}
extensions: 引入哪些类型的文件时可以省略后缀名
resolve: {
  // 默认为['.js', '.json'],表示引入js、json、vue文件时不需要写后缀名
    extensions: ['.js', '.json', '.vue']
}

2、externals:设置某些库不被打包,从外部获取

// index.html 中 body 底部添加
<body>
  <% for (let src of htmlWebpackPlugin.options.cdnList) { %>
    <script src="<%= src %>"></script>
  <% } %>
  <!-- 直接写死,与上面二选一,推荐上面的方式 -->
  <script src="https://libs.baidu.com/jquery/2.0.0/jquery.min.js"></script>
</body>

// webpack.config.js 添加下面内容
module.exports = {
  externals: {
    jquery: "jQuery" // key 为引入的包名,value 为全局变量名
  },
  plugins: [
    new HtmlWebpackPlugin({
      // 用来在 html 中通过 script 引入,cdnList 为自定义变量。若直接写死则不需要此配置
      cdnList: ["https://libs.baidu.com/jquery/2.0.0/jquery.min.js"]
    })
  ]
}

3、watch:初始构建之后,继续监听任何已解析文件的更改

将此参数设置为 true,使用 webpack 打包之后,每当有文件内容发生变化,就会自动重新打包(devServer默认开启)

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

推荐阅读更多精彩内容

  • 我们实现功能丰富的应用,拥有复杂的js代码和一大堆依赖包,模块化,less等css预处理。 这些改进确实大大提供了...
    雪夜酱阅读 289评论 0 0
  • 1.为什么要使用webpack 现今的很多网页其实可以看做是功能丰富的应用,它们拥有着复杂的JavaScript代...
    YINdevelop阅读 502评论 0 5
  • 2018年8月25日更新,目前 webpack 已经更新值 4.17.1 ,本文所用到的各种库或多或少有些过时,跟...
    AizawaSayo阅读 255评论 0 2
  • 2017年12月7日更新,添加了clean-webpack-plugin,babel-env-preset,添加本...
    ZombieBrandg阅读 1,161评论 0 19
  • GitChat技术杂谈 前言 本文较长,为了节省你的阅读时间,在文前列写作思路如下: 什么是 webpack,它要...
    萧玄辞阅读 12,681评论 7 110