前端面试题 -- webpack

前言

上一篇 前端面试题-小程序

随着前端的不断发展,现代前端开发的复杂度和规模越来越庞大。工程化的思想催生了很多流行框架的进程,作为现在最流行的前端构建工具--webpack,很多面试、工作场景中都会出现了它的身影。所以对于现在的前端来说,了解并能够使用webpack,无论对个人技能或者职场求职来说,都是一种有力的提升

感兴趣的小伙伴可以点击 这里,查看完整版前端面试题

如果文章中有出现纰漏、错误之处,还请看到的小伙伴留言指正,先行谢过

以下 ↓

1. 对webpack的了解

官方文档

本质上,webpack 是一个现代 JavaScript 应用程序的静态模块打包器(module bundler),将项目当作一个整体,通过一个给定的的主文件,webpack将从这个文件开始找到你的项目的所有依赖文件,使用loaders处理它们,最后打包成一个或多个浏览器可识别的js文件

核心概念:

  • 入口(entry)

入口起点(entry point)指示 webpack 应该使用哪个模块,来作为构建其内部依赖图的开始

可以通过在 webpack 配置中配置 entry 属性,来指定一个入口起点(或多个入口起点)

module.exports = {
  entry: './path/to/my/entry/file.js'
};
  • 输出(output)

output 属性告诉 webpack 在哪里输出它所创建的 bundles ,以及如何命名这些文件,默认值为 ./dist

  • loader

