美文网首页玄说前端
从一片飞地说起的地图画法

从一片飞地说起的地图画法

作者: 玄魂 | 来源:发表于2020-04-03 22:30 被阅读0次

本文作者:字节跳动数据平台 早千


这是一张再熟悉不过的地图,但右上角黑龙江与内蒙古之间的那片飞地是什么呢?

image

查一下,摘自互联网:

大兴安岭是黑龙江省下辖的一个非常特殊的地级行政单位,大兴安岭行政公署和大兴安岭林业集团公司实行政企合一的管理体制。行政公署为省政府派驻机构,所辖加格达奇和松岭两区,地权属于内蒙古自治区呼伦贝尔市。黑龙江每年向内蒙古支付租金,加格达奇属于“租界”。所以,加格达奇人是黑龙江的,地是内蒙古的,树是林业部的。

哦豁,原来如此。知道了,但是我突然更好奇这个地图是怎么画在网页上的呢?


将地图画在页面上,概括一下分三步。1. 拿到地图信息数据 2. 投影 3. 画

地图数据

这里介绍两种可以用于表示地理信息的数据类型

  • GeoJSON

  • TopoJSON

GeoJSON

GeoJSON是一种基于JSON的地理空间数据交换格式,它定义了几种类型JSON对象以及它们组合在一起的方法,以表示有关地理要素、属性和它们的空间范围的数据。

如下表所示,坐标表示点。点连成线,线连成面,从而表现出各种各样的形状。

image image image

一段GeoJSON示例

{
  "type": "FeatureCollection",
  "features": [
    {
      "type": "Feature",
      "geometry": {
        "type": "Point",
        "coordinates": [102.0, 0.5]
      },
      "properties": {
        "prop0": "value0"
      }
    },
    {
      "type": "Feature",
      "geometry": {
        "type": "LineString",
        "coordinates": [
          [102.0, 0.0], [103.0, 1.0], [104.0, 0.0], [105.0, 1.0]
        ]
      },
      "properties": {
        "prop0": "value0",
        "prop1": 0.0
      }
    },
    {
      "type": "Feature",
      "geometry": {
        "type": "Polygon",
        "coordinates": [
          [
            [100.0, 0.0], [101.0, 0.0], [101.0, 1.0],
            [100.0, 1.0], [100.0, 0.0]
          ]
        ]
      },
      "properties": {
        "prop0": "value0",
        "prop1": { "this": "that" }
      }
    }
  ]
}

TopoJSON

TopoJSON 是 GeoJSON 按拓扑学编码后的扩展形式,是由 D3 的作者 Mike Bostock 制定的。相比 GeoJSON 直接使用 Polygon、Point 之类的几何体来表示图形的方法,TopoJSON 中的每一个几何体都是通过将共享边(被称为arcs)整合后组成的。

简单说,它的线并不是由点构成,而是有自己的id。

边在一起组成几何形状时,最后一个坐标必须与后续边的第一个坐标(如果有)相同。例如,如果边6代表点序列A→B→C,边7代表点序列C→D→E,则[6,7]代表点序列A→B→C→D→E。

由于边是共享的所以在表示不同方向时需要反转。使用补码表示。-1(〜0)为反0,-2(〜1)为反1,依此类推。

image

以上图举例说明。绿色点与红色点坐标分别代表两个点。紫色边为3。如果定义蓝色边顺时针方向为1,黑色边顺时针方向为0,则蓝色边与黑色边组成几何图形为[[0, 1]]。在此基础上红色顺时针为2,则红色边与逆黑色边(~0 = -1)组成图形为[[2, -1]]。

获取数据

  1. http://datav.aliyun.com/tools/atlas/ 直接下载中国及省市GeoJSON

  2. http://www.naturalearthdata.com/ 获取各种各样的shape格式(.shp)地图数据,使用mapshaper(https://github.com/mbloch/mapshaper)进行处理后导出为GeoJSON或TopoJSON。

投影

地图投影,是指按照一定的数学法则将地球椭球面上的经纬网转换到平面上,使地面的地理坐标与平面直角坐标建立起函数关系,是绘制地图的数学基础之一。由于地球是一个不可展的球体,使用物理方法将其展平会引起褶皱、拉伸和断裂,因此要使用地图投影实现由曲面向平面的转化。

下表是一些地图投影与经纬线形状特征

