Webpack 初步概念

今天学习了一下 Webpack ,只是入门,对其有了一个初步的概念,大致只是到官方文档的概念这里。这里将我所学习到的这一些皮毛做一个整理,以加深自己的理解,并帮助其他和我一样正在学习的同学。

前置知识 至少对前端开发有些了解
阅读时间 10 ~ 20 min
demo 仓库地址:https://github.com/doooooit/hello-webpack

一、什么是 Webpack,用它来干嘛

每一个 Web App,即便是最简单的一个页面,也会使用一些其他资源,比如*.js*.css等等。不使用 Webpack 前,我们只能自己手动在 HTML 页面里管理这些依赖,写一堆<link ... />, <script src="..."></script>,一个两个还好,但现代的 Web 每一个都有一大堆依赖,而且不光是数量的问题,资源有可能是多种类型的,有可能是.js或者.ts,可能是.css或者.less,这些文件还需要编译。如果这些资源用手动管理将是一场灾难,不仅重复简单低效的劳动,还容易出错,既然是程序员,那就应该尽可能将任务交给计算机自动处理。

Webpack 正是为此而生,作为一个现代 JavaScript 应用程序的模块打包器(module bundler)而存在。它将每一个资源文件都视为一个模块(module),当 webpack 处理应用程序时,它会递归地构建一个依赖关系图(dependency graph),其中包含应用程序需要的每个模块,然后将所有这些模块打包成少量的捆包(bundle)。bundle 通常只有一个,由浏览器加载。

Webpack

这张从 Webpack 官网首页截取的图片基本可以直观地看出它是干啥的了。

在模块打包这一基本功能的基础上,Webpack 还有很多令人欣喜的特性,但这不在本文的讨论范围,请参考官方文档。

二、它是怎么工作的

Webpack 依照配置文件进行工作,文件名为webpack.config.js。在配置文件中,有四个 Webpack 的核心概念,如果不了解这些概念,就不能了解 Webpack。

1. 入口(Entry)

为了能够打包所有模块,Webpack 会构建一个依赖关系图,而这个图需要一个入口,来告诉 Webpack 从哪里开始构建。可以将它理解为运行 app 时第一个执行的文件。

module.exports = {
    entry: './path/to/my/entry/file.js'
};

入口可以有多个

2. 出口(Output)

既然是打包,那把所有东西捆起来之后就会生成(emit)一个捆包(bundle),这个就是出口。在配置文件中,出口需要两个属性,一个是路径,一个是文件名。

const path = require('path');

module.exports = {
    entry: './path/to/my/entry/file.js', 
    output: {
        path: path.resolve(__dirname, 'dist'),  // 路径
        filename: 'my-webpack-output.bundle.js' // 文件名
    }
};

3. Loader

Webpack 的目标是将所有模块都打包起来,而使浏览器不必关注这些事情。这些模块包括但不限于.js, .css, .sass, .png, etc,但是 Webpack 只能理解 JavaScript 不能理解其他文件模块类型可咋办。上帝说要有光,于是便有了 Loader。它的作用就是把所有非 JavaScript 的模块转换成 Webpack 能够理解的模块。

不同类型的模块有不同的 Loader,比如.css可能会用到 css-loader,而.txt可能会用到raw-loader

在配置文件中,对 Loader 需要配置两个必须的属性:
1. 识别出是哪种文件( test 属性)
2. 对识别出的文件使用对应的 Loader 以转换它( use 属性)

const path = require('path');

const config = {
    entry: './path/to/my/entry/file.js', 
    output: {
        path: path.resolve(__dirname, 'dist'), 
        filename: 'my-webpack-output.bundle.js'
    }, 
    module: {   // Loader 模块转换
        rules: [ // test 和 use 要写在 rules 中,而 rules 要写在 module 中
            {
                test: /\.txt$/,     // test 属性,可以使用正则表达式
                use: 'raw-loader'   // .txt 文件对应的 Loader
            }
        ]
    }
};

module.exports = config;

Loader 通常是第三方的,要使用哪个必须先安装,比如上文中的 raw-loader

npm install --save-dev raw-loader

