3D GIS 应用开发 —— 基于 Mapbox GL 的实践总结
最近在折腾的 web 端的可视化项目,由于相关业务的需要,用到了 Mapbox 这一地图开发的神器。在此先奉上一个基于mapbox-gl实现的demo(来源:uber的deck.gl项目):
下面我们从这个项目一步步来介绍 Mapbox 的前端 GIS 引擎 Mapbox GL JS.
一、简单了解
首先,Mapbox 在地图领域是一家很 Newbee 的公司。已为 Foursquare、Pinterest、Evernote、金融时报、天气频道、优步科技 等公司的网站提供了订制在线地图服务。
自2010年起,该公司快速地拓展了订制地图的市场地位,以回应 Google地图 等地图供应商提供的有限选择。Mapbox 是一些开放源代码地图库及应用程序的创建者或最大的贡献者,其中包含了MBTiles 规范、TileMill 制图 IDE、Leaflet JavaScript 库,以及 CartoCSS 地图格式化语言与语法分析器等。
该公司的数据同时从开放与专有的来源获取,开放的数据源如 开放街图(OSM, Open Street Map) 以及 NASA 等,而专有的数据源则包含了 DigitalGlobe。其技术奠基于 Node.js、CouchDB、Mapnik、GDAL 与 Leafletjs。
Mapbox 针对不同平台均开发了相应的 GIS 引擎以满足开发者或相关用户的需要,如:iOS SDK(用于iOS端开发)、Android SDK(用于Andriod端开发)、Navigation SDK(用于Navigation端开发)、Unity SDK(用于Unity端开发)、GL JS(用于web端开发)。不同平台的SDK,除使用方式不同外,功能特性上也多多少少存在不同。此外,Uber还针对react开发了 react-map-gl。总的来说,Mapbox的开源技术栈是非常全面的。
二、轻松上手
mapbox-gl 的 文档 由 API、Style Specification、Example、Plugins 四部分内容组成。
顾名思义,API 是一般框架(类库)提供给用户的全部接口(方法)的说明书;Style Specification 是 Mapbox 地图的样式规范;Example 是一些常用功能或常见业务的代码示例,囊括了使用 Mapbox 所能实现的大部分功能效果;Plugins 则是官方推荐的可与 mapbox-gl 一同使用的一些增效插件和开源项目,如一些第三方的UI控件、显示类插件、框架集成工具、开发辅助工具、实用工具类库等等。
对于初了解 Mapbox 的童鞋,建议先从官网的 Example 入手,能够较快掌握 mapbox-gl 的使用并投入开发实践。
三、快速实践
下面以文章开头展示的项目为主,介绍其实战步骤。
1. 加载地图:
由于使用在线地图服务和 style 时需要验证用户 token,所以在使用 mapboxgl 时需要先配置用户 token(在Mapbox官网注册用户即可获取)。
import mapboxgl from 'mapbox-gl';
mapboxgl.accessToken = '<Your Token Here>';
接下来使用创建地图实例。主要配置项如下:
const myMap = new mapboxgl.Map({
container: '<Id of Container Element>',
style: '<Your Style Here>',
center: [112.508203125, 37.97980872872457],
zoom: 4,
pitch: 0,
bearing: 0,
});
其中,container 是地图容器的元素 id,style 是地图样式的 url,或者你自己定义的 style(需遵循Mapbox样式规范),center 是地图加载后默认的中心点位置,用以定位地图加载时的位置。zoom pitch bearing 分别指缩放级别、地面法线偏移角、地轴偏移角等,用以确定当前视窗所显示的地图区域和空间关系。配置项的意义均可查看官网文档。
2. 绘制图形
这里主要介绍视频中的3D建筑、飞线动画等是如何实现的。这里以相关代码片段来介绍实践的方法。
在Mapbox中绘制图形时, layer 和 source 是最重要的一组概念,后者用于存储图形的数据内容,前者则是图形在3D场景中的表现(图层)。在Mapbox中,图层一旦被创建,与其同名(id相同)的数据源源(即source)也必然被创建。反之,也可以在创建source后再创建一个图层使用这个已创建的数据源,这时数据源与图层间并不要求同名。而我们通过改变数据来驱动图形变化,便是才去的第二种方式:
// 创建id为buildings的数据源
myMap.addSource('buildings', {
type: 'geojson',
data: '<GeoJson Contents>',
});
// 使用buildings的数据来绘制id为building_layer的图形
myMap.addLayer({
id: 'building_layer',
type: 'fill-extrusion',
source: 'buildings',
...<Other Options>,
});
基于上面的方式,当数据改变时,我们只需要重设数据源的数据,即可驱动图层重绘:
if (myMap.getLayer('building_layer')) {
myMap.getSource('buildings').setData(<New GeoJson Contents>);
}
至于3D效果及动画的具体实现,这里给出两个官网上的示例,相信大家能一目了然:
i. 用3D形式呈现建筑
ii. 给路径中的一个点添加动画效果
3. 图形交互
Mapbox提供的交互方法是比较灵活的,活学活用API文档便能实现各种炫酷、实用的交互效果。比如:使用myMap.on('zoom', callback) 可以将图形与地图的缩放相绑定,当缩放系数小于某个值时,可以隐藏掉一些图形元素:
myMap.on('zoom', () => {
if (myMap.getZoom() <= 4) {
myMap.setLayoutProperty('building_layer', 'visibility', 'none');
} else {
myMap.setLayoutProperty('building_layer', 'visibility', 'visible');
}
});
再比如,连续调用 myMap.flyTo() 的方法使视图在地图上按照一定的轨迹缓慢移动,可以给用户一种模拟飞行的体验。视频中的自动巡视的效果正是这样实现的。
诸如 click mouseover popup 等效果,官网文档中的示例已经具体呈现,这里就不详细展开了。
4. tiles-server的本地化
由于 Mapbox 地图服务使用 MBTiles 存储数据,目前很多地图服务都接受了这套标准(如:OSM,Open Street Map)。所以可以通过搭建自己的 tiles-server 以替代直接使用 Mapbox 的在线地图服务。
这样做的好处是显而易见的:一是可以通过负载均衡等手段提高数据接口的访问速度,有效提高数据的加载速度;一是保障应用能在零带宽的环境下仍能有效部署和使用。
这里墙裂安利一个docker开源镜像:openmaptiles-server ,在其 官网 和 dockerhub 上均可下载。个人认为其最大的亮点在于——即使不了解内部实现,也不影响其使用。
运行 tiles-server 服务的 docker 命令如下:
$ docker run --rm -it -v $(pwd):/data -p 8080:80
然后剩下来需要做的事情就是打开其导航页面 http://localhost:8080/(端口号取决于你的启动命令),然后跟着页面上的提示一步一步设置就好了(最后一步设置后会从OSM走动下载地图,所以一开始你不用担心数据从哪来),完全是傻瓜式的部署。
四、性能调优
在 Mapbox GL 实践的过程中,发现了一些影响应用整体性能的因素,故而在此陈述一番,为之后填坑的童鞋提供一些经验:
- 使用geo数据(如 GeoJson 格式数据)来定义图形的时候,若数据量过大,则会拖慢数据加载的速度,此时可考虑:
i. 在 http 请求前后对数据进行合理的压缩和解压,以尽可能节省 http 请求传输的数据量;
ii. 条件允许的情况下,可将一组数据分片加载,以空间换时间。
- 在 Mapboox 中绘制的图层不宜过多,一是不方便管理(当然,github上有很多管理Mapbox图层的第三方工具),一是图层过多会明显降低GL的渲染和响应性能。所以在绘制图形前,可以先考虑一下图层的划分,以最少的图层实现尽可能多的效果。
- 数据量相同的情况下,使用
mapboxgl.Marker来添加标记,其性能不如使用type为symbol类型的图层来添加标记。原因在于前者生成的标记是一个个 DOM 元素,如果你可以想象在一个 web 页面中同时操作成百上千个 DOM 节点会是什么结果,那么你或许能明白我的建议。
五、一点总结
最后,在此总结下个人对 Mapbox 的一些感观。
Mapbox 的产品定位是随时随地的 GIS(跨平台、应用),它为我们提供了一系列的简单操作的 API,使得 GIS 开发变得灵活而有趣。尤其对于开发 GIS 类型的数据可视化应用,Mapbox 是绝佳的选择。
然而,如果你只是为了那些绚丽的 3D 效果的话,或许选择专门的框架更为合适。
3D GIS 应用开发 —— 基于 Mapbox GL 的实践总结的更多相关文章
- 利用WPF建立自己的3d gis软件(非axhost方式)(十)SDK中一些自带的展示面板应用
原文:利用WPF建立自己的3d gis软件(非axhost方式)(十)SDK中一些自带的展示面板应用 先下载SDK:https://pan.baidu.com/s/1M9kBS6ouUwLfrt0zV ...
- mapbox.gl源码解析——基本架构与数据渲染流程
加载地图 Mapbox GL JS是一个JavaScript库,使用WebGL渲染交互式矢量瓦片地图和栅格瓦片地图.WebGL渲染意味着高性能,MapboxGL能够渲染大量的地图要素,拥有流畅的交互以 ...
- 利用WPF建立自己的3d gis软件(非axhost方式)(十三)万能的用户层接口,(强大的WPF)
原文:利用WPF建立自己的3d gis软件(非axhost方式)(十三)万能的用户层接口,(强大的WPF) 先下载SDK:https://pan.baidu.com/s/1M9kBS6ouUwLfrt ...
- 利用WPF建立自己的3d gis软件(非axhost方式)(十一)SDK中的动画系统
原文:利用WPF建立自己的3d gis软件(非axhost方式)(十一)SDK中的动画系统 先下载SDK:https://pan.baidu.com/s/1M9kBS6ouUwLfrt0zV0bPew ...
- [Intel Edison开发板] 05、Edison开发基于MRAA实现IO控制,特别是UART通信
一.前言 下面是本系列文章的前几篇: [Intel Edison开发板] 01.Edison开发板性能简述 [Intel Edison开发板] 02.Edison开发板入门 [Intel Edison ...
- {VS2010C#}{WinForm}{ActiveX}VS2010C#开发基于WinForm的ActiveX控件
在VS2010中使用C#开发基于WinForm的ActiveX控件 常见的一些ActiveX大部分是使用VB.Delphi.C++开发,使用C#开发ActiveX要解决下面三个问题: 使.NET组件可 ...
- Form_Form Builder开发基于视图页面和自动代码生成包(案例)
2014-01-06 Created By BaoXinjian
- 转】Mahout分步式程序开发 基于物品的协同过滤ItemCF
原博文出自于: http://blog.fens.me/hadoop-mahout-mapreduce-itemcf/ 感谢! Posted: Oct 14, 2013 Tags: Hadoopite ...
- Masstransit开发基于消息传递的分布式应用
使用Masstransit开发基于消息传递的分布式应用 Masstransit作为.Net平台下的一款优秀的开源产品却没有得到应有的关注,这段时间有机会阅读了Masstransit的源码,我觉得我有必 ...
随机推荐
- 创建一个 Spring Boot 项目,你会几种方法?
我最早是 2016 年底开始写 Spring Boot 相关的博客,当时使用的版本还是 1.4.x ,文章发表在 CSDN 上,阅读量最大的一篇有 42W+,如下图: 2017 年由于种种原因,就没有 ...
- docker(4)docker的网络,自定义网桥
Docker 的网络 运行 ifconfig 找到 docker0 : 虚拟网卡默认网卡名称为docker0 查看docker 的网桥: 我这里默认们没有进行安装 网桥管理设备:进行安装一下: yum ...
- 【开发记录】如何在B/S项目中使用中国天气的实时天气功能
好久没有更新我的博客了,正好手头有一个比较合适的项目经验可以分享出来,就是这个如何使用中国天气的天气预报功能,也正好做个项目经验记录. 功能需求 这个功能需求比较简单,就是想在网页端显示实时天气数据. ...
- 前端笔记之HTML
前端三层:内容层(结构层)HTML.样式层(表现层)CSS.行为层JavaScript 层 语言 含义 结构层 HTML 由 HTML 或 XHTML之类的标记语言负责创建.标签,也就是那些出现在尖括 ...
- 联发科MT8788基带处理器介绍
MT8788设备具有集成的蓝牙.fm.wlan和gps模块,是一个高度集成的基带平台,包括调制解调器和应用处理子系统,启用LTE/LTE-A和C2K智能设备应用程序.该芯片集成了工作在2.0GHz的A ...
- Linux下使用ntpdate进行时间同步
转摘自Linux下使用ntpdate进行时间同步https://www.cnblogs.com/zhi-leaf/p/6281549.html1.安装ntpdate,执行以下命令 # yum inst ...
- Kubernetes初探[1]:部署你的第一个ASP.NET Core应用到k8s集群
Kubernetes简介 Kubernetes是Google基于Borg开源的容器编排调度引擎,作为CNCF(Cloud Native Computing Foundation)最重要的组件之一,它的 ...
- 基于开源CA系统ejbca community 6.3.1.1构建私有CA管理数字证书
最后更新于2017年01月24日 一.为什么 为什么写这篇文章?ca是什么?数字证书是什么?ejbca又是什么? 让我们从http与https说起.http是超文本传输协议(HyperText Tra ...
- POST不同提交方式对应的Content-Type,及java服务器接收参数方式
POST不同提交方式对应的Content-Type,及java服务器接收参数方式 注:本博客参考了网上的文章结合自己工作总结后所写,主要用于记录自己工作所得,如有错误请批评指正. 简介: Conten ...
- Java Excel导入导出(实战)
一.批量导入(将excel文件转成list) 1. 前台代码逻辑 1)首先在html页面加入下面的代码(可以忽略界面的样式) <label for="uploadFile" ...