Vue 3 中集成 Three.js 场景的完整实践指南

本文详解如何在 vue 3(javascript 版本)项目中正确集成 three.js 基础场景,涵盖 dom 挂载、生命周期协调、渲染循环管理及常见陷阱规避。 本文详解如何在 vue 3(javascript 版本)项目中正确集成 three.js 基础场景,涵盖 dom 挂载、生命周期协调、渲染循环管理及常见陷阱规避。在 Vue 3 的响应式架构下直接嵌入 Three.js 渲染逻辑,需特别注意 DOM 元素可用性、渲染器生命周期与 Vue 组件生命周期的协同。原生 Three.js 示例依赖 document.body.appendChild(),但在 Vue 单文件组件(SFC)中,必须通过 ref 获取真实 DOM 节点,并在 onMounted 钩子中安全挂载渲染器------这是避免 Cannot read property 'appendChild' of null 等错误的关键。以下是一个可直接运行的 App.vue 完整实现(使用 JavaScript,非 TypeScript):<script setup>import { ref, onMounted, onUnmounted } from 'vue';import * as THREE from 'three';// 创建 DOM 引用容器const target = ref();// 初始化 Three.js 核心对象const scene = new THREE.Scene();const camera = new THREE.PerspectiveCamera( 75, window.innerWidth / window.innerHeight, 0.1, 1000);const renderer = new THREE.WebGLRenderer({ antialias: true });renderer.setSize(500, 500); // 推荐显式设置尺寸,避免默认拉伸renderer.setPixelRatio(window.devicePixelRatio); // 提升高 DPI 屏幕显示质量// 创建绿色立方体const geometry = new THREE.BoxGeometry(1, 1, 1);const material = new THREE.MeshBasicMaterial({ color: 0x00ff00 });const cube = new THREE.Mesh(geometry, material);scene.add(cube);camera.position.z = 5;// 动画循环函数(闭包内维持引用,避免重复创建)let animationId = null;function animate() { animationId = requestAnimationFrame(animate); cube.rotation.x += 0.01; cube.rotation.y += 0.01; renderer.render(scene, camera);}// 组件挂载后:挂载渲染器 DOM 并启动动画onMounted(() => { if (target.value) { target.value.appendChild(renderer.domElement); animate(); }});// 【重要】组件卸载前清理资源,防止内存泄漏onUnmounted(() => { if (animationId) { cancelAnimationFrame(animationId); } if (renderer.domElement && renderer.domElement.parentNode) { renderer.domElement.parentNode.removeChild(renderer.domElement); } renderer.dispose(); // 释放 WebGL 资源});</script><template> <div ref="target" class="three-container"></div></template><style scoped>.three-container { width: 500px; height: 500px; margin: 2rem auto; border: 1px solid #eee;}</style>? 关键要点说明: ref + onMounted 是黄金组合:确保 renderer.domElement 只在真实 DOM 节点就绪后才插入; 显式 setSize() 优于 renderer.setSize(window.innerWidth, ...):避免因响应式容器尺寸变化导致渲染异常,推荐结合 CSS 控制容器宽高,Three.js 渲染器按需适配; 务必调用 renderer.dispose() 和 cancelAnimationFrame:Vue 组件可能被复用或销毁,未清理的动画帧和 WebGL 上下文将引发内存泄漏; 添加 antialias: true 和 setPixelRatio:显著提升视觉质量,尤其在 Retina 屏幕上; 避免在 setup() 顶层执行渲染逻辑:Three.js 对象(如 scene, camera)可定义在顶层,但 DOM 操作与动画启动必须延迟至 onMounted。? 进阶提示:如需响应窗口缩放,可监听 window.resize 事件并更新 camera.aspect 与 renderer.setSize(),但需注意防抖以避免高频重绘。 Zeemo AI 一款专业的视频字幕制作和视频处理工具

相关推荐
92year3 小时前
用Google ADK从零搭一个能调工具的AI Agent:Python实操全过程
python·ai·mcp
woxihuan1234564 小时前
SQL删除数据时存在依赖关系_设置外键级联删除ON DELETE
jvm·数据库·python
东风破1374 小时前
DM8达梦共享存储集群DSC搭建步骤
数据库·学习·dm达梦数据库
雪碧聊技术4 小时前
当数据库字段数大于Java实体类属性数时,MyBatis还能映射成功吗?一文详解
数据库·自动映射·mybatis映射机制·java实体类·宽容映射机制
Jetev4 小时前
如何确定SQL字段是否为空_使用IS NULL与IS NOT NULL
jvm·数据库·python
蛐蛐蛐5 小时前
昇腾910B4上安装新版本CANN的正确流程
人工智能·python·昇腾
m0_702036535 小时前
mysql如何处理不走索引的OR查询_使用UNION ALL优化重写
jvm·数据库·python
代钦塔拉5 小时前
Qt4 vs Qt5 带参数信号槽的连接方式详解
开发语言·数据库·qt
2401_846339565 小时前
MySQL在云环境如何选择存储类型_SSD与高性能云盘配置建议
jvm·数据库·python
2601_957780846 小时前
Claude 4.6 对阵 GPT-5.4:2026 开发者大模型 API 选型深度解析
人工智能·python·gpt·ai·claude