引入外部js脚本加载慢与页面白屏问题的解决


Posted in Javascript onDecember 10, 2018

问题背景

最近做的一个项目需要引入一个外部的第三方js脚本。由于这是一个关于渲染3D建筑的脚本,所以体积比较大,大概有2M,加载完成也得要个好几秒,网速慢的时候十几秒都有可能。

之前也遇到脚本加载慢的问题,但是没这么慢,所以这次就特别写个文章记录一下我的解决过程。

首先上两张项目已完成的截图。

下图是通过第三方脚本渲染出来的3D建筑页面

引入外部js脚本加载慢与页面白屏问题的解决

下图是首页,不需要用到第三方脚本

引入外部js脚本加载慢与页面白屏问题的解决

遇到的问题和需求

  • 引入外部脚本太大,加载时间太长
  • 首页用不到外部脚本,需要先渲染出来
  • 用到外部脚本的页面,要是脚本还没加载好就点进去会报错

解决问题的过程

我一开始通过<script>标签直接引入到入口文件的头部,如下

<head>
<script src="./DDEarth.js"></script>
</head>

这样页面是可以正常加载的,但是页面出来的很慢,一开始会白屏一段时间等待这个js脚本加载完成。虽然脚本体积大是事实,但这用户体验肯定是可以优化的。

后来我又把这个脚本放到了页面底部,也就是</body>标签下面。这样可以先让页面渲染出来,再慢慢加载这个庞大的脚本,于是首页是出来的很快,但是从首页跳转到需要用到这个脚本的页面就会报错,如下

引入外部js脚本加载慢与页面白屏问题的解决

这个错误原因是这个页面需要用到window.DDEarth这个对象,但是由于此时这个脚本还没有加载完成,所以window下并没有这个对象,所以就报错了。

于是我又想到等脚本加载完成再执行相关方法,这时就需要用到onload这个方法了,onload这个方法在脚本加载完成的时候会执行。我引入脚本的时候给它加了个id,方便以后通过dom找到,代码如下:

// 入口文件
 </body>
 <script id="ddEarthScript" src="./DDEarth.js"></script>

// PageTwo.js
componentDidMount() {
   const scriptEle = document.getElementById('ddEarthScript'); // 找到脚本节点
   if (scriptEle) {
    scriptEle.onload = () => {
      // 脚本加载完成执行加载地图的操作
     this.loadEarthMap();
    };
   }
 }

有了以上代码我跳转到PageTwo这个页面的时候,会等到DDEarth.js这个脚本加载完成,再执行加载建筑地图的操作,这样就不会报错了。

但是这又有一个问题,就是如果我跳转到PageTwo的之前,DDEarth.js已经加载完成了,onload这个事件在PageTwo这个页面中就不生效了,loadEarthMap这个方法自然也就不会执行了。

这个时候需要加一个判断,完整代码如下:

// PageTwo.js
componentDidMount() {
  if (window.DDEarth) { // 如果跳转到此页面之前,脚本已加载完成
   this.loadEarthMap();
  } else {
  const scriptEle = document.getElementById('ddEarthScript');
  if (scriptEle) {
   scriptEle.onload = () => {
    this.loadEarthMap();
   };
  }
  }
 }

总结一下我以上解决问题的步骤

在入口文件的底部引入第三方脚本,并给它加个id。当然也可以放在<head>里,但是需要额外加上html5新增的 async 这个属性,这样脚本才能异步加载。

在需要用到这个脚本的页面,先判断脚本有没有加载完成(我这里是直接判断window.DDEarth对象是否为空)。如果已经加载完成,就直接执行相关操作;如果没有,先通过document.getElementById找到那个脚本,然后监听脚本的onload事件,再做相关操作。

