源码分析之Openlayers中Circle类

embedded/2025/1/6 5:20:01/

概述

Openlayers 中,Circle类用于圆的实例化,Circle类继承于SimpleGeometry类,关于SimpleGeometry类,可以参考源码分析之Openlayers中SimpleGeometry类

源码分析

Circle类的源码实现

Circle类的源码实现如下:

class Circle extends SimpleGeometry {constructor() {}clone() {const circle = new Circle(this.flatCoordinates.slice(),undefined,this.layout);circle.applyProperties(this);return circle;}closestPointXY(x, y, closestPoint, minSquaredDistance) {const flatCoordinates = this.flatCoordinates;const dx = x - flatCoordinates[0];const dy = y - flatCoordinates[1];const squaredDistance = dx * dx + dy * dy;if (squaredDistance < minSquaredDistance) {if (squaredDistance === 0) {for (let i = 0; i < this.stride; ++i) {closestPoint[i] = flatCoordinates[i];}} else {const delta = this.getRadius() / Math.sqrt(squaredDistance);closestPoint[0] = flatCoordinates[0] + delta * dx;closestPoint[1] = flatCoordinates[1] + delta * dy;for (let i = 2; i < this.stride; ++i) {closestPoint[i] = flatCoordinates[i];}}closestPoint.length = this.stride;return squaredDistance;}return minSquaredDistance;}containsXY(x, y) {const flatCoordinates = this.flatCoordinates;const dx = x - flatCoordinates[0];const dy = y - flatCoordinates[1];return dx * dx + dy * dy <= this.getRadiusSquared_();}getCenter() {return this.flatCoordinates.slice(0, this.stride);}computeExtent(extent) {const flatCoordinates = this.flatCoordinates;const radius = flatCoordinates[this.stride] - flatCoordinates[0];return createOrUpdate(flatCoordinates[0] - radius,flatCoordinates[1] - radius,flatCoordinates[0] + radius,flatCoordinates[1] + radius,extent);}getRadius() {return Math.sqrt(this.getRadiusSquared_());}getRadiusSquared_() {const dx = this.flatCoordinates[this.stride] - this.flatCoordinates[0];const dy = this.flatCoordinates[this.stride + 1] - this.flatCoordinates[1];return dx * dx + dy * dy;}getType() {return "Circle";}intersectsExtent(extent) {const circleExtent = this.getExtent();if (intersects(extent, circleExtent)) {const center = this.getCenter();if (extent[0] <= center[0] && extent[2] >= center[0]) {return true;}if (extent[1] <= center[1] && extent[3] >= center[1]) {return true;}return forEachCorner(extent, this.intersectsCoordinate.bind(this));}return false;}setCenter(center) {const stride = this.stride;const radius = this.flatCoordinates[stride] - this.flatCoordinates[0];const flatCoordinates = center.slice();flatCoordinates[stride] = flatCoordinates[0] + radius;for (let i = 1; i < stride; ++i) {flatCoordinates[stride + i] = center[i];}this.setFlatCoordinates(this.layout, flatCoordinates);this.changed();}setCenterAndRadius(center, radius, layout) {this.setLayout(layout, center, 0);if (!this.flatCoordinates) {this.flatCoordinates = [];}const flatCoordinates = this.flatCoordinates;let offset = deflateCoordinate(flatCoordinates, 0, center, this.stride);flatCoordinates[offset++] = flatCoordinates[0] + radius;for (let i = 1, ii = this.stride; i < ii; ++i) {flatCoordinates[offset++] = flatCoordinates[i];}flatCoordinates.length = offset;this.changed();}getCoordinates() {return null;}setCoordinates(coordinates, layout) {}setRadius(radius) {this.flatCoordinates[this.stride] = this.flatCoordinates[0] + radius;this.changed();}rotate(angle, anchor) {const center = this.getCenter();const stride = this.getStride();this.setCenter(rotate(center, 0, center.length, stride, angle, anchor, center));this.changed();}
}

Circle类的构造函数

Circle类的构造函数接受三个参数:中心点坐标center、半径radius和坐标布局方式layout。构造函数会先判断,若layout存在,且半径radius不存在,则调用this.setFlatCoordinates方法设置this.layoutthis.stridethis.flatCoordinates;否则,判断若radius不存在或为false,则修改为0,最后调用this.setCenterAndRadius方法,设置中心点和半径。

Circle类的主要方法