loaderwebpack 能够去处理那些非 JavaScript 文件(webpack 自身只理解 JavaScript

  • 插件(plugins)

loader 被用于转换某些类型的模块,而插件则可以用于执行范围更广的任务。插件的范围包括,从打包优化和压缩,一直到重新定义环境中的变量

  • 模式

通过选择 developmentproduction 之中的一个,来设置 mode 参数,你可以启用相应模式下的 webpack 内置的优化

module.exports = {
  mode: 'production'
};

2. webpack,里面的webpack.config.js怎么配置

let webpack = require('webpack');

module.exports = {
    entry:'./entry.js',  //入口文件
    
    output:{
        //node.js中__dirname变量获取当前模块文件所在目录的完整绝对路径
        path:__dirname, //输出位置
        filename:'build.js' //输入文件
    },
    
    module:{  
        // 关于模块的加载相关,我们就定义在module.loaders中
        // 这里通过正则表达式去匹配不同后缀的文件名,然后给它们定义不同的加载器。
        // 比如说给less文件定义串联的三个加载器(!用来定义级联关系):
        rules:[
          {
            test:/\.css$/,      //支持正则
            loader:'style-loader!css-loader'
          }
        ]
    },
    
    //配置服务
    devServer:{
        hot:true, //启用热模块替换
        inline:true 
        //此模式支持热模块替换:热模块替换的好处是只替换更新的部分,而不是页面重载.
    },
    
    //其他解决方案配置
    resolve:{ 
        extensions:['','.js','.json','.css','.scss']  
    },
    
    //插件
    plugins:[
        new webpack.BannerPlugin('This file is create by baibai')
    ]

}

3. webpack本地开发怎么解决跨域的

  • 下载 webpack-dev-server 插件
  • 配置 webpack.config.js 文件
// webpack.config.js

var WebpackDevServer = require("webpack-dev-server");

module.exports = {
    ...
    
    devServer: {
        ...
        port: '8088', //设置端口号
        // 代理设置
        proxy: {
            '/api': {
                target: 'http://localhost:80/index.php', // 目标代理
                pathRewrite: {'^/api' : ''}, // 重写路径
                secure: false, // 是否接受运行在 HTTPS 上
                
            }
        }
    }
}

4. 如何配置多入口文件

配置多个入口文件
entry: {
  home: resolve(__dirname, "src/home/index.js"),
  about: resolve(__dirname, "src/about/index.js")
}

5. webpack与grunt、gulp的不同

三者都是前端构建工具

gruntgulp 是基于任务和流的。找到一个(或一类)文件,对其做一系列链式操作,更新流上的数据, 整条链式操作构成了一个任务,多个任务就构成了整个web的构建流程

webpack 是基于入口的。webpack 会自动地递归解析入口所需要加载的所有资源文件,然后用不同的 Loader 来处理不同的文件,用 Plugin 来扩展 webpack 功能

webpack 与前者最大的不同就是支持代码分割,模块化(AMD,CommonJ,ES2015),全局分析

为什么选择webpack

6. 有哪些常见的Loader?他们是解决什么问题的

  • css-loader:加载 CSS,支持模块化、压缩、文件导入等特性
  • style-loader:把 CSS 代码注入到 JavaScript 中,通过 DOM 操作去加载 CSS
  • slint-loader:通过 SLint 检查 JavaScript 代码
  • babel-loader:把 ES6 转换成 ES5
  • file-loader:把文件输出到一个文件夹中,在代码中通过相对 URL 去引用输出的文件
  • url-loader:和 file-loader 类似,但是能在文件很小的情况下以 base64 的方式把文件内容注入到代码中去

7. 有哪些常见的Plugin?他们是解决什么问题的

  • define-plugin:定义环境变量
  • commons-chunk-plugin:提取公共代码

8. Loader和Plugin的不同

  • loader 加载器
webpack 将一切文件视为模块,但是 webpack 原生是只能解析 js 文件. loader 的作用是让 webpack 拥有了加载和解析非 JavaScript 文件的能力

module.rules 中配置,也就是说他作为模块的解析规则而存在,类型为数组

  • Plugin 插件
扩展 webpack 的功能,让 webpack 具有更多的灵活性

plugins 中单独配置。类型为数组,每一项是一个 Plugin 的实例,参数都通过构造函数传入

9. webpack的构建流程是什么

  1. 初始化参数:从配置文件和 Shell 语句中读取与合并参数,得出最终的参数
  2. 开始编译:用上一步得到的参数初始化 Compiler 对象,加载所有配置的插件,执行对象的 run 方法开始执行编译
  3. 确定入口:根据配置中的 entry 找出所有的入口文件
  4. 编译模块:从入口文件出发,调用所有配置的 loader 对模块进行翻译,再找出该模块依赖的模块,再递归本步骤直到所有入口依赖的文件都经过了本步骤的处理
  5. 完成模块编译:在经过第4步使用 loader 翻译完所有模块后,得到了每个模块被翻译后的最终内容以及它们之间的依赖关系
  6. 输出资源:根据入口和模块之间的依赖关系,组装成一个个包含多个模块的 Chunk,再把每个 Chunk 转换成一个单独的文件加入到输出列表,这步是可以修改输出内容的最后机会
  7. 输出完成:在确定好输出内容后,根据配置确定输出的路径和文件名,把文件内容写入到文件系统
在以上过程中,webpack 会在特定的时间点广播出特定的事件,插件在监听到感兴趣的事件后会执行特定的逻辑,并且插件可以调用 webpack 提供的 API 改变 webpack 的运行结果

10. 是否写过Loader和Plugin?描述一下编写loader或plugin的思路

编写 loader 时要遵循单一原则,每个 loader 只做一种"转义"工作。 每个 loader 的拿到的是源文件内容(source),可以通过返回值的方式将处理后的内容输出,也可以调用 this.callback() 方法,将内容返回给 webpack 。 还可以通过 this.async() 生成一个 callback 函数,再用这个 callback` 将处理后的内容输出出去

相对于 loader 而言,Plugin 的编写就灵活了许多。 webpack 在运行的生命周期中会广播出许多事件,Plugin 可以监听这些事件,在合适的时机通过 webpack 提供的 API 改变输出结果

11. webpack的热更新是如何做到的?说明其原理

具体可以参考 这里

12. 如何利用webpack来优化前端性能

  • 压缩代码。删除多余的代码、注释、简化代码的写法等等方式
  • 利用 CDN 加速。在构建过程中,将引用的静态资源路径修改为 CDN 上对应的路径
  • 删除死代码 Tree Shaking)。将代码中永远不会走到的片段删除掉
  • 优化图片,对于小图可以使用 base64 的方式写入文件中
  • 按照路由拆分代码,实现按需加载,提取公共代码
  • 给打包出来的文件名添加哈希,实现浏览器缓存文件

13. 如何提高webpack的构建速度

参考 这里

14. 怎么配置单页应用?怎么配置多页应用

  • 单页应用可以理解为 webpack 的标准模式,直接在 entry 中指定单页应用的入口即可
  • 多页应用的话,可以使用 webpackAutoWebPlugin 来完成简单自动化的构建,但是前提是项目的目录结构必须遵守他预设的规范

15. 什么是bundle,什么是chunk,什么是module

bundle 是由 webpack 打包出来的文件,Chunk 是指 webpack 在进行模块的依赖分析的时候,代码分割出来的代码块。module是开发中的单个模块

后记

前端发展日新月异,只有保持不断学习的姿态,才能走的更远

希望这些面试题可以帮助大家温故知新、查缺补漏,不断充实自己的专业技能,走的更远

最后推荐一波 GitHub前端面试题完整版,欢迎小伙伴们 star 关注

不定期更新,期待同行

以上

相关推荐