• webpack之深入浅出externals


    我们通常在做项目时可能会把第三方库打包到bundle中,比如下面这张图

    如果不想把第三方库打包到bundle中,这就有了externals。官方的使用externals比较简单,只需三步——

    1.在HTML中引入第三方库的cdn

    2.在webpack中配置externals

    externals: {
        jquery: "jQuery",
    }

    3.在js中引用

    const $ = require("jquery");
    $("#content").html("<h1>hello world</h1>");

    好,现在我们可以随心所欲的使用jquery插件并保证不会打包到bundle中。external是怎么办到的呢?下面我们通过bundle的源码来分析下原理。

    这里的/* 0 */和__webpack_require__分别指打包前js对应的模块函数(详细过程见我上一篇博客),这里就不细说了。这里可以看到module.exports = jQuery,就是说我们externals中的key指的是require的东西,value指的就是它,就是说“当require的参数是jquery的时候,使用jQuery这个全局变量引用它”。这种最简洁的externals配置方式为默认的global模式,就是在window上挂一个全局变量,然后直接可以使用这个变量。具体的流程是这样,我们在源码中使用require('jquery')后,可以直接把jquery加到externals中,得到一个打包的trunk.js,但是在引入这个trunkjs之前,肯定要先引入jquery这个库文件,这个库文件会创建一个全局变量jQuery,而咱们的trunkjs中externals的jquery是global模式,所以实际上trunkjs引入jquery的时候,就会从全局变量中引用,即module.export = jQuery

     

    当然,既然是通过这种externals方式,其实我们可以不用require引入,直接使用全局变量也是可以的。

    jQuery("#content").html("<h1>hello world</h1>");

     大家如果注意到我刚说过的global模式的话,没错,你也许已经猜到了,我可以任意的使用不同的输出方式。如果打包文件我想运行到node环境下,我得使用commonjs规范,所以你要这么写。

    externals: {
        jquery: "commonjs2 jQuery",
    }

    打包后会是这样子。

    然后我的项目中还用到了lodash,也想把它从bundle中移除,之前我的代码是这样子,引的是npm包

     现在我们的externals配置如下

    externals: {
            jquery: "jQuery",
            _: "lodash"
    
    }

    我们必须要去掉这个const ,否则的话会报一个错误 lodash is not defined。为什么会这样呢?因为我们的lodash输出是global格式的,我在这里先卖一个关子,我们先统一一下输出格式,加一个libraryTarget字段

    这个东西是干嘛用的呢?

    他是我们输出文件的模块化规范,想想我们上面配置的commonjs jquery是运行在node下,总之记住一句话——我们最长使用的模块化方案是commonjs2和umd,前者是为node环境,后者是为浏览器环境。一共有这几种规范:

    "var" - Export by setting a variable: var Library = xxx (default)

    "this" - Export by setting a property of this: this["Library"] = xxx
    "commonjs" - Export by setting a property of exports: exports["Library"] = xxx
    "commonjs2" - Export by setting module.exports: module.exports = xxx
    "amd" - Export to AMD (optionally named - set the name via the library option)
    "umd" - Export to AMD, CommonJS2 or as property in root

    然后报这个错误,也就是说我们的模块没有正确的输出,回到我们的externals,它更多的是指定当你引用一个包的时候,这个包(lodash)应该遵循哪种模块化方式(common,root,amd等等)引入,这意思就是说,打包的时候不需要关心他到底怎么输出。

    externals: {
            jquery: "jQuery",
            lodash: {
                commonjs: 'lodash',
                commonjs2: 'lodash',
                amd: 'lodash',
                root: '_'
            }
        },

    ok,记得要将之前的覆盖掉,替换成下面的require,因为在externals中我们规范的commmonjs规范为lodash

     

    也就是说,这就是我们最初的代码,即没有用过externals时候的代码,看,也就是说我们只需要配置externals和libraryTarget就可以,其他的业务逻辑代码不需要改变。包括我们的项目中还用了echarts,这个通通不用改变!!!!!

    也就是说最终的代码是externals配合libraryTarget一起使用,如果去掉umd的话,会报这个错误

    相应的源码是这样子

     

      就是说我不知道通过那种方式输出,所以我应该告诉webpack,我通过umd方式输出,即将你的 lodash 暴露为所有的模块定义下都可运行的方式。它将在 CommonJS, AMD 环境下运行,或将模块导出到 global 下的变量.加上umd的源码如下

     看到了吧,我通过require('lodash')引入模块,输出走的是commonjs规范,贴下最终的配置

    entry: {
            main: './src/index.js'
        },
        externals: {
            jquery: "jQuery",
            lodash: {
                commonjs: 'lodash',
                commonjs2: 'lodash',
                amd: 'lodash',
                root: '_'
            }
        },
        output: {
            filename: '[name].[chunkhash].js',
            path: path.resolve(__dirname,'dist'),
            libraryTarget: 'umd'
        },
  • 相关阅读:
    关于ListView的注意点
    推荐一波 瀑布流的RecylceView
    RecycleView的简单应用
    Java Junit单元测试
    Java 2 个 List 集合数据求并、补集操作
    Java @Validated 遇到的大坑
    Java中的Validated验证
    使用一条sql语句查询多表的总数
    Java thymeleaf模板获取资源文件的内容
    Java热启动
  • 原文地址:https://www.cnblogs.com/ssh-007/p/7957369.html
Copyright © 2020-2023  润新知