判断顶点凹凸性、判断多边形的凹凸性、填充凹坑将凹多边形处理为凸多边形【java实现+原理讲解】

简介: 判断顶点凹凸性、判断多边形的凹凸性、填充凹坑将凹多边形处理为凸多边形【java实现+原理讲解】

问题

  1. 如何识别多边形中一个顶点的凹凸性
  2. 如何判断一个多边形的凹凸性(若多边形有一个点为凹的,则多边形为凹的)
  3. 如何通过填充凹坑的方式将凹多边形转化为凸多边形


方法

将多边形点集顺序修改为逆时针顺序

判断多边形点击顺序的方法可以参考判断多边形点集顺序,如果判断出多边形点击顺序为顺时针的话,直接将集合反转一下即可


判断一个点是否为凹点


注意:使用这方法判断的前提是多边形点击顺序是逆时针的才行

最终实现


/**
* 将凹多边形转化为凸多边形
*/
public class ConcaveToConvexApi {
   /**
    * 将凹多边形转化为凸多边形
    *
    * @param pointList
    * @return
    */
   public List<Point> concaveToConvex(List<Point> pointList) {
       List<Point> pointListClone = (List<Point>) deepClone(pointList);
       将点集方向转化为逆时针方向
       this.correctPolygonDirection(pointListClone);
       删除掉凹点
       int j = 0;
       while (j < pointListClone.size()) {
           int i = (j - 1 + pointListClone.size()) % pointListClone.size();
           int k = (j + 1) % pointListClone.size();
           double x1 = pointListClone.get(j).getX() - pointListClone.get(i).getX();
           double y1 = pointListClone.get(j).getY() - pointListClone.get(i).getY();
           double x2 = pointListClone.get(k).getX() - pointListClone.get(i).getX();
           double y2 = pointListClone.get(k).getY() - pointListClone.get(i).getY();
           //叉积
           double crossProduct = x1 * y2 - x2 * y1;
           if (crossProduct > 0) {
               //凸顶点
               j++;
           } else {
               //凹顶点或者共线
               pointListClone.remove(j);
               if (j == pointListClone.size()) {
                   /*
                   一开始是 A B C D E F G 开始时j=0,GAB不触发删除
                   但是FGA触发删除之后,序列变成A B C D E F,可能FAB也会触发删除,因此需要将j重置于0
                    */
                   j = 0;
               } else {
                   if (j > 0) {
                      //这里为什么j--,可以看程序下面的图
                       j--;
                   }
               }
           }
       }
       return pointListClone;
   }
   /**
    * 修正零件的点位方向
    * 先找出最高那个点的索引i,然后在对{i.x-(i-1).x,i.y-(i-1).y}和{(i+1).x-(i-1).x,(i+1).y-(i-1).y}来求叉积,看正负就知道了
    *
    * @param pointList
    */
   private void correctPolygonDirection(List<Point> pointList) {
       if (pointList.size() == 0) {
           return;
       }
       找到零件的最高点
       int highestPointIndex = -1;
       double highestY = -Double.MAX_VALUE;
       for (int i = 0; i < pointList.size(); i++) {
           if (pointList.get(i).getY() > highestY) {
               highestY = pointList.get(i).getY();
               highestPointIndex = i;
           }
       }
       纠正坐标
       ///判断随着索引的进位,夹角是否为正数
       //存储方向,1:顺时针;-1:逆时针
       int direction = 0;
       Point lastPoint = pointList.get((highestPointIndex - 1 + pointList.size()) % pointList.size());
       Point curPoint = pointList.get(highestPointIndex);
       Point nextPoint = pointList.get((highestPointIndex + 1) % pointList.size());
       double x1 = curPoint.getX() - lastPoint.getX();
       double y1 = curPoint.getY() - lastPoint.getY();
       double x2 = nextPoint.getX() - lastPoint.getX();
       double y2 = nextPoint.getY() - lastPoint.getY();
       //计算向量的叉积
       double crossProduct = x1 * y2 - x2 * y1;
       if (crossProduct > 0) {
           direction = -1;
       } else if (crossProduct < 0) {
           direction = 1;
       }
       //如果是顺时针的话,需要改为逆时针
       if (direction == 1) {
           //将集合的元素反转
           Collections.reverse(pointList);
       }
   }
}

ac54ee3a5202475d8e9001ed12b95efe.png


测试


目录
相关文章
|
3月前
|
监控 Java API
现代 Java IO 高性能实践从原理到落地的高效实现路径与实战指南
本文深入解析现代Java高性能IO实践,涵盖异步非阻塞IO、操作系统优化、大文件处理、响应式网络编程与数据库访问,结合Netty、Reactor等技术落地高并发应用,助力构建高效可扩展的IO系统。
95 0
|
3月前
|
存储 缓存 安全
深入讲解 Java 并发编程核心原理与应用案例
本教程全面讲解Java并发编程,涵盖并发基础、线程安全、同步机制、并发工具类、线程池及实际应用案例,助你掌握多线程开发核心技术,提升程序性能与响应能力。
128 0
|
3月前
|
人工智能 安全 Java
Go与Java泛型原理简介
本文介绍了Go与Java泛型的实现原理。Go通过单态化为不同类型生成函数副本,提升运行效率;而Java则采用类型擦除,将泛型转为Object类型处理,保持兼容性但牺牲部分类型安全。两种机制各有优劣,适用于不同场景。
99 24
|
4月前
|
存储 缓存 Java
我们来详细讲一讲 Java NIO 底层原理
我是小假 期待与你的下一次相遇 ~
157 2
|
4月前
|
XML JSON Java
Java 反射:从原理到实战的全面解析与应用指南
本文深度解析Java反射机制,从原理到实战应用全覆盖。首先讲解反射的概念与核心原理,包括类加载过程和`Class`对象的作用;接着详细分析反射的核心API用法,如`Class`、`Constructor`、`Method`和`Field`的操作方法;最后通过动态代理和注解驱动配置解析等实战场景,帮助读者掌握反射技术的实际应用。内容翔实,适合希望深入理解Java反射机制的开发者。
332 13
|
4月前
|
算法 Java 索引
说一说 Java 并发队列原理剖析
我是小假 期待与你的下一次相遇 ~
|
4月前
|
安全 Java 编译器
JD-GUI,java反编译工具及原理: JavaDecompiler一个Java反编译器
Java Decompiler (JD-GUI) 是一款由 Pavel Kouznetsov 开发的图形化 Java 反编译工具,支持 Windows、Linux 和 Mac Os。它能将 `.class` 文件反编译为 Java 源代码,支持多文件标签浏览、高亮显示,并兼容 Java 5 及以上版本。JD-GUI 支持对整个 Jar 文件进行反编译,可跳转源码,适用于多种 JDK 和编译器。其原理基于将字节码转换为抽象语法树 (AST),再通过反编译生成代码。尽管程序可能带来安全风险,但可通过代码混淆降低可读性。最新版修复了多项识别错误并优化了内存管理。
1966 1
|
4月前
|
存储 算法 安全
Java中的对称加密算法的原理与实现
本文详细解析了Java中三种常用对称加密算法(AES、DES、3DES)的实现原理及应用。对称加密使用相同密钥进行加解密,适合数据安全传输与存储。AES作为现代标准,支持128/192/256位密钥,安全性高;DES采用56位密钥,现已不够安全;3DES通过三重加密增强安全性,但性能较低。文章提供了各算法的具体Java代码示例,便于快速上手实现加密解密操作,帮助用户根据需求选择合适的加密方案保护数据安全。
347 58

热门文章

最新文章