Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 583|回复: 0

解决Vue引入百度地图JSSDK:BMap is undefined 问题

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-4-22 13:51:24 | 显示全部楼层 |阅读模式

    百度地图官网文档介绍使用JSSDK时,仅提供了2种引入方式:

    • script引入
    • 异步加载
    解决跨域问题,实例调用百度地图

    但vue项目中仅某一两个页面需要用到百度地图,所以不想在 index.html 中全局引用。

    那在单个vue组件页面中如何引入呢?

    刚开始时,是直接通过 DOM 操作方式插入script标签到当前document中,如下:

    
    let scriptNode = document.createElement("script");
    scriptNode.setAttribute("type", "text/javascript");
    scriptNode.setAttribute("src", "http://api.map.baidu.com/api?v=3.0&ak=您的密钥");
    document.body.appendChild(scriptNode);
    

    结果是不行的。

    然后考虑使用异步加载的方式,结合参考网上方案,单独创建baidu-map.js脚本:

    
    export default {
      init: function (){
        const AK = "AK密钥";
        const apiVersion = "3.0";
        const timestamp = new Date().getTime();
        const BMap_URL = "http://api.map.baidu.com/api?v="+ apiVersion +"&ak="+ AK +"&services=&t=" + timestamp;
        return new Promise((resolve, reject) => {
          // 插入script脚本
          let scriptNode = document.createElement("script");
          scriptNode.setAttribute("type", "text/javascript");
          scriptNode.setAttribute("src", BMap_URL);
          document.body.appendChild(scriptNode);
    
          // 等待页面加载完毕回调
          window.onload = function () {  
             resolve(BMap)  
           } 
        });
      }
    }
    
    // -------------------------
    // vue引入调用
    import BaiduMap from 'baidu-map';
    
    ...
    mounted(){
        BauduMap.init()
        .then((BMap) => {
            console.log(BMap)
            console.log("加载成功...")
        })
    }
    ...  
    

    结果还是不行。

    想了下原因,一、可能是vue中window.onload没有触发,二、百度地图JSSDK没有真正加载成功。

    继续验证测试,发现window.onload能够正常触发,那就是JSSDK没有加载成功。

    直接复制JSSDK URL浏览器中打开 http://api.map.baidu.com/api?v=3.0&ak=您的密钥关键点来了,打开后内容如下:

    
    (function(){ 
    window.BMap_loadScriptTime = (new Date).getTime(); 
    document.write('<script type="text/javascript" src="http://api.map.baidu.com/getscript?v=3.0&ak=您的密钥&services=&t=20180102163224"></script>');
    })();
    

    从返回内容中看出,立即执行函数中再次插入了另外一个<scirpt>标签,经检查发现这个<scirpt>实际并没有插入成功。

    既然如此,那就直接把脚本放到我们上面的代码中去加载,结果就真的成功了。

    修改优化后的代码如下:

    
    export default {
      init: function (){
        console.log("初始化百度地图脚本...");
        const AK = "AK密钥";
        const apiVersion = "3.0";
        const timestamp = new Date().getTime();
        const BMap_URL = "http://api.map.baidu.com/getscript?v="+ apiVersion +"&amp;ak="+ AK +"&amp;services=&amp;t=" + timestamp;
        return new Promise((resolve, reject) =&gt; {
          if(typeof BMap !== "undefined") {
            resolve(BMap);
            return true;
          }
    
          // 插入script脚本
          let scriptNode = document.createElement("script");
          scriptNode.setAttribute("type", "text/javascript");
          scriptNode.setAttribute("src", BMap_URL);
          document.body.appendChild(scriptNode);
    
          // 等待页面加载完毕回调
          let timeout = 0;
          let interval = setInterval(() =&gt; {
            // 超时10秒加载失败
            if(timeout &gt;= 20) {
              reject();
              clearInterval(interval);
              console.error("百度地图脚本初始化失败...");
            }
            // 加载成功
            if(typeof BMap !== "undefined") {
              resolve(BMap);
              clearInterval(interval);
              console.log("百度地图脚本初始化成功...");
            }
            timeout += 1;
          }, 500);
        });
      }
    }  
    

    问题到此就解决了,至于为什么用官网提供的地址没有真正加载到JSSDK这个问题有空再研究下。


    最新解决方案

    
    export default {
      init: function (){
        //console.log("初始化百度地图脚本...");
        const AK = "AK密钥";
        const BMap_URL = "https://api.map.baidu.com/api?v=2.0&amp;ak="+ AK +"&amp;s=1&amp;callback=onBMapCallback";
        return new Promise((resolve, reject) =&gt; {
          // 如果已加载直接返回
          if(typeof BMap !== "undefined") {
            resolve(BMap);
            return true;
          }
          // 百度地图异步加载回调处理
          window.onBMapCallback = function () {
            console.log("百度地图脚本初始化成功...");
            resolve(BMap);
          };
    
          // 插入script脚本
          let scriptNode = document.createElement("script");
          scriptNode.setAttribute("type", "text/javascript");
          scriptNode.setAttribute("src", BMap_URL);
          document.body.appendChild(scriptNode);
        });
      }
    }  
    

    优化如下:

    • 直接使用官网提供的引用地址:http://api.map.baidu.com/api?v=2.0&ak=您的密钥
    • 启用 callback 参数,异步加载必须使用此参数才可以生效
    • 启用 https 配置,通过 s=1 参数实现
    • API版本为2.0,经测试使用,发现3.0版本在HTTPS环境下是有问题的,脚本内部某些请求固定使用HTTP,无法正常使用。

    原文地址:https://segmentfault.com/a/1190000012815739

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-1-24 10:43 , Processed in 0.060492 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表