当前位置: 首页 > news >正文

Vue.js 高级组件开发:设计模式与实践

Vue.js 高级组件开发:设计模式与实践

    • 引言
    • 一、组合式 API 与动态依赖注入
      • 1. 基于 `provide/inject` 的动态依赖
      • 2. 动态依赖注入与懒加载
    • 二、动态渲染与自定义渲染函数
      • 1. 使用 Render 函数动态生成内容
      • 2. 自定义 `vnode` 操作
    • 三、复杂场景下的动态表单生成与验证
    • 四、高性能虚拟滚动与增量渲染
    • 五、结合 TypeScript 提高类型安全性
    • 六、总结
    • 参考资料

引言

在复杂的前端项目中,Vue.js 组件开发不仅要求模块化与复用性,还需要设计灵活的交互模式,同时优化性能与扩展性。本文将聚焦以下几个高难度主题:

  1. 组合式 API 与动态依赖注入
  2. 动态渲染模式与自定义渲染函数
  3. 复杂场景下的动态表单生成与验证
  4. 高性能虚拟滚动与增量渲染
  5. 结合 TypeScript 提高类型安全性

一、组合式 API 与动态依赖注入

1. 基于 provide/inject 的动态依赖

组合式 API 提供了更灵活的 provide/inject 机制,支持动态传递依赖。

案例:动态主题切换系统

<!-- ThemeProvider.vue -->
<template><div :class="`theme-${theme}`"><slot></slot></div>
</template><script>
import { provide, reactive } from "vue";export default {setup() {const themeState = reactive({ theme: "light" });provide("theme", themeState);return themeState;},
};
</script>
<!-- ChildComponent.vue -->
<template><div>当前主题:{{ theme.theme }}</div>
</template><script>
import { inject } from "vue";export default {setup() {const theme = inject("theme");return { theme };},
};
</script>

2. 动态依赖注入与懒加载

支持懒加载依赖,只有在组件使用时才初始化依赖,从而优化性能。

provide("service", () => import("./heavyService"));

在子组件中:

const service = inject("service");
service().then((module) => {module.default.doSomething();
});

二、动态渲染与自定义渲染函数

1. 使用 Render 函数动态生成内容

Render 函数提供了更强大的动态渲染能力,适合复杂的动态内容场景。

案例:动态生成树形菜单

<script>
export default {props: ["nodes"],render(h) {const renderNode = (node) =>h("li", { key: node.id }, [h("span", node.label),node.children && h("ul", node.children.map(renderNode)),]);return h("ul", this.nodes.map(renderNode));},
};
</script>

使用:

<DynamicTree :nodes="treeData" />

2. 自定义 vnode 操作

借助 Vue 的虚拟 DOM,可以对节点直接进行操作,实现动态插入复杂节点。

export default {render(h) {const vnode = h("div", { attrs: { id: "dynamic" } }, "动态节点");this.$nextTick(() => {vnode.elm.textContent = "内容已更新";});return vnode;},
};

三、复杂场景下的动态表单生成与验证

动态表单生成通常需要解决以下问题:

  1. 动态配置项支持
  2. 异步数据加载
  3. 多级嵌套验证

案例:基于 JSON Schema 动态表单