什么情况可以用我以上思路?

  • 引入的第三方脚本较大,加载所需时间较长
  • 页面按需加载,整个项目只有其中某几个页面需要用到引入的第三方脚本
  • 第三方脚本没加载完就渲染页面导致的页面报错

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
jQuery动态创建html元素的常用方法汇总
Sep 05 Javascript
javascript省市级联功能实现方法实例详解
Oct 20 Javascript
js事件驱动机制 浏览器兼容处理方法
Jul 23 Javascript
对Angular.js Controller如何进行单元测试
Oct 25 Javascript
Node.js用readline模块实现输入输出
Dec 16 Javascript
HTML5+JS+JQuery+ECharts实现异步加载问题
Dec 16 jQuery
VUE基于NUXT的SSR 服务端渲染
Nov 30 Javascript
VueJS 组件参数名命名与组件属性转化问题
Dec 03 Javascript
使用webpack搭建vue项目实现脚手架功能
Mar 15 Javascript
js/jQuery实现全选效果
Jun 17 jQuery
通过高德地图API获得某条道路上的所有坐标用于描绘道路的方法
Aug 24 Javascript
vue中的计算属性和侦听属性
Nov 06 Javascript
JQuery Ajax执行跨域请求数据的解决方案
Dec 10 #jQuery
发布Angular应用至生产环境的方法
Dec 10 #Javascript
webpack优化的深入理解
Dec 10 #Javascript
BootStrap模态框闪退问题实例代码详解
Dec 10 #Javascript
详解为生产环境编译Angular2应用的方法
Dec 10 #Javascript
深入理解Vue.js轻量高效的前端组件化方案
Dec 10 #Javascript
es6基础学习之解构赋值
Dec 10 #Javascript
You might like
PHP extract 将数组拆分成多个变量的函数
2010/06/30 PHP
Linux下编译redis和phpredis的方法
2016/04/07 PHP
使用Codeigniter重写insert的方法(推荐)
2017/03/23 PHP
使用jscript实现二进制读写脚本代码
2008/06/09 Javascript
javascript 在网页中的运用(asp.net)
2009/11/23 Javascript
获取offsetTop和offsetLeft值的js代码(兼容)
2013/04/16 Javascript
jQuery插件开发的五种形态小结
2015/03/04 Javascript
javascript中去除数组重复元素的实现方法【实例】
2016/04/12 Javascript
Javascript中document.referrer隐藏来源的方法
2017/01/16 Javascript
详解angular笔记路由之angular-router
2017/09/12 Javascript
jQuery实现表单动态添加数据并提交的方法
2018/07/19 jQuery
优雅的elementUI table单元格可编辑实现方法详解
2018/12/23 Javascript
浅谈KOA2 Restful方式路由初探
2019/03/14 Javascript
Vue 实现v-for循环的时候更改 class的样式名称
2020/07/17 Javascript
Python获取Linux系统下的本机IP地址代码分享
2014/11/07 Python
Linux系统上Nginx+Python的web.py与Django框架环境
2015/12/25 Python
利用ctypes提高Python的执行速度
2016/09/09 Python
python中文件变化监控示例(watchdog)
2017/10/16 Python
详解如何设置Python环境变量?
2019/05/13 Python
python  文件的基本操作 菜中菜功能的实例代码
2019/07/17 Python
Python Tkinter模块 GUI 可视化实例
2019/11/20 Python
CSS3 box-shadow属性实例详解
2020/06/19 HTML / CSS
台湾团购、宅配和优惠券:17Life
2017/08/14 全球购物
COS美国官网:知名服装品牌
2019/04/08 全球购物
英国最大的宠物商店:Pets at Home
2019/04/17 全球购物
英国最大的天然和有机产品在线零售商之一:Big Green Smile
2020/05/06 全球购物
生产文员岗位职责
2014/04/05 职场文书
建筑院校毕业生求职信
2014/06/13 职场文书
个人授权委托书范本
2014/09/14 职场文书
个人批评与自我批评总结
2014/10/17 职场文书
工厂见习报告范文
2014/10/31 职场文书
中秋节慰问信
2015/02/15 职场文书
涨价通知
2015/04/23 职场文书
CentOS8.4安装Redis6.2.6的详细过程
2021/11/20 Redis
配置Kubernetes外网访问集群
2022/03/31 Servers
天谕手游15杯全调酒配方和调酒券的获得方式
2022/04/06 其他游戏