1. 概述

AntV L7 是蚂蚁集团 AntV 数据可视化团队推出的基于 WebGL 的开源大规模地理空间数据可视分析引擎,其特点是通过简单的代码进行配置,即可在前端网页中绘制精美的地图以及相关的图表,并且基于 WebGL 的渲染方式使得 L7 在大数据渲染时具有较为流畅体验

图1. AntV L7 官方图表截图

提到地理数据,就不得不说 GIS(Geographic Information System,地理理信息系统),它是一种用于捕捉、存储、管理、分析和展示地理空间数据的技术,通过 GIS ,用户可以将地理空间数据与属性数据相结合,进行复杂的空间分析、制作地图和展示地理信息,GIS 的核心包括地理数据的收集和整合、空间分析和地图制作

L7 专注于地理数据可视化,即地图制作,如果能有一个前端空间分析工具与 L7 结合,丰富 GIS 的核心,会极大地利好前端开发者和用户

Turf.js 就是一个经典地空间分析库,提供了许多用于处理地理空间数据的函数和算法,其基于JavaScript编写,可以用于浏览器端和Node.js环境

图2. Turf.js 的官网介绍

所以,将 Turf.js 与 L7 结合,就可以实现在前端进行空间分析与空间可视化,拓展 L7 的使用方向

以下章节,笔者首先记述 Turf.js 与 L7 的快速入门使用,然后记录一些实践案例

(笔者注:为了代码易于复现与使用,本文采用原始HTML的方式编写代码)

2. L7 的快速入门

第一步,使用CDN加载 L7

<! --引入最新版的L7,笔者使用时为2.20.5-->
<script src = 'https://unpkg.com/@antv/l7'></script>
  • CDN 引用的方式,是在使用时通过 L7 命名空间获取所有对象并初始化,如 L7.scene、L7.GaodeMap

第二步,创建一个 DIV 容器并设置CSS样式

<div id="map"></div>

第三步,加载高德底图

<script>
const scene = new L7.Scene({
id: 'map',
map: new L7.GaodeMap({
center: [116.3956, 39.9392],
zoom: 10,
mapStyle: 'amap://styles/darkblue'
})
});
</script>

此时的完整代码如下:

<!DOCTYPE html>
<html lang="en"> <head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<script src='https://unpkg.com/@antv/l7'></script>
<style>
body,
#map {
height: 100vh;
width: 100vw;
margin: 0;
}
</style>
</head> <body>
<div id="map"></div>
<script>
const scene = new L7.Scene({
id: 'map',
map: new L7.GaodeMap({
center: [116.3956, 39.9392],
zoom: 10,
mapStyle: 'amap://styles/darkblue'
})
});
</script>
</body> </html>

结果图如下:

图3. 加载初始底图

第四步,加载GeoJSON矢量数据并设置样式

  • L7 对于GeoJSON有很高的支持度,默认就支持GeoJSON数据
  • 数据是北京地铁线路
// 加载底图之后
scene.on('loaded', () => {
fetch(
'https://gw.alipayobjects.com/os/basement_prod/0d2f0113-f48b-4db9-8adc-a3937243d5a3.json'
)
.then(res => res.json())
.then(data => {
const layer = new L7.LineLayer({})
.source(data)
.size(4)
.shape('line')
.color('标准名称', [ '#5B8FF9', '#5CCEA1', '#F6BD16' ])
.style({
borderWidth: 0.4,
borderColor: '#fff'
});
scene.addLayer(layer);
});
});

此时的完整代码如下:

<!DOCTYPE html>
<html lang="en"> <head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<script src='https://unpkg.com/@antv/l7'></script>
<style>
body,
#map {
height: 100vh;
width: 100vw;
margin: 0;
}
</style>
</head> <body>
<div id="map"></div>
<script>
const scene = new L7.Scene({
id: 'map',
map: new L7.GaodeMap({
center: [116.3956, 39.9392],
zoom: 10,
mapStyle: 'amap://styles/darkblue'
})
});
scene.on('loaded', () => {
fetch(
'https://gw.alipayobjects.com/os/basement_prod/0d2f0113-f48b-4db9-8adc-a3937243d5a3.json'
)
.then(res => res.json())
.then(data => {
const layer = new L7.LineLayer({})
.source(data)
.size(4)
.shape('line')
.color('标准名称', ['#5B8FF9', '#5CCEA1', '#F6BD16'])
.style({
borderWidth: 0.4,
borderColor: '#fff'
});
scene.addLayer(layer);
});
}); </script>
</body> </html>

结果图如下:

图4. 加载GeoJSON数据

由上述代码不难看出,使用 L7 进行可视化,只需要使用极少的代码,就可配置成精美的地图图表

文章篇幅有限,更多的样式配置信息,更详尽的API文档,请翻阅 官方文档官方示例

3. Turf.js 的快速入门

第一步, 引入 Turf.js 的 CDN

<script src='https://unpkg.com/@turf/turf@6/turf.min.js'></script>
  • CDN 的方式,通过 turf 命名空间访问 Turf.js 的函数,如 turf.bbox 等

第二步,引入GeoJSON数据

  • 数据和上面一样,是北京地铁数据
<script>
fetch('https://gw.alipayobjects.com/os/basement_prod/0d2f0113-f48b-4db9-8adc-a3937243d5a3.json')
.then(res => res.json())
.then(data => {
console.log(data) // JSON对象
});
</script>

第三步,进行空间分析得到结果GeoJSON

  • 这里示例为求地理包围盒 BBOX
<script>
fetch('https://gw.alipayobjects.com/os/basement_prod/0d2f0113-f48b-4db9-8adc-a3937243d5a3.json')
.then(res => res.json())
.then(data => {
const bbox = turf.bbox(data);
const bboxPolygon = turf.bboxPolygon(bbox);
console.log(bboxPolygon);
// 下面是地理包围盒的坐标数据
// 0: 116.10214436813241
// 1: 39.6703694682177
// 2: 116.68907341874268
// 3: 40.20693349910422
});
</script>

此时的完整代码为:

<!DOCTYPE html>
<html lang="en"> <head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head> <body>
<script src='https://unpkg.com/@turf/turf@6/turf.min.js'></script>
<script>
fetch('https://gw.alipayobjects.com/os/basement_prod/0d2f0113-f48b-4db9-8adc-a3937243d5a3.json')
.then(res => res.json())
.then(data => {
const bbox = turf.bbox(data);
const bboxPolygon = turf.bboxPolygon(bbox);
console.log(bboxPolygon);
});
</script>
</body> </html>

上述 Turf.js 的代码可以看到,Turf.js 接收 GeoJSON 数据,输出 GeoJSON 结果数据

更多的空间分析函数与API可参考 Turf.js官方文档

Turf.js 专注于空间分析,L7 专注于数据可视化,并且两者对于 GeoJSON 格式有着最高优先度的支持,因此,可以将地理数据在 Turf.js 中进行处理,然后直接输出结果到 L7 中进行可视化,实现数据分析与数据可视化的连接

4. 案例一:寻找地铁站的换乘点

4.1 分析场景与数据

地铁换乘点,通常在不同地铁线路之间的交叉处,所以寻找换乘点,通常就是计算地铁线路的交叉处

如何计算交叉点呢?用空间分析的视角,就是进行求交运算

从 Turf.js 的官方文档中,可以找到计算线相交的函数lineIntersect

图5. lineIntersect 函数

上述的北京地铁线路数据示例如下:

{
"type": "FeatureCollection",
"crs": {
"type": "name",
"properties": {
"name": "urn:ogc:def:crs:OGC:1.3:CRS84"
}
},
"features": [
{
"type": "Feature",
"properties": {
"标准名称": "地铁二号线",
"分类代码": 430101,
"数据来源": "正射影像",
"现状时间": "2010/08/14",
"备注": null,
"SHAPE_LENG": 23177.0298819,
"Shape_Le_1": 23177.0298784,
"Shape_Le_2": 30241.8106532
},
"geometry": {
"type": "MultiLineString",
"coordinates": [
[
[
116.38050072430798,
39.94888011518406
],
[
116.38714780612922,
39.94892587302933
],
// ...

根据 Turf.js 的示例与文档,结合这份数据示例的类型为 FeatureCollection ,需要将里面的每个线路 MultiLineString 进行两两求交运算,最后将结果在 L7 中以点的形式绘制出来

4.2 进行空间分析

将 FeatureCollection 的每个 MultiLineString 进行两两求交运算得到交点的点集:

<script src='https://unpkg.com/@turf/turf@6/turf.min.js'></script>
<script>
fetch('https://gw.alipayobjects.com/os/basement_prod/0d2f0113-f48b-4db9-8adc-a3937243d5a3.json')
.then(res => res.json())
.then(data => {
const lines = data; // 新建一个Point类型的FeatureCollection用来存储结果
const points = turf.featureCollection([]); for (let i = 0; i < lines.features.length; i++) {
for (let j = i + 1; j < lines.features.length; j++) {
const intersections = turf.lineIntersect(lines.features[i], lines.features[j]);
if (intersections) {
if (intersections.features.length > 0) { // 将交点添加到points中
intersections.features.forEach(item => {
points.features.push(item);
});
}
}
}
}
console.log(points); // 得到四十个交点的JSON对象
});
</script>

4.3 数据可视化

使用 L7 绘制结果代码很简单:

const pointLayer = new L7.PointLayer({})
.source(points)
.shape('circle')
.size(6)
.color('#f00')
.style({
stroke: '#fff',
strokeWidth: 2
}); scene.addLayer(pointLayer);

最后,地铁路线与路线交点的完整代码如下:

<!DOCTYPE html>
<html lang="en"> <head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<script src='https://unpkg.com/@antv/l7'></script>
<script src='https://unpkg.com/@turf/turf@6/turf.min.js'></script>
<style>
body,
#map {
height: 100vh;
width: 100vw;
margin: 0;
}
</style>
</head> <body>
<div id="map"></div>
<script>
const scene = new L7.Scene({
id: 'map',
map: new L7.GaodeMap({
center: [116.3956, 39.9392],
zoom: 10,
mapStyle: 'amap://styles/darkblue'
})
});
scene.on('loaded', () => {
fetch(
'https://gw.alipayobjects.com/os/basement_prod/0d2f0113-f48b-4db9-8adc-a3937243d5a3.json'
)
.then(res => res.json())
.then(data => {
const layer = new L7.LineLayer({})
.source(data)
.size(4)
.shape('line')
.color('标准名称', ['#5B8FF9', '#5CCEA1', '#F6BD16'])
.style({
borderWidth: 0.4,
borderColor: '#fff'
});
scene.addLayer(layer); const lines = data; // 新建一个Point类型的FeatureCollection
const points = turf.featureCollection([]); for (let i = 0; i < lines.features.length; i++) {
for (let j = i + 1; j < lines.features.length; j++) {
const intersections = turf.lineIntersect(lines.features[i], lines.features[j]);
if (intersections) {
if (intersections.features.length > 0) { // 将交点添加到points中
intersections.features.forEach(item => {
points.features.push(item);
});
}
}
}
} const pointLayer = new L7.PointLayer({})
.source(points)
.shape('circle')
.size(6)
.color('#f00')
.style({
stroke: '#fff',
strokeWidth: 2
}); scene.addLayer(pointLayer); });
}); </script>
</body> </html>

结果图如下(红点为换乘点):

图6. 寻找地铁换乘点

5. 案例二:计算地铁线路的服务范围

5.1 分析场景与数据

居民乘坐地铁,通常是前往最近的地铁站,地铁线路的服务范围其实就是地铁线路附近的区域

如果求地铁线路附近的区域呢?从空间分析的角度来说,就是进行缓冲区分析

那地铁线路的服务半径一般是多少呢?ChatGPT提示笔者是1-2公里,笔者这里就取1.5公里作为缓冲半径

地铁线路数据还是使用上述的那份北京地铁线路数据

从 Turf.js 的官方文档中,可以找到计算缓冲区的函数buffer

图7. buffer函数

根据 Turf.js 的示例与文档,结合这份数据示例的类型为 FeatureCollection ,所以只需要将数据传入 buffer 函数即可得到结果

5.2 进行空间分析

调用 turf.buffer 计算缓冲区:

<script src='https://unpkg.com/@turf/turf@6/turf.min.js'></script>
<script>
fetch('https://gw.alipayobjects.com/os/basement_prod/0d2f0113-f48b-4db9-8adc-a3937243d5a3.json')
.then(res => res.json())
.then(data => {
const lines = data; const buffered = turf.buffer(data, 1500, { units: 'meters' }); console.log(buffered); // 缓冲区的JSON对象,Polygon
});
</script>

5.3 数据可视化

L7 进行可视化还是一如既往的简单

地铁路线与缓冲区分析完整代码如下:

<!DOCTYPE html>
<html lang="en"> <head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<script src='https://unpkg.com/@antv/l7'></script>
<script src='https://unpkg.com/@turf/turf@6/turf.min.js'></script>
<style>
body,
#map {
height: 100vh;
width: 100vw;
margin: 0;
}
</style>
</head> <body>
<div id="map"></div>
<script>
const scene = new L7.Scene({
id: 'map',
map: new L7.GaodeMap({
center: [116.3956, 39.9392],
zoom: 10,
mapStyle: 'amap://styles/darkblue'
})
});
scene.on('loaded', () => {
fetch(
'https://gw.alipayobjects.com/os/basement_prod/0d2f0113-f48b-4db9-8adc-a3937243d5a3.json'
)
.then(res => res.json())
.then(data => {
const layer = new L7.LineLayer({})
.source(data)
.size(4)
.shape('line')
.color('标准名称', ['#5B8FF9', '#5CCEA1', '#F6BD16'])
.style({
borderWidth: 0.4,
borderColor: '#fff'
}); const lines = data; const buffered = turf.buffer(data, 1500, { units: 'meters' }); const polygonLayer = new L7.PolygonLayer({})
.source(buffered)
.shape('fill')
.color('#fff')
.style({
opacity: 0.2
}); scene.addLayer(polygonLayer); scene.addLayer(layer);
});
}); </script>
</body> </html>

最后结果图如下:

图8. 计算地铁线路服务范围

6. 总结

经过上述两个小案例,可以说使用 L7 结合 Turf.js 进行简单的空间分析与可视化简直是易如反掌,尤其是对于前端开发人员来说,有时简单的空间分析功能在客户端完成即可,而不必进行后端开发

L7 具有强大的制图能力,在结合Turf.js后拥有了更广泛的使用场景

7. 参考资料

[1] L7·蚂蚁地理空间数据可视化 | AntV (antgroup.com)

[2] Turf.js | Advanced geospatial analysis (turfjs.org)

L7结合Turf.js实现空间分析与数据可视化的更多相关文章

  1. 前端使用d3.js调用地图api 进行数据可视化

    前段时间自己研究了demo就是把某个区域的某个位置通过经纬度在地图上可视化.其实就是使用了第三方插件,比现在比较火的可视化插件d3.js echart.js.大致思路就是,把要用到的位置的geojso ...

  2. BI报表分析和数据可视化,推荐这三个开源工具!

    开源篇 一.Superset 1.技术架构:Python + Flask + React + Redux + SQLAlchemy 2.使用人群: (1)开发/分析人员做好看板,业务人员浏览看板数据 ...

  3. JavaScript 空间分析库——JSTS和Turf【转】

    https://blog.csdn.net/neimeng0/article/details/80363468 前言 项目中有管线的空间拓扑关系查询需求,在npm中检索到JSTS和Turf两个Java ...

  4. 从零开始的全栈工程师——js篇2.7(JS数据类型具体分析)

    JS数据类型具体分析与数据的三大存储格式 1. 字符串 string2. 数字 number3. 布尔 boolean4. null 空5. undefined 未定义↑↑↑叫基本数据类型 基本数据类 ...

  5. d3.js:数据可视化利器之快速入门

    hello,data! 在进入d3.js之前,我们先用一个小例子回顾一下将数据可视化的基本流程. 任务 用横向柱状图来直观显示以下数据: var data = [10,15,23,78,57,29,3 ...

  6. mapbox-gl空间分析插件turf.js使用介绍

    mapbox-gl能够方便地显示地图,做一些交互,但是缺少空间分析功能,比如绘制缓冲区.判断点和面相交等等. turf.js是一个丰富的用于浏览器和node.js空间分析库,官网 http://tur ...

  7. JS内存空间详细图解

    JS内存空间详细图解 变量对象与堆内存 var a = 20; var b = 'abc'; var c = true; var d = { m: 20 } 因为JavaScript具有自动垃圾回收机 ...

  8. GIS应用|快速开发REST空间分析服务

    随着计算机的快速发展,GIS已经在各大领域得到应用,和我们的生活息息相关, 但是基于GIS几大厂商搭建服务,都会有一定的门槛,尤其是需要server,成本高,难度大,这里介绍一种在线GIS云平台,帮你 ...

  9. 基于IGServer的Web地图要素空间分析

    1. 引言 MapGIS IGServer 是中地数码的一款跨平台GIS 服务器产品,提供了空间数据管理.分析.可视化及共享服务 MapGIS IGServer的下载(试用)地址:MapGIS IGS ...

  10. ArcGIS空间分析工具

    1. 3D分析 1.1. 3D Features toolset 工具 工具 描述 3D Features toolset (3D 要素工具集) Add Z Information 添加 Z 信息 添 ...

随机推荐

  1. 好书推荐之《Java 核心技术:卷 1 基础知识》

    大佬推荐 <Java 核心技术:卷 1 基础知识>,这本书本来是 Sun 公司的官方用书,是一本 Java 的入门参考书. 对于 Java 初学者来说,是一本非常不错的值得时常翻阅的技术手 ...

  2. NC51179 选课

    题目链接 题目 题目描述 学校实行学分制. 每门的必修课都有固定的学分,同时还必须获得相应的选修课程学分. 学校开设了 N 门的选修课程,每个学生可选课程的数量 M 是给定的. 学生选修了这 M 门课 ...

  3. Javascript中的var变量声明作用域问题

    先看一下这两段代码的执行结果 var name2 = 'What!'; function a() { if (typeof name2 === 'undefined') { console.log(' ...

  4. 【C#】基于JsonConvert解析Json数据

    1 解析字典 ​ 1)解析为 JObject private void ParseJson() { // 解析为JObject string jsonStr = "{'name': 'zha ...

  5. 【Android 逆向】【攻防世界】ill-intentions

    1. apk 安装到手机, 啥输入框都没有 2. apk拖入到jadx中看看 public class MainActivity extends Activity { @Override // and ...

  6. swagger 文档优化 knife4j 增强 Swagger

    swagger 省去了程序员开发过程中拟写接口文档的时间,是团队开发必不可少的工具,原生的swagger 界面功能比较少,也不支持文档导出,业界也有不少针对swagger 文档界面优化的插件,良莠不齐 ...

  7. HttpURLConnection使用分析

    在项目中遇到各种版本的httpClient,所以想了解一下httpClient的实现 首先查看的是JDK1.1中自带的HttpURLConnection,看一下最初的设计是怎么样的 代码分析 使用Ht ...

  8. Ubuntu18.04搭建Vue开发环境

    更新软件源列表 sudo apt update nodejs安装 sudo apt install nodejs nodejs -v #查看版本 npm安装 sudo apt install npm ...

  9. centos7安装桌面-GNOME

    CENTOS7安装桌面系统 GNOME桌面 # yum安装 # 更新已安装软件 yum upgrade -y # 安装额外yum源 yum install epel-release -y # 安装X ...

  10. 【Java复健指南07】OOP中级02-重写与多态思想

    前情提要:https://www.cnblogs.com/DAYceng/category/2227185.html 重写 注意事项和使用细节 方法重写也叫方法覆法,需要满足下面的条件 1.子类的方法 ...