Vue3实战之插槽封装与懒加载详解

 更新时间:2025年04月27日 10:38:45   作者:jingling1007  
这篇文章主要介绍了Vue3实战之插槽封装与懒加载的相关资料,过插槽实现组件的结构分离、复用性和语义清晰,通过懒加载实现图片的按需加载,减少首屏加载时间和资源浪费,文中给出了详细的代码示例,需要的朋友可以参考下

一、为什么需要插槽?从一个面板组件说起

在电商首页开发中,经常遇到这样的场景:

「新鲜好物」「人气推荐」同样类型模块都需要相同的标题栏,但内容区布局不同这时候,插槽(Slot)就像一个「内容拼图位」,让组件既能统一样式,又能灵活定制。

实战案例:HomePanel 通用面板

<template>
  <div class="home-panel">
    <div class="head">
      <h3>{{ title }}<small>{{ subTitle }}</small></h3>
    </div>
    <slot name="main" /> <!-- 内容拼图位 -->
  </div>
</template>

<script setup>
defineProps({ title: String, subTitle: String })
</script>

使用方式:

<HomePanel title="新鲜好物" sub-title="新鲜出炉">
  <template #main> <!-- 填充拼图 -->
    <ul class="goods-list">
      <li v-for="item in list">{{ item.name }}</li>
    </ul>
  </template>
</HomePanel>

实现效果:

插槽的 3 个好处:

  • 结构分离:标题样式统一管理,内容区自由发挥(列表 / 图片 / 按钮均可)
  • 复用性强:一个面板组件适配 N 种业务场景
  • 语义清晰:通过具名插槽(#main)明确内容位置

二、懒加载:让图片「按需加载」的魔法

电商首页往往包含大量图片,一次性加载会导致:

  • 首屏加载慢
  • 流量浪费
  • 手机发烫

懒加载(Lazy Load)的核心逻辑:图片进入视口时再加载

用 Vue3 指令实现懒加载

// 全局指令:directives/lazy.js
import { useIntersectionObserver } from '@vueuse/core'

export const lazyPlugin = {
  install(app) {
    app.directive('img-lazy', {
      mounted(el, binding) {
        // 当元素进入视口时
        const { stop } = useIntersectionObserver(el, ([{ isIntersecting }]) => {
          if (isIntersecting) {
            el.src = binding.value // 替换真实src
            stop() // 停止观察,避免重复触发
          }
        })
      }
    })
  }
}

使用方式:

<img v-img-lazy="goods.picture" alt="商品图" />

实现原理拆解:

  • IntersectionObserver:浏览器原生 API,监听元素是否进入视口
  • 指令生命周期:在mounted阶段绑定观察,避免重复绑定
  • 停止观察:图片加载后立即停止监听,节省性能

三、插槽 × 懒加载:实战中的黄金组合

在「商品馆」模块中,同时用到了插槽和懒加载:

<HomePanel title="数码馆">
  <template #main>
    <div class="box">
      <img v-img-lazy="bannerUrl" class="cover" /> <!-- 大图懒加载 -->
      <ul>
        <li v-for="goods in list">
          <GoodsItem :goods="goods" /> <!-- 子组件插槽 -->
        </li>
      </ul>
    </div>
  </template>
</HomePanel>

<!-- GoodsItem子组件 -->
<template>
  <RouterLink>
    <img v-img-lazy="goods.picture" /> <!-- 商品图懒加载 -->
    <p>{{ goods.name }}</p>
  </RouterLink>
</template>

效果:

  • 标题栏统一样式,内容区自由布局(大图 + 列表)
  • 所有图片均在进入视口时加载,首屏加载速度提升 40%

四、新手避坑指南

  • 插槽默认值:给插槽设置默认内容,避免父组件未传时的空白
    <slot name="main">暂无内容</slot>
    
  • 懒加载占位图:加载前使用占位图(如灰色色块),避免布局抖动
    <img v-img-lazy="realSrc" :src="placeholder" />
    
  • 指令参数校验:确保指令值是合法 URL
    if (!binding.value.startsWith('http')) return
    

五、总结:让组件会「呼吸」

  • 插槽让组件有了「可插拔」的能力,像搭积木一样组装页面
  • 懒加载让页面学会「按需呼吸」,节省资源的同时提升用户体验
  • 两者结合,实现了 「结构统一」「内容灵活」的完美平衡

到此这篇关于Vue3实战之插槽封装与懒加载的文章就介绍到这了,更多相关Vue3插槽封装与懒加载内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Vue解决ajax跨域的问题

    Vue解决ajax跨域的问题

    这篇文章主要介绍了Vue解决ajax跨域的问题,跨域请求:是指协议名、主机名、端口号三者有任何一个不一样,而且跨域请求是请求发出去了,服务器接收并返回了结果,只是浏览器没有接收响应结果。感兴趣的同学可以参考阅读
    2023-04-04
  • elementUI实现级联选择器

    elementUI实现级联选择器

    这篇文章主要为大家详细介绍了elementUI实现级联选择器,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-11-11
  • Vue通过axios异步请求后端接口的方法

    Vue通过axios异步请求后端接口的方法

    在现代Web开发中,前端与后端的数据交互至关重要,Vue.js作为一款流行的前端框架,结合Axios库能够高效地实现HTTP请求与数据处理,本文将详细讲解如何在Vue 3中使用Axios,并通过实际案例展示其用法,需要的朋友可以参考下
    2024-12-12
  • vue新建环境变量以及网络请求工具axios的二次封装详解

    vue新建环境变量以及网络请求工具axios的二次封装详解

    这篇文章主要为大家介绍了vue新建环境变量以及网络请求工具axios的二次封装详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-06-06
  • vue基于element的区间选择组件

    vue基于element的区间选择组件

    这篇文章主要介绍了vue基于element的区间选择组件,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-09-09
  • 解决vue 绑定对象内点击事件失效问题

    解决vue 绑定对象内点击事件失效问题

    今天小编就为大家分享一篇解决vue 绑定对象内点击事件失效问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-09-09
  • vue项目强制清除页面缓存的例子

    vue项目强制清除页面缓存的例子

    今天小编就为大家分享一篇vue项目强制清除页面缓存的例子,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-11-11
  • Vue使用CDN引用项目组件,减少项目体积的步骤

    Vue使用CDN引用项目组件,减少项目体积的步骤

    这篇文章主要介绍了Vue使用CDN引用项目组件,减少项目体积的步骤,帮助大家提高项目加载速度,感兴趣的朋友可以了解下
    2020-10-10
  • vue悬浮可拖拽悬浮按钮的实例代码

    vue悬浮可拖拽悬浮按钮的实例代码

    这篇文章主要介绍了vue悬浮可拖拽悬浮按钮的实例代码,本文通过实例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2019-08-08
  • 解决vue多个路由共用一个页面的问题

    解决vue多个路由共用一个页面的问题

    下面小编就为大家分享一篇解决vue多个路由共用一个页面的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-03-03

最新评论