面试官:ui组件可以自动加载,那么业务组件可以吗?

news/2024/11/15 6:53:45/

大厂面试题分享 面试题库

前端面试题库 (面试必备)   推荐:★★★★★

地址:前端面试题库

背景

笔者在最近在公司接手了一个老的对内使用的项目,接手后体验了下 发现首屏加载比较慢。分析了下大概的原因是main.js挂载了太多了东西,没有开启gzip的话app.js有4.2M。

按照常规的思路就是把全局引入的东西手动去掉,可是手动这个项目设计到的页面太多了,纯人工来改的话涉及到很多人天的工作量,

问题

基于的代码我们可以明显的发现一些可以优化的小点点

  • 项目在main.js中有全局引入业务组件(还有directives filters),但是业务组件不一定是在每个页面都使用
  • 项目在main.js中有全局引入常量(还有utils)

这两个点在很多项目都会有,常规的思路是我们手动的一个组件一个组件的搜,然后修改。可是对于文件很多的老系统来说不太现实,需要大量的人去做这种事情。

思路

  • 将所有的业务组件遍历出来生成一个Map<componentName,componentPath>
  • 使用glob库拿到所有的vue js 文件
  • 定义2个方法
    • 使用 vue-template-compiler 解析模板 看一下当前页面是否使用公共业务组件,有的话放到一个数组内
    • 使用@babel/parser 解析js生成ast
    • 解析 ast 看看是否引入 并且在components内注册,如果没有的话引入,并且注册

动手

首先我们把所需要的包安装一下,我们先操作单个文件,

pnpm i @babel/parser @babel/generator @babel/traverse @babel/types vue-template-compiler glob
复制代码

生成 Map<componentName,componentPath>

我们项目的业务组件还比较规范(如果实在不规范,其实手动维护一下这个Map也工作量不大),src/common/components有2个文件夹 basic 是基础组件,business里面是基于基础组件生成的业务组件,当然代码库内的代码都是随便写的,只是为了展示如何做自动加载组件

这一部分代码比较简单,引入fs,然后遍历文件夹就可以了


/**** @param {string} p 路径*/
function resolve (p) {return path.resolve(__dirname, '..', p)
}
// 所有组件的组件的 名称和路径映射
const allComponentMaps = fs.readdirSync(resolve('src/common/components/basic')).reduce((prev, cur) => {// 我们项目不存在直接放外面的组件if (!cur.includes('.')) {prev[cur] = `@/common/components/basic/${cur}`}return prev
}, {})fs.readdirSync(resolve('src/common/components/business')).reduce((prev, cur) => {// 我们项目不存在直接放外面的组件if (!cur.includes('.')) {prev[cur] = `@/common/components/business/${cur}`}return prev
}, allComponentMaps)console.log(allComponentMaps)
复制代码

用nodemon运行这个js 可以得出如下结果

解析vue文件

首先准备一个app.vue内容如下

<template><div id="app"><div id="nav"><s-input /><s-file /></div><router-view/></div>
</template><script>
export default {data () {return {}},components: {}
}
</script>
复制代码

首先通过fs模块得到源码的内容 content

// 目标文件
const targetFile = path.resolve(__dirname, './App.vue')// 得到文件内容
const content = fs.readFileSync(targetFile).toString()
复制代码

然后编写一个方法解析html模板

/**** @param {t.node} node* @param {Set} result*/
function parseHTML (node, result = new Set()) {if (allComponentList.some(item => item === node.tag)) {result.add(node.tag)} else {(node.children || []).forEach(element => {parseHTML(element, result)})}return result
}const result = parseHTML(compiler.compile(content).ast)
复制代码

可以看到控制台输出

得到当前文件使用了哪些业务组件后,接下来就是解析js,然后动态的import进去并注册就好了

借助一个网站我们可以知道 直接import A from ‘b'中的A是 ImportDefaultSpecifier类型

通过看ast。我们可以得知一条import 语句是ImportDeclaration类型的,所以借助@babel/types 可以生成ImportDeclaration,

如何使用@babel/types 生成语句

我们使用 t 代表@babel/types
一个import 语句的type是ImportDeclaration 那么就调用 t.importDeclaration方法 看api文档可以得知 t.importDeclaration 方法第一个入参就是ImportDefaultSpecifier类型 ImportDefaultSpecifier 可以借助t.importDefaultSpecifier方法生成

在入口文件生成imort 语句