Loader 的特性

  • loader 支持链式传递。能够对资源使用流水线(pipeline)。loader 链式地按照先后顺序进行编译。loader 链中的第一个 loader 返回值给下一个 loader。在最后一个 loader,返回 webpack 所预期的 JavaScript。
  • loader 可以是同步或异步函数。
  • loader 运行在 Node.js 中,并且能够执行任何可能的操作。
  • loader 接收查询参数。用于 loader 间传递配置。
  • loader 也能够使用 options 对象进行配置。
  • 除了使用 package.json 常见的 main 属性,还可以将普通的 npm 模块导出为 loader,做法是在 package.json 里定义一个 loader 字段。
  • 插件(plugin)可以为 loader 带来更多特性。
  • loader 能够产生额外的任意文件。

4. 插件(plugins)

Loader 通常只做转换,但有时候我们会需要进行一些更多的操作,最简单的比如添加一些注释信息,这时我们就可以使用插件,它可以完成 Loader 不能完成的任务。

Webpack 本身内置了一些插件,同时还可以安装第三方插件或自己写。使用 npm 进行安装。

const HtmlWebpackPlugin = require('html-webpack-plugin'); //通过 npm 安装
const webpack = require('webpack'); //访问内置的插件
const path = require('path');

const config = {
  entry: './path/to/my/entry/file.js',
  output: {
    filename: 'my-first-webpack.bundle.js',
    path: path.resolve(__dirname, 'dist')
  },
  module: {
    rules: [
      {
        test: /\.(js|jsx)$/,
        use: 'babel-loader'
      }
    ]
  },
  plugins: [
    new webpack.optimize.UglifyJsPlugin(),  // 使用 Webpack 内置插件
    new HtmlWebpackPlugin({template: './src/index.html'})   // 使用第三方插件
  ]
};
// 以上插件功能请查阅相应文档

module.exports = config;

三、使用

1. 安装

Webpack 可以全局安装,也可以在项目内安装,推荐后者。

npm install --save-dev webpack     // 安装最新版
npm install --save-dev webpack@<version>    // 安装特定版本

2. 编写配置文件

配置文件名为 webpack.config.js,位于项目的根目录下。对于本 demo ,配置文件内容如下

const webpack = require('webpack'); // webpack 内置插件
const path = require('path');       // 路径处理

module.exports = {
    entry: './entry.js',    // 入口
    output: {
        path: path.resolve(__dirname, 'dist'),  // 生成到当前目录的 dist 目录下
        filename: 'bundle.js'   // 生成文件名
    }, 
    module: {   // Loader 写在 module.rules 中
        rules: [
            {
                test: /\.css$/, 
                use: [
                    // 对一个模块可以使用多个 Loader ,顺序传递
                    { loader: 'style-loader' }, 
                    { loader: 'css-loader', 
                        option: {
                            modules: true
                        }
                    }
                ]
            }
        ]
    }
}

3. 打包

执行打包有多种方法,推荐使用在 package.json 中添加 script 字段实现,前提是你已经使用 npm 安装了 Webpack 。如下

...
"script": {
    "build": "webpack"
}, 
...

然后在终端执行命令

npm run build

然后 Webpack 会输出打包过程的一些信息,执行完后会在我们指定的 __dirname + dist 目录下生成 bundle.js 文件。

原来我们的 HTML 文件结构是这样的:

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="utf-8">
    <title>Hello Webpack</title>
    <link href="./src/css/main.css" rel="stylesheet">
</head>
<body>
    <h1>Hello Webpack</h1>
    <script src="./src/js/main.js"></script>
</body>
</html>

现在我们只需要这样:

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="utf-8">
    <title>Hello Webpack</title>
</head>
<body>
    <h1>Hello Webpack</h1>
    <script src="dist/bundle.js"></script>
</body>
</html>

后记

本人也是初学,所以可能有错误疏漏,欢迎大家指正。

引用

本文中部分文字描述和代码引用了 Webpack 中文官网 的相关内容。


版权声明 自由转载 - 保持署名 - 不可商用 - 不可演绎 (CC3.0 创意共享3.0许可证

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

推荐阅读更多精彩内容