<template><form @submit.prevent="submit"><componentv-for="field in schema":is="field.component":key="field.name"v-model="formData[field.name]"v-bind="field.props"/></form>
</template><script>
export default {props: ["schema"],data() {return {formData: {},};},methods: {submit() {// 提交表单数据console.log(this.formData);},},
};
</script>

配合 AJV 进行动态验证:

import Ajv from "ajv";
const ajv = new Ajv();
const validate = ajv.compile(schema);if (!validate(formData)) {console.error(validate.errors);
}

四、高性能虚拟滚动与增量渲染

当数据量巨大时,传统渲染方法会导致性能瓶颈。虚拟滚动技术能有效解决此问题。

案例:自定义虚拟列表组件

<template><div ref="container" class="virtual-list" @scroll="handleScroll"><div class="spacer" :style="{ height: totalHeight + 'px' }"></div><divclass="item"v-for="(item, index) in visibleItems":key="index":style="{ transform: `translateY(${itemOffsets[index]}px)` }">{{ item }}</div></div>
</template><script>
export default {props: ["items", "itemHeight", "containerHeight"],data() {return {startIndex: 0,visibleCount: Math.ceil(this.containerHeight / this.itemHeight),};},computed: {visibleItems() {return this.items.slice(this.startIndex,this.startIndex + this.visibleCount);},totalHeight() {return this.items.length * this.itemHeight;},itemOffsets() {return Array.from({ length: this.visibleItems.length },(_, i) => (this.startIndex + i) * this.itemHeight);},},methods: {handleScroll() {const scrollTop = this.$refs.container.scrollTop;this.startIndex = Math.floor(scrollTop / this.itemHeight);},},
};
</script>

五、结合 TypeScript 提高类型安全性

在大型项目中,使用 TypeScript 可以避免常见类型错误,提高代码可靠性。

案例:为组件添加类型声明

<script lang="ts">
import { defineComponent, PropType } from 'vue';export default defineComponent({props: {title: {type: String as PropType<string>,required: true},count: {type: Number as PropType<number>,default: 0}},setup(props) {console.log(props.title, props.count);}
});
</script>

案例:泛型组件

<script lang="ts">
import { defineComponent } from 'vue';export default defineComponent({props: {items: {type: Array as PropType<T[]>,required: true}},setup<T>(props: { items: T[] }) {console.log(props.items);}
});
</script>

六、总结

在复杂场景下,Vue.js 的组件开发不仅需要基础特性的支持,还需要灵活运用动态渲染、自定义逻辑、性能优化以及类型安全工具。

通过掌握组合式 API、虚拟 DOM 操作、动态表单生成与 TypeScript 等高级特性,开发者可以应对各种复杂需求,并构建高效、可维护的大型前端项目。


参考资料

  • Vue 3 官方文档
  • Vue Composition API
  • 虚拟滚动库 Vue Virtual Scroller
  • JSON Schema
  • TypeScript 官方文档

相关文章:

Vue.js 高级组件开发:设计模式与实践

Vue.js 高级组件开发&#xff1a;设计模式与实践 引言一、组合式 API 与动态依赖注入1. 基于 provide/inject 的动态依赖2. 动态依赖注入与懒加载 二、动态渲染与自定义渲染函数1. 使用 Render 函数动态生成内容2. 自定义 vnode 操作 三、复杂场景下的动态表单生成与验证四、高…...

《一文读懂卷积网络CNN:原理、模型与应用全解析》

《一文读懂卷积网络CNN&#xff1a;原理、模型与应用全解析》 一、CNN 基本原理大揭秘&#xff08;一&#xff09;从人类视觉到 CNN 灵感&#xff08;二&#xff09;核心组件详解 二、经典 CNN 模型巡礼&#xff08;一&#xff09;LeNet-5&#xff1a;开山鼻祖&#xff08;二&a…...

MONI后台管理系统-数据敏感字段存储加密

前言&#xff1a;     在我们数据库中&#xff0c;存在很多的敏感数据&#xff0c;如用户表中&#xff0c;存在用户电话、身份证号、邮箱等属于用户的敏感信息&#xff0c;我们通常在存入数据库后&#xff0c;将其进行加密存储&#xff0c;以此来保证数据安全性。     …...

熟悉各类游戏设计模式的用途与限制,如 factory、strategy、mvc、object pool 等

良好的系统分析与设计能力要求开发者熟悉并正确运用各种设计模式来解决特定问题。设计模式是一种针对特定问题的通用解决方案&#xff0c;可提高代码的可复用性、可维护性和可扩展性。以下是对一些常见游戏设计模式的详细分析&#xff0c;包括其用途、限制和代码示例。 一、工厂…...

【RabbitMQ高级篇】消息可靠性问题(1)

目录 1.消息可靠性 1.1.生产者消息确认 1.1.1.修改配置 1.1.2.定义Return回调 1.1.3.定义ConfirmCallback 1.2.消息持久化 1.2.1.交换机持久化 1.2.2.队列持久化 1.2.3.消息持久化 1.3.消费者消息确认 1.3.1.演示none模式 1.3.2.演示auto模式 1.4.消费失败重试机制…...

ASP.NET |日常开发中常见问题归纳讲解

ASP.NET &#xff5c;日常开发中常见问题归纳讲解 前言一、性能问题1.1 数据库访问性能1.2 视图状态&#xff08;在ASP.NET Web Forms 中&#xff09; 二、安全问题2.1 SQL 注入2.2 跨站脚本攻击&#xff08;XSS&#xff09; 三、状态管理问题3.1 会话状态&#xff08;Session …...

【【深入浅出TinyRisc-v】】

深入浅出TinyRisc-v 本代码参考于 https://gitee.com/liangkangnan/tinyriscv 自己理解之后又重新写了一遍 tinyriscv.v // 涓嬮潰鏄鏁翠釜top妯″潡鐨勪功鍐? module tinyriscv(input clk ,input rst_n …...

常见的限流算法

常见的限流算法 限流的定义固定窗口算法滑动窗口算法漏桶算法&#xff08;推荐&#xff09;令牌桶算法(推荐)限流粒度本地限流&#xff08;单机限流&#xff09;分布式限流&#xff08;多机限流&#xff09;分布式限流的实现 限流的定义 限流&#xff0c;也称流量控制。是指系统…...

【Leetcode 每日一题】3159. 查询数组中元素的出现位置

问题背景 给你一个整数数组 n u m s nums nums&#xff0c;一个整数数组 q u e r i e s queries queries 和一个整数 x x x。 对于每个查询 q u e r i e s [ i ] queries[i] queries[i]&#xff0c;你需要找到 n u m s nums nums 中第 q u e r i e s [ i ] queries[i] q…...

xadmin后台首页增加一个导入数据按钮

xadmin后台首页增加一个导入数据按钮 效果 流程 1、在添加小组件中添加一个html页面 2、写入html代码 3、在urls.py添加导入数据路由 4、在views.py中添加响应函数html代码 <!DOCTYPE html> <html lang...

行为树详解(5)——事件驱动

【分析】 如果行为树的节点很多&#xff0c;那么会存在要经过很多节点才会走到动作节点的情况。显然&#xff0c;性能上不如状态机。 每帧都需要重新遍历一系列节点才会走到动作节点&#xff0c;而实际上很多条件节点在数帧内不会有变化&#xff0c;这是造成性能问题的重要原…...

3.若依前端项目拉取、部署、访问

因为默认RuoYi-Vue是使用的Vue2,所以需要另外去下载vue3来部署。 拉取代码 git clone https://gitee.com/ys-gitee/RuoYi-Vue3.git 安装node才能执行npm相关的命令 执行命令npm install 如果npm install比较慢的话&#xff0c;需要添加上国内镜像 npm install --registrhttp…...

Debian操作系统相对于Ubuntu有什么优势吗?

更高的稳定性&#xff1a;Debian 以其出色的稳定性闻名&#xff0c;得益于严格的软件包测试和发布流程。其稳定版经过长时间测试与验证&#xff0c;确保了系统的高度稳定&#xff0c;更适合对稳定性要求极高的长期运行服务器环境。而 Ubuntu 虽有稳定版本&#xff0c;但更新周期…...

【漏洞复现】CVE-2015-3337 Arbitrary File Reading

漏洞信息 NVD - CVE-2015-3337 Directory traversal vulnerability in Elasticsearch before 1.4.5 and 1.5.x before 1.5.2, when a site plugin is enabled, allows remote attackers to read arbitrary files via unspecified vectors. 在安装了具有“site”功能的插件以…...

win10、win11-鼠标右键还原、暂停更新

系统优化 win 10jihuo win 11jihuo鼠标右键还原暂停更新 update 2024.12.28win 10 jihuo winx&#xff0c;打开powershell管理员&#xff0c;输入以下命令,选择1并等待 irm https://get.activated.win | iex参考&#xff1a;https://www.bilibili.com/video/BV1TN411M72J/?sp…...

FFmpeg来从HTTP拉取流并实时推流到RTMP服务器

当使用FFmpeg来从HTTP拉取流并实时推流到RTMP服务器时&#xff0c;你可以使用以下命令&#xff1a; ffmpeg -i http://输入流地址 -c:v copy -c:a copy -f flv rtmp://RTMP服务器地址/应用名称/流名称 这是一个基本的命令示例&#xff0c;其中&#xff1a; - -i http://输入流地…...

Quo Vadis, Anomaly Detection? LLMs and VLMs in the Spotlight 论文阅读

文章信息&#xff1a; 原文链接&#xff1a;https://arxiv.org/abs/2412.18298 Abstract 视频异常检测&#xff08;VAD&#xff09;通过整合大语言模型&#xff08;LLMs&#xff09;和视觉语言模型&#xff08;VLMs&#xff09;取得了显著进展&#xff0c;解决了动态开放世界…...

Rust : tokio中select!

关于tokio的select宏&#xff0c;有不少的用途。包括超时和竞态选择等。 关于select宏需要关注&#xff0c;相关的异步条件&#xff0c;会同时执行&#xff0c;只是当有一个最早完成时&#xff0c;会执行“抛弃”和“对应”策略。 说明&#xff1a;对本文以下素材的来源表示感…...

【hackmyvm】hacked靶机wp

tags: HMVrootkitDiamorphine Type: wp 1. 基本信息^toc 文章目录 1. 基本信息^toc2. 信息收集2.1. 端口扫描2.2. 目录扫描2.3. 获取参数 3. 提权 靶机链接 https://hackmyvm.eu/machines/machine.php?vmHacked 作者 sml 难度 ⭐️⭐️⭐️⭐️️ 2. 信息收集 2.1. 端口扫描…...

MaixBit k210学习记录

开发背景&#xff1a;Window系统主机&#xff0c;在主机上安装了虚拟机&#xff08;VirtualBoxUbuntu23.04&#xff09; 目标实现&#xff1a;在虚拟机&#xff08;Ubuntu&#xff09;中&#xff0c;实现对Maix bit&#xff08;k210&#xff09;开发板的开发 虚拟机的安装参考…...

简易版抽奖活动的设计技术方案

1.前言 本技术方案旨在设计一套完整且可靠的抽奖活动逻辑,确保抽奖活动能够公平、公正、公开地进行,同时满足高并发访问、数据安全存储与高效处理等需求,为用户提供流畅的抽奖体验,助力业务顺利开展。本方案将涵盖抽奖活动的整体架构设计、核心流程逻辑、关键功能实现以及…...

深入浅出:JavaScript 中的 `window.crypto.getRandomValues()` 方法

深入浅出&#xff1a;JavaScript 中的 window.crypto.getRandomValues() 方法 在现代 Web 开发中&#xff0c;随机数的生成看似简单&#xff0c;却隐藏着许多玄机。无论是生成密码、加密密钥&#xff0c;还是创建安全令牌&#xff0c;随机数的质量直接关系到系统的安全性。Jav…...

postgresql|数据库|只读用户的创建和删除(备忘)

CREATE USER read_only WITH PASSWORD 密码 -- 连接到xxx数据库 \c xxx -- 授予对xxx数据库的只读权限 GRANT CONNECT ON DATABASE xxx TO read_only; GRANT USAGE ON SCHEMA public TO read_only; GRANT SELECT ON ALL TABLES IN SCHEMA public TO read_only; GRANT EXECUTE O…...

ESP32 I2S音频总线学习笔记(四): INMP441采集音频并实时播放

简介 前面两期文章我们介绍了I2S的读取和写入&#xff0c;一个是通过INMP441麦克风模块采集音频&#xff0c;一个是通过PCM5102A模块播放音频&#xff0c;那如果我们将两者结合起来&#xff0c;将麦克风采集到的音频通过PCM5102A播放&#xff0c;是不是就可以做一个扩音器了呢…...

学习STC51单片机31(芯片为STC89C52RCRC)OLED显示屏1

每日一言 生活的美好&#xff0c;总是藏在那些你咬牙坚持的日子里。 硬件&#xff1a;OLED 以后要用到OLED的时候找到这个文件 OLED的设备地址 SSD1306"SSD" 是品牌缩写&#xff0c;"1306" 是产品编号。 驱动 OLED 屏幕的 IIC 总线数据传输格式 示意图 …...

QT: `long long` 类型转换为 `QString` 2025.6.5

在 Qt 中&#xff0c;将 long long 类型转换为 QString 可以通过以下两种常用方法实现&#xff1a; 方法 1&#xff1a;使用 QString::number() 直接调用 QString 的静态方法 number()&#xff0c;将数值转换为字符串&#xff1a; long long value 1234567890123456789LL; …...

GC1808高性能24位立体声音频ADC芯片解析

1. 芯片概述 GC1808是一款24位立体声音频模数转换器&#xff08;ADC&#xff09;&#xff0c;支持8kHz~96kHz采样率&#xff0c;集成Δ-Σ调制器、数字抗混叠滤波器和高通滤波器&#xff0c;适用于高保真音频采集场景。 2. 核心特性 高精度&#xff1a;24位分辨率&#xff0c…...

Reasoning over Uncertain Text by Generative Large Language Models

https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829 1. 概述 文本中的不确定性在许多语境中传达,从日常对话到特定领域的文档(例如医学文档)(Heritage 2013;Landmark、Gulbrandsen 和 Svenevei…...

AGain DB和倍数增益的关系

我在设置一款索尼CMOS芯片时&#xff0c;Again增益0db变化为6DB&#xff0c;画面的变化只有2倍DN的增益&#xff0c;比如10变为20。 这与dB和线性增益的关系以及传感器处理流程有关。以下是具体原因分析&#xff1a; 1. dB与线性增益的换算关系 6dB对应的理论线性增益应为&…...

【Go语言基础【12】】指针:声明、取地址、解引用

文章目录 零、概述&#xff1a;指针 vs. 引用&#xff08;类比其他语言&#xff09;一、指针基础概念二、指针声明与初始化三、指针操作符1. &&#xff1a;取地址&#xff08;拿到内存地址&#xff09;2. *&#xff1a;解引用&#xff08;拿到值&#xff09; 四、空指针&am…...