/**** @param {string} str* @returns*/
function camelToStr (str) {return str.replace(/-([a-z])/g, function (all, letter) {return letter.toUpperCase()})
}
// 将用到的业务组件,并且没有引入的 引入一下
traverse(scriptAst, {Program (path, state) {const node = path.nodeconst body = node.bodytempRes.forEach(componentName => {const importDefaultSpecifier = t.importDefaultSpecifier(t.identifier(camelToStr(componentName)))const importDeclaration = t.importDeclaration([importDefaultSpecifier],t.StringLiteral(allComponentMaps[componentName]))body.unshift(importDeclaration)})}
})const sc = generator(scriptAst.program)
// 先随便生成到一个地方做测试
fs.writeFileSync('./a.vue',content.replace(/<script>([\s\S]+?)<\/script>/,`<script>\n${sc.code}\n</script>`)
)
复制代码

生成的效果图如下,我们可以明显看到需要引入的s-file 和 s-input都引入进来了

那下一步就是判断export default 里面是否有 components 并注册 组件了

自动注册组件

看一下ast 想要构建一个components 我们需要 ObjectProperty 然后ObjectPropertyvalue是 ObjectExpression ObjectPropertyproperties属性是 一个 ObjectProperty[]

所以我们可以得出全部的代码如以下链接

show一下成果

通过图片得知,我们当前的替换是成功了,简单版的业务业务组件自动导入就做好了

批量替换

这里批量替换笔者先不做,准备用一整篇文章来写,因为里面内容很多,也会因为一些人写代码的方式问题遇到非常多的问题和跳转

大厂面试题分享 面试题库

前端面试题库 (面试必备)   推荐:★★★★★

地址:前端面试题库


http://www.ppmy.cn/news/7040.html

相关文章

LeetCode题目笔记——1759. 统计同构子字符串的数目

文章目录题目描述题目难度——中等方法一&#xff1a;数学代码/C代码/Python总结题目描述 给你一个字符串 s &#xff0c;返回 s 中 同构子字符串 的数目。由于答案可能很大&#xff0c;只需返回对 109 7 取余 后的结果。 同构字符串 的定义为&#xff1a;如果一个字符串中的…

Arco 属性

文章目录Arco介绍1. 简介1.1 背景1.2 运行环境1.3 浏览器兼容性2. 设计价值观2.1 清晰2.2 一致2.3 韵律2.4 开放3. 设计原则3.1 及时反馈3.2 贴近现实3.3 系统一致性3.4 防止错误发生3.5 遵从习惯3.6 突出重点3.7 错误帮助3.8 人性化帮助4. 界面总体风格4.1 页面风格4.1.1 主色…

HMS Core 3D流体仿真技术,打造移动端PC级流体动效

移动设备硬件的高速发展&#xff0c;让游戏行业发生翻天覆地的变化&#xff0c;许多酷炫的游戏效果不再局限于电脑端&#xff0c;玩家在移动端就能享受到场景更逼真、画质更清晰、体验更流畅的游戏服务。但由于移动设备算力不足&#xff0c;为了实现真实感的水体效果&#xff0…

求多折线交点

目录 <font color=blue size=4 face="楷体">numpy.isclose<font color=blue size=4 face="楷体">基本思路<font color=blue size=4 face="楷体">代码1<font color=blue size=4 face="楷体">代码2——函数式,…

少儿Python每日一题(2):整数的位数

原题解答 本次的题目如下所示&#xff08;原题出处&#xff1a;蓝桥杯&#xff09;&#xff1a; 【编程实现】 输入一个正整数&#xff0c;输出这个正整数是一个几位数。 输入描述&#xff1a;输入一个正整数 输出描述&#xff1a;输出这个正整数是一个几位数 【样例输入】 12…

分布式理论之分布式事务

写在前面 我们知道&#xff0c;像MySQL的InnoDB存储引擎提供了事务的能力&#xff0c;严格遵守AICD的事务要求&#xff0c;但是在分布式环境中&#xff0c;一个请求会在多个服务实例存在多个事务&#xff0c;如购物&#xff0c;会有订单系统&#xff0c;支付系统&#xff0c;物…

【Python数据可视化】使用geoplotlib绘制地理空间数据

geoplotlib前言一、安装geoplotlib包二、读取csv数据使用1.点密度可视化2.直方图3. Voronoi图总结前言 ❤️❤️希望大家能多多点赞。❤️❤️ 需要数据集的可以评论。 Geoplotlib 是地理空间数据可视化的开源Python库&#xff0c;包含了大量的地理空间可视化操作&#xff0c…

UVM入门和进阶实验0

一. 概述 UVM学习流程仍然按照SV时候的核心流程&#xff0c;即&#xff1a; 如何搭建验证框架验证组件之间的连接和通信如何编写测试用例&#xff0c;继而完成复用和覆盖率的收敛 我们UVM入门和进阶实验0还是同之前SV验证实验0思想一样&#xff0c;让大家通过简单的实验要求…