位置:首页 > web前端 > css

vue渲染长列表时如何使用css优化提高页面性能

dearweb 发布:2021-09-25 09:38:21阅读:

相信不少和我一样的开发者在进行长列表网页开发时都遇到过,浏览器渲染会很慢的情况,通常我们会经常采用虚拟滚动、分页、上拉加载更多等不同的方式来进行优化,这些方式的思想都是一样的,都是只渲染可见区域,等用户需要时再加载更多的内容。而以上的方式无论哪种,都需要写大量的js或者css逻辑去实现。

而现在,我们多了一种方式——content-visibility。只需要一行CSS代码,就可以实现可见网页只加载可见区域内容,使网页的渲染性能得到数倍的提升!

content-visibility介绍

content-visibility是一个css属性,它控制一个元素是否呈现其内容,能让用户潜在地控制元素的呈现。用户可以使用它跳过元素的呈现(包括布局和绘制),直到用户需要为止,让页面的初始渲染得到极大的提升。

content-visibility的属性值

visible: 默认值。对布局和呈现不会产生什么影响。

hidden: 元素跳过其内容的呈现。用户代理功能(例如,在页面中查找,按Tab键顺序导航等)不可访问已跳过的内容,也不能选择或聚焦。类似于对其内容设置了display: none属性。

auto: 对于用户可见区域的元素,浏览器会正常渲染其内容;对于不可见区域的元素,浏览器会暂时跳过其内容的呈现,等到其处于用户可见区域时,浏览器在渲染其内容。

使用属性前后对比

使用前:

如下代码,在浏览器中简单的使用100个卡片,并对其设置扫光效果动画:

<style type="text/css">
      .card {
        position: relative;
        overflow: hidden;
        transition-duration: 0.3s;
        margin-bottom: 10px;
        width: 200px;
        height: 100px;
        background-color: #ffaa00;
      }
      .card::before {
        display: block;
        content: '';
        position: absolute;
        left: -200px;
        top: -0px;
        width: 300px;
        height: 15px;
        background: rgba(255, 255, 255, 0.3);
        transform: rotate(-45deg);
        position: relative;
        animation: mymove 1s linear infinite;
        -webkit-animation: mymove 1s linear infinite; /*Safari and Chrome*/
      }

      @keyframes mymove {
        from {
          left: -200px;
        }
        to {
          left: 200px;
        }
      }

      @-webkit-keyframes mymove /*Safari and Chrome*/ {
        from {
          left: -200px;
        }
        to {
          left: 200px;
        }
      }
    </style>
  
<body>
    <div class="card"></div>
    <div class="card"></div>
    <div class="card"></div>
    <div class="card"></div>
  </body>

渲染效果:从chrome可以看出,渲染时间花费了1454ms:

3ae34d716efe91e2955ca0baf75ee5af.jpg

使用后:

在class为card中添加 content-visibility: auto;

.card {
    position: relative;
    overflow: hidden;
    transition-duration: 0.3s;
    margin-bottom: 10px;
    width: 200px;
    height: 100px;
    background-color: #ffaa00;
    content-visibility: auto;
  }
  .card:before {
    content: '';
    // ...

渲染效果 可以从下图中明显的看到,使用content-visibility: auto;后渲染时间只需要381ms,性能提升了近4倍!而且随着元素内容变复杂,提升的性能会有更明显的上升。

413cb45ce997a37f661d5034bde18031.jpg

dom结构变化

再从下图的dom结构变化中也可以看到,当card未出现在屏幕可见区域内是,其内容(::before等动画)在元素出现在可见效果时才出现:

46a5b88df0e94490b3f0eed5e8321c77.jpg

对于这么好的一个属性,目前使用的情况并不是特别多,大部分开发者还没有用到实际项目中去,主要还是因为兼容性不高,不过小编觉得,兼容性的问题在不久的将来迟早是会得到解决的。

f63e25423da6925b75e52d883b72dd7e.jpg

部分元素导致浏览器渲染出问题

当元素的部分内容如标签这种,元素的高度是有图片内容决定的,因此在这种情况下,如果使用content-visibility,则可见视图外的img初始未渲染,高度为0,随着滚动条向下滑动,页面高度增加,会导致滚动条的滚动有问题。

 <style type="text/css">
      .card {
        margin-bottom: 10px;
        content-visibility: auto;
      }
    <div class="card">
      <img
        src="https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=1057266467,784420394&fm=26&gp=0.jpg"
        alt="小狗"
      />    
      <img
        src="https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=1057266467,784420394&fm=26&gp=0.jpg"
        alt="小狗"
      />

效果如下,可以看出滚动条随着图片的呈现,会出现问题:

1632535689(1).png

总体来说content-visibility是一个非常实用的CSS属性,通过一行CSS可以代替虚拟滚动、上拉加载更多等多种长列表渲染优化方式。

虽然其兼容性现在不是很好,但是相信不久的将来这并不是问题。现在来看是部分场景下它对浏览器的滚动条影响问题,如果你的列表项高度相同,那么可以通过contain-intrinsic-size来设置一个初始高度解决。如果列表项高度不固定而又非常重视用户的滚动条体验,那么不建议使用此属性。当然了,这一css属性出来的时间并不是太长,虽然它的完善,这一问题或许在将来也能够得到解决。


24人点赞 返回栏目 提问 分享一波

小礼物走一波,支持作者

还没有人赞赏,支持一波吧

留言(问题紧急可添加微信 xxl18963067593) 评论仅代表网友个人 留言列表

暂无留言,快来抢沙发吧!

本刊热文
网友在读
手机扫码查看 手机扫码查看