Circle类的主要方法如下:

  • clone方法:用于复制一个圆几何对象,内部就是实例化Circle类,然后调用实例对象的applyProperties方法,最后返回实例对象。

  • closestPointXY方法:接受四个参数:给定点xy、最近点坐标closestPoint和最小距离平方;closestPointXY方法就是获取给定点距离几何对象(即圆)的最小距离平方,并且修正最近点的坐标值。方法内部会先计算给定点到中心点的距离平方squaredDistance,然后比较squaredDistanceminSquaredDistance的大小;若squaredDistance大于或等于minSquaredDistance,则直接返回minSquaredDistance;否则判断,若squaredDistance等于0,这说明给定点就是中心点,然后修改最近点坐标为中心点坐标,并且返回0;否则,通过几何对象的半径大小和给定点到圆心距离的比例计算出最近点的坐标,然后多维的数据比如M就保持不变。最后返回给定点到圆心的距离。

  • containsXY方法:接受两个参数即给定点的坐标xy,然后计算给定点和几何对象中心点的距离平方,比较它和半径平方的大小,若半径平方较大或等于计算的平方,则说明给定点在几何对象内部或者是在圆上。

  • getCenter方法:用于获取几何对象的中心点坐标。

  • computeExtent方法:用于获取几何对象的包围盒。

  • getRadius方法:用于获取几何对象的半径,内部就是调用this.getRadiusSquared_方法获取半径的平方。

  • getRadiusSquared方法:获取半径的平方。

  • getType方法:获取几何对象的类型,Circle

  • intersectsExtent方法:用于判断矩形extent是否与几何对象相交

  • setCenter方法:设置几何对象中心点坐标;接受一个参数center,先从this.flatCoordinates中计算出半径,然后 重新组装变量flatCoordinates,再调用this.setFlatCoordinates方法设置this.flatCoordinatesthis.layoutthis.stride,最后调用this.changed方法。

  • setCenterAndRadius方法:设置几何对象的中心点坐标和半径。

  • getCoordinates方法:返回null,因为几何对象圆重要的属性是圆心(即中心点坐标)和半径大小。

  • setCoordinates方法:未实现。

  • setRadius方法:用于设置半径,几何对象的半径也是存在this.flatCoordinates,即this.flatCoordinates的第三项的大小为x坐标值加上半径大小,最后会调用this.changed方法。

  • rotate方法:接受两个参数:旋转角度angle和旋转锚点anchor。方法内部会先分别调用this.getCenterthis.getStride方法获取几何对象的中心点坐标和步幅this.stride,然后调用rotate方法对中心坐标进行旋转,修改中心点坐标并返回修改后的中心点坐标,然后调用this.setCenter方法设置中心点坐标,最后调用this.changed方法。

总结

本文主要介绍了Circle类的源码实现和原理,精华的地方就是closestPointXY方法中当给定点(x,y)在几何对象内部但又不是圆心时,计算最近点距离的逻辑,需要具备一定的数据知识。


http://www.ppmy.cn/embedded/150539.html

相关文章

ctr方法下载的镜像能用docker save进行保存吗?

ctr 和 docker 是两个不同的容器运行时工具,它们使用的镜像存储格式是兼容的(都是 OCI 标准镜像),但它们的镜像管理方式和存储路径不同。因此,直接使用 docker save 保存 ctr 拉取的镜像可能会遇到问题。 关键点 ctr 和 docker 的镜像存储位置不同: ctr(containerd)的镜…

2 秒杀系统架构

第一步 思考面临的问题和业务场景 秒杀系统面临的问题: 短时间内并发非常高&#xff0c;如果按照秒杀的并发做相应的承载会造成大量资源的浪费。第二解决超卖的问题。 第二步 思考目前的处境和解决方案 因为秒杀系统属于短时间内的高并发问题&#xff0c;我们不可能使用那么…

学习threejs,导入pdb格式的模型

&#x1f468;‍⚕️ 主页&#xff1a; gis分享者 &#x1f468;‍⚕️ 感谢各位大佬 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍⚕️ 收录于专栏&#xff1a;threejs gis工程师 文章目录 一、&#x1f340;前言1.1 ☘️THREE.PDBLoader pdb模型加…

学习路之VScode--自定义按键写注释(插件)

1. 安装 "KoroFileHeader" 插件 首先&#xff0c;在 VScode 中搜索并安装名为 "KoroFileHeader" 的插件。你可以通过在扩展商店中搜索插件名称来找到并安装它。 2. 进入 VScode 设置页面 点击 VScode 左下角的设置图标&#xff0c;然后选择 "设置&q…

CANape 新建工程和连接

文章目录 简介1、新建工程1.1 打开 CANape1.2 新建工程1.3 新建 Device1.3.1 添加NEW DEVICE1.3.2 添加 NEW From Database1.4 配置 Memory Segment1.5 新建trace窗口和观测窗口 2、硬件连接2.1 更改与canape盒子通道一致的通道编号&#xff0c;选择驱动配置2.2 选择硬件配置 问…

探究人工智能在教育领域的应用——以大语言模型为例

摘要&#xff1a;随着人工智能技术的不断发展&#xff0c;大语言模型&#xff08;LLMs&#xff09;已经成为各行各业的重要工具。本文旨在探讨大语言模型在教育领域的应用现状、挑战和前景&#xff0c;提出一套切实可行的解决方案&#xff0c;以期为教育行业带来革新。通过对大…

Spark生态圈

Spark 主要用于替代Hadoop中的 MapReduce 计算模型。存储依然可以使用 HDFS&#xff0c;但是中间结果可以存放在内存中&#xff1b;调度可以使用 Spark 内置的&#xff0c;也可以使用更成熟的调度系统 YARN 等。 Spark有完善的生态圈&#xff1a; Spark Core&#xff1a;实现了…

图像处理-Ch7-小波函数

个人博客&#xff01;无广告观看&#xff0c;因为这节内容太多了&#xff0c;有点放不下&#xff0c;分了三节 文章目录 多分辨率展开(Multi-resolution Expansions)序列展开(Series Expansions)尺度函数(Scaling Function)例&#xff1a;哈尔尺度函数(Haar scaling func)多分…