image

示例的中国地图就是采用了墨卡托投影。

经过上面的介绍,接下来只需要将二维平面内容绘制出来即可。

这里使用d3与d3-geo举例绘制上面提到的中国地图。为了美观,使用d3-scale-chromatic的一个主题随意上了颜色。

 import * as d3 from 'd3';
import * as geo from 'd3-geo';
import * as d3Color from 'd3-scale-chromatic';

在页面中创建一个svg元素

<svg id="svg" width="1080" height="600"></svg>

创建投影,设置参数

const projection = geo.geoMercator() // 墨卡托投影
.scale(450) // 投影的比例因子,可以按比例放大投影。
.center([105, 38]) // 中心点设置为中国的中心位置 大约经度105,纬度38

利用得到的投影创建path,补充颜色与边线

const svg =  d3.select('#svg');

const path = geo.geoPath(projection);
const colors = d3.scaleOrdinal(d3Color.schemeBlues[9]);

const area = svg.selectAll('path')
                .data(geoData.features) // 这里的geoData是下载好的中国地图GeoJSON格式数据
                .enter()
                .append('path')
                .attr('d', path)
                .attr('fill', (_, i) => {
                  return colors(i);
                })
                .attr('stroke', '#fff')
                .attr('stroke-width', 1);

到此为止,中国地图就已经出现在了我们面前。

如果在这个基础上将北京上海两座城市标注出来呢?定义经纬度

const places = [
  {
    'name': '北京',
    'log': '116.3',
    'lat': '39.9'
  },
  {
    'name': '上海',
    'log': '121.4',
    'lat': '31.2'
  }
];

故技重施,用svg画出两个点

const location = svg.selectAll('g')
                    .data(places)
                    .enter()
                    .append('g')
                    .attr('transform', (d) => {
                        const coor = projection([d.log, d.lat]); // 经纬度转换为绘制坐标
                        return 'translate(' + coor[0] + ',' + coor[1] + ')';
                    });

// 绿色圆点
location.append('circle')
        .attr('r', 4)
        .attr('fill', '#a0d911')

现在地图上的北京与上海两地就被标注出来咯

image

看来要绘制一幅地图也并不难呀。


原文链接

-------------------

相关文章

  • 从一片飞地说起的地图画法

    本文作者:字节跳动数据平台 早千 这是一张再熟悉不过的地图,但右上角黑龙江与内蒙古之间的那片飞地是什么呢? 查一下...

  • 从一片树叶说起

    今天上午赵校长针对“271教育职业追求和学科主任的使命担当”继续为我们每一位教师阐释学科主任的重要性和学科下学期的...

  • 从一片叶子说起

    晚上刷朋友圈时,看到王老师发的图片,在叶子上写诗,在叶子上作画,图片儿美的不要不要的。于是和王老师联系,怎么才...

  • 从一片卫生巾说起

    好久没写文章了,懒懒散散没有动力,后面坚持每周至少写一篇吧。 昨天是听了“护你妹”后特别有感觉,一家成立时间不到两...

  • 从一部纪录片说起

    下午,班上有一节自习课。由于天气阴沉,坐在教室里的学生,看起来精神头不是那么足。正琢磨着自习课上让学生做什么,灵机...

  • 开放和雄心:从一张地图说起

    这张地图摘自《人类简史》第十五章 科学与帝国的联姻。特别之处在于,不同于这个时代以往的地图那样——关于不熟悉的区域...

  • 飞地

    我看见 有的人不知道去哪 有的人哪也不去 如果我也走得够快 是否就可以赶上身后的大风? 我们去吹开那些挤在一起的房...

  • 与孩子共享自然,从一片落叶说起

    从一片落叶说起 前天儿子突然指着路上的叶子,咿咿呀呀想要告诉我什么,我猛地反应出来,哦,原来是说落叶变少了。 于是...

  • 三行情书——第十八封

    我很想你,只是不知从何说起 从一片叶落旋转到归根的距离 从一缕阳光穿越万千里抵达你的窗前

  • 会思考,很重要

    01 我们先从一部名为《幼儿园大学》(《Nursery university》)的纪录片说起。片子取自纽约曼哈顿,...

网友评论

    本文标题:从一片飞地说起的地图画法

    本文链接:https://www.haomeiwen.com/subject/cdilphtx.html