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

本文涉及的产品
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
简介: 面试官:ui组件可以自动加载,那么业务组件可以吗?

大厂面试题分享 面试题库

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

地址:前端面试题库

背景

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

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

问题

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

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

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

思路

  • 将所有的业务组件遍历出来生成一个Map<componentName,componentPath>
  • 使用glob库拿到所有的vuejs 文件
  • 定义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.node
    const body = node.body
    tempRes.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 然后ObjectPropertyvalueObjectExpressionObjectPropertyproperties属性是 一个 ObjectProperty[]

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

show一下成果

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

批量替换

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

大厂面试题分享 面试题库

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

地址:前端面试题库

相关文章
|
2月前
|
缓存 搜索推荐 索引
「Mac畅玩鸿蒙与硬件12」鸿蒙UI组件篇2 - Image组件的使用
在鸿蒙应用开发中,Image 组件用于加载和显示图片资源,并提供多种属性来控制图片的显示效果和适配方式。本篇将带你学习如何在鸿蒙应用中加载本地和远程图片、设置图片样式以及实现简单的图片轮播功能。
126 7
|
2月前
|
搜索推荐 Android开发 开发者
探索安卓开发中的自定义视图:打造个性化UI组件
【10月更文挑战第39天】在安卓开发的世界中,自定义视图是实现独特界面设计的关键。本文将引导你理解自定义视图的概念、创建流程,以及如何通过它们增强应用的用户体验。我们将从基础出发,逐步深入,最终让你能够自信地设计和实现专属的UI组件。
|
14天前
「Mac畅玩鸿蒙与硬件46」UI互动应用篇23 - 自定义天气预报组件
本篇将带你实现一个自定义天气预报组件。用户可以通过选择不同城市来获取相应的天气信息,页面会显示当前城市的天气图标、温度及天气描述。这一功能适合用于动态展示天气信息的小型应用。
113 38
|
5天前
|
人工智能 开发框架 JavaScript
LowCodeEngine:阿里开源的企业级低代码开发平台,提供预制的 UI 组件和模板,覆盖完整的研发周期
LowCodeEngine 是阿里巴巴开源的低代码开发框架,旨在通过拖拽、配置等简单操作,帮助开发者快速构建复杂的系统页面,提升开发效率和质量。
42 4
LowCodeEngine:阿里开源的企业级低代码开发平台,提供预制的 UI 组件和模板,覆盖完整的研发周期
|
2月前
|
前端开发 搜索推荐 开发者
「Mac畅玩鸿蒙与硬件20」鸿蒙UI组件篇10 - Canvas组件自定义绘图
Canvas 组件在鸿蒙应用中用于绘制自定义图形,提供丰富的绘制功能和灵活的定制能力。通过 Canvas,可以创建矩形、圆形、路径、文本等基础图形,为鸿蒙应用增添个性化的视觉效果。本篇将介绍 Canvas 组件的基础操作,涵盖绘制矩形、圆形、路径和文本的实例。
85 12
「Mac畅玩鸿蒙与硬件20」鸿蒙UI组件篇10 - Canvas组件自定义绘图
|
2月前
|
搜索推荐 前端开发 开发者
「Mac畅玩鸿蒙与硬件19」鸿蒙UI组件篇9 - 自定义动画实现
自定义动画让开发者可以设计更加个性化和复杂的动画效果,适合表现独特的界面元素。鸿蒙提供了丰富的工具,支持通过自定义路径和时间控制来创建复杂的动画运动。本篇将带你学习如何通过自定义动画实现更多样化的效果。
86 11
|
2月前
|
UED 开发者
「Mac畅玩鸿蒙与硬件18」鸿蒙UI组件篇8 - 高级动画效果与缓动控制
高级动画可以显著提升用户体验,为应用界面带来更流畅的视觉效果。本篇将深入介绍鸿蒙框架的高级动画,包括弹性动画、透明度渐变和旋转缩放组合动画等示例。
83 12
「Mac畅玩鸿蒙与硬件18」鸿蒙UI组件篇8 - 高级动画效果与缓动控制
|
2月前
|
UED
「Mac畅玩鸿蒙与硬件31」UI互动应用篇8 - 自定义评分星级组件
本篇将带你实现一个自定义评分星级组件,用户可以通过点击星星进行评分,并实时显示评分结果。为了让界面更具吸引力,我们还将添加一只小猫图片作为评分的背景装饰。
76 6
|
2月前
|
UED
「Mac畅玩鸿蒙与硬件17」鸿蒙UI组件篇7 - Animation组件基础
在应用开发中,动画效果可以增强用户体验。鸿蒙框架提供了 translate、scale 和 rotate 等动画功能,允许对组件进行平移、缩放和旋转等操作。本篇将介绍 Animation 组件的基础知识和示例代码。
99 10
|
2月前
|
前端开发 开发者
「Mac畅玩鸿蒙与硬件21」鸿蒙UI组件篇11 - Canvas 组件的静态进阶应用
在鸿蒙应用开发中,Canvas 组件不仅用于基础绘图,还提供了处理复杂路径和渐变效果的多种手段,帮助开发者实现精美的静态图形。本篇将介绍如何在 Canvas 中绘制复杂路径、创建渐变填充效果。
57 7
「Mac畅玩鸿蒙与硬件21」鸿蒙UI组件篇11 - Canvas 组件的静态进阶应用

热门文章

最新文章