当前位置: 首页 > news >正文

基于Leaflet调用天地图在线API的多层级地名检索实战

目录

前言

一、天地图在线检索

1、在线检索功能

2、再谈后后接口

二、Leaflet多层级实现实例

1、层级调用实现原理

2、Leaflet中多层级调用

3、成果展示

三、总结


前言

        “地图是世界的索引,而地名则是索引中的索引。”当互联网地图进入 Web 2.0 时代,开发者不再满足于“看得见”的瓦片,更希望“问得准”——输入一个模糊的地名,就能在毫秒之间锁定它在全球、全国、省市、区县乃至乡镇的多级位置。天地图作为国家地理信息公共服务平台的官方出口,提供了完全符合国标的行政区划与兴趣点(POI)检索能力;而轻量级的 Leaflet 则在前端以极简的 API 哲学,让地图可视化回归“纯粹的快乐”。当两者相遇,我们便有了“用 200 行代码完成一个国家级别地名搜索引擎”的可能。本文所记录的,正是这样一次“小而美”的实战:如何以 Leaflet 为壳、天地图为芯,逐级剖开中国行政体系的“套娃式”结构,把“模糊输入–多级联想–精准定位”做成一套可复用的前端组件。在WebGIS的可视化效果中,在省级节点中展示统计数量,随着地图的深入逐级进行细化,将地名逐渐呈现在大家的面前。

        在之前的博客中曾经对如何调用天地图的检索API进行了详细的介绍。但是如何进行详细层级展示没有进行详细的说明,因此有朋友留言表示希望可以对层级展示进行一个具体的说明。本文即在此背景下产生。

        阅读全文并运行示例后,你将掌握:

  1. 如何正确调用天地图全部检索接口;

  2. 如何为 Leaflet 编写一个“可复用”的入口,支持调用天地图的地名检索接口;

  3. 如何在Javascript中处理层级展示问题;

  4. 如何实现下钻的行政区检索。

一、天地图在线检索

        本节将详细的介绍天地图的在线检索功能,分别介绍天地图的搜索功能和后端接口。通过参考学习天地图的地图展现形式,为我们后面的技术和页面实现提供参考。

1、在线检索功能

        为了展示天地图的多层级展示功能,首先我们来看一下其官方网站的实例。大家可以访问天地图后,点击“在线”地图的tab标签页,从而跳转到在线地图,如下图所示:

         在其页面的左上角就有一个输入框,在这个输入框中输入相应的信息就可以完成信息的检索。比如在输入框中输入“自然”,点击检索,在界面上就可以看到以下的展示:

        可以看到包含了自然的地名最多的省级行政区划是浙江省,共有599616条记录。第二名是广东省,约有3307条记录,第三名是江苏省,大约有2842条记录。

2、再谈后后接口

        为了照顾第一次看博文的朋友,这里将天地图的检索接口API给大家详细的介绍一下,让大家了解后台的数据接口和服务,在下一节的多层级调用展示过程实现提供技术基础。

        输入参数说明

参数值参数说明参数类型是否必备备注(值域)
keyWord搜索的关键字String必填
specify指定行政区的国标码(行政区划编码表)严格按照行政区划编码表中的(名称,gb码)String必填下载行政区划编码表。9位国标码,如:北京:156110000或北京。
queryType服务查询类型参数String必填12:行政区划区域搜索服务。
start返回结果起始位(用于分页和缓存)默认0String必填0-300,表示返回结果的起始位置。
count返回的结果数量(用于分页和缓存)String必填1-300,返回结果的条数。
dataTypes数据分类(分类编码表)String可选下载分类编码表,参数可以分类名称或分类编码。多个分类用","隔开(英文逗号)。
show返回poi结果信息类别String可选取值为1,则返回基本poi信息; 取值为2,则返回详细poi信息

        返回参数说明

参数值参数说明参数类型返回条件备注(值域)
resultType返回结果类型Int必返回取值1-5,对应不同的响应类型: 1(普通POI),2(统计),3(行政区),4(建议词搜索),5(线路结果)
count返回总条数Int必返回
keyword搜索关键词String必返回搜索的关键字。
pois针对点(类型1)集合返回Pois Json数组resultType=1
namePoi点名称String必返回
phone电话String
address地址String
lonlat坐标String必返回坐标 x,y
poiTypepoi类型Int必返回101:POI数据 102:公交站点
eaddress英文地址String
enamepoi点英文名称String
hotPointIDpoi热点IDString必返回热点id
province所属省名称String
provinceCode省行政区编码String
city所属城市名称String
cityCode市行政区编码String
county所属区县名称String
countyCode区县行政区编码String
source数据信息来源String必返回
typeCode分类编码String
typeName分类名称String
stationData车站信息结构体 数据Json 数组poiType=102
lineName线路名称String必返回
uuid线路的idString必返回
stationUuid公交站uuidString必返回
statistics针对统计(类型2)集合返回Json 数组resultType=2
count本次统计POI总数量Int必返回
adminCount行政区数量Int必返回
priorityCitys推荐行政区名称Json数组必返回
name行政区名称String必返回
count城市数量Int必返回
lonlat行政区经纬度String必返回坐标 x,y
ename英文行政名称String必返回
adminCode城市国标码Int必返回9位国标码。
allAdmins各省包含信息集合Json数组必返回
name行政名称String必返回
count包含数量Int必返回
lonlat行政区经纬度String必返回坐标x,y
adminCode省国标码String必返回
ename英文行政名称String必返回
isleaf有无下一级行政区boolean必返回有则false,无则true

        在官网的调用示例中,使用浏览器的调试工具来看一下数据返回的结果:

         请特别注意以上的接口返回值,这里取其中的一个值对象信息如下:

{"adminName": "浙江省","ename": "Zhejiang Province","count": 599616,"adminCode": 156330000,"isleaf": false,"lonlat": "120.06772699999999,29.174674999999997"
}

        这里返回的adminCode参数非常重要,在后面的层级调用中其实就是依赖这个adminCode来进行多层级实现的。

二、Leaflet多层级实现实例

        本节将具体详述如何使用Leaflet来调用天地图的检索API和实现多层级调用,可以在Leafelt中实现我们自己的逻辑,可以展示省-市-POI详情等信息,最后给大家展示一些实际的效果。

1、层级调用实现原理

        关于如何实现层级调用,在前面一节中也进行了简单说明。其实就是一个递归的调用,在进行API调用时,首先需要判断返回的数据类型是什么?如果从接口中返回的类型是1表示是poi数据,直接展示即可,如果是2表示是展示统计数据,再从统计数据中获取所有的信息,循环统计信息中的行政区划信息,然后再递归调用其行政区划的编码实现向下检索。实现的核心代码展示如下:

/**
* 调用天地图查询
*/
function callTdtSearch(keyWord,specify){var queryUrl = "http://api.tianditu.gov.cn/v2/search?postStr={'keyWord':'"+ keyWord +"','queryType':12,'start':0,'count':10,'specify':'" + specify + "','show':'2'}&type=query&tk="+tdt_client_key;$.ajax({type: "get",url:queryUrl,data: {},success: function(rsData) {// 移除所有图层myLayerGroup.clearLayers();var rsObj = rsData; var loc_info = rsObj.location;var resultType = rsObj.resultType;switch(resultType){case 1 :showPoi(rsObj);break;case 2:showStatistics(rsObj);break;default:console.log("不详");}map.addLayer(myLayerGroup);	}});
}

2、Leaflet中多层级调用

        根据接口中返回的类型不一样来进行个性化的展示,以此达到区别展示的效果。这里重点介绍如何来进行统计信息的展示和如何调用下一层级的POI信息,核心方法如下:

//点击还可以进行查询
function buildStatHtml(stat,index){var html = "";html += "<div class='marsBlackPanel' style='background:#ff9800;' animation-spaceInDown>";html += "<div class='marsBlackPanel-text' style='' onclick='execQueryByCode("+stat.adminCode+")'>" + (index +1) + "、" +stat.adminName + "(" + stat.count + ")</div>";html += "</div>";return html;
}
function showStatistics(rsObj){var statistics = rsObj.statistics.allAdmins;for(var i = 0;i<statistics.length;i++){var stat = statistics[i];var lonlat = stat.lonlat;var lonlatStr = lonlat.split(",");var marker = L.marker([lonlatStr[1], lonlatStr[0]], {icon: L.divIcon({iconSize: null,className: '',popupAnchor:[5,5],shadowAnchor:[5,5],html: buildStatHtml(stat,i)})}).addTo(myLayerGroup);}map.fitBounds(myLayerGroup.getBounds());
}

        这里实现的关键就是在自定义的标注中,绑定了一个执行查询的方法,然后再这个执行方法中又可以进行下一个层级的调用。 关键的方法如下:

function execQueryByCode(specify){var keyWord = $("#address").val();callTdtSearch(keyWord,specify);
}

        这样就实现了层次的调用,当返回的值是具体的POI时,直接展示。反之则会进行省份的标注,同时可以点击当前省份,从而实现多层级的调用和展示。

3、成果展示

        下面结合一些实际的例子来展示一下成果,具体是如何来进行相关的实现的。

        在Web浏览器中输入请求地址,可以看到以上的界面效果。这样就基本模拟省份这个层级的统计展示。 需要说明的是,通过POI检索API的数据结果与官网的检索结果之前还是有一丢丢的区别,比如官网返回的结果是599616,而通过我们的接口调用,返回的结果是:596756;下面将结果整理一个表示,供大家参考,具体出现差异的原因可能需要咨询客服人员:

序号省份官方返回结果自主调用结果差额
1浙江省5996165967562860
2广东省33073103204
3江苏省28422519323
4江西省22711918353
5湖南省22301895335

        由此可以看出,搜索的结果还是存在一定的区别的。

        下面来看一个有意思的地方,包含舒服两个字的最多的省份是湖北省。

        而在湖北省中,武汉又是最多的,有41个地方包含舒服,

        来看看具体包含舒服的地名都在武汉的哪些地方,

         最后来看看天地图中包含火锅的最多的省份是哪里?

        可以看到,出现火锅最多的地方居然是江苏省,在大家的印象里,江浙应该都是甜口居多。而喜好火锅的云贵川桔田前三都挤不进去,第四名为重庆,第六名四川。前三中,除江苏外,第二名是山东省,第三名是河南省,有没有出乎你的意料呢?

        在江苏省中,苏州市的火锅又是地第一名的,有2028个, 如下:

        确实基本上都是火锅了,这个确实没有想到,大苏州的火锅居然这么多。 

三、总结

        以上就是本文的主要内容,本文所记录的,正是这样一次“小而美”的实战:如何以 Leaflet 为壳、天地图为芯,逐级剖开中国行政体系的“套娃式”结构,把“模糊输入–多级联想–精准定位”做成一套可复用的前端组件。在WebGIS的可视化效果中,在省级节点中展示统计数量,随着地图的深入逐级进行细化,将地名逐渐呈现在大家的面前。本文不仅详细的介绍了如何在Leaflet中进行层级展示的实现,并且基于实际的地名进行了实际的检索展示,如果大家感兴趣,不妨来这里看看。行文仓促,定有不足之处,欢迎各位朋友在评论区批评指正,不胜感激。

http://www.xdnf.cn/news/1116793.html

相关文章:

  • 深度学习16(对抗生成网络:GAN+自动编码器)
  • 跨网络连接不同机器上的虚拟机
  • UNet改进(22):融合CNN与Transformer的医学图像分割新架构
  • 15. JVM调优的参数设置
  • [Linux 入门] Linux 引导过程、系统管理与故障处理全解析
  • word设置多级标题
  • Cursor的使用
  • docker容器高级管理-dockerfile创建镜像
  • 树莓派5-ollama-linux-arm64.tgz 下载
  • OkHttp SSE 完整总结(最终版)
  • cuda编程笔记(7)--多GPU上的CUDA
  • 敦煌藻井配色:姜黄×钴蓝的东方色彩应用手册
  • CVE-2022-0609
  • 用信号量实现进程互斥,进程同步,进程前驱关系(操作系统os)
  • hercules zos 安裝 jdk 8
  • CTFSHOW pwn161 WP
  • 整流电路Multisim电路仿真实验汇总——硬件工程师笔记
  • 使用macvlan实现容器的跨主机通信
  • KL散度:信息差异的量化标尺 | 从概率分布对齐到模型优化的核心度量
  • C++高频知识点(十一)
  • ALB、NLB、CLB 负载均衡深度剖析
  • 开源工具DeepFilterNet:实时语音降噪
  • 更换docker工作目录
  • 06.计算两个日期之间的差值
  • lambdastream深入剖析
  • 【LeetCode100】--- 4.移动零【复习回顾】
  • mmap映射文件
  • 理解 Robots 协议:爬虫该遵守的“游戏规则”
  • HTML 标题标签
  • AI驱动的软件工程(上):人机协同的设计与建模