LeetCode题练习与总结:安装栅栏--587

news2025/1/30 14:40:29

一、题目描述

给定一个数组 trees,其中 trees[i] = [xi, yi] 表示树在花园中的位置。

你被要求用最短长度的绳子把整个花园围起来,因为绳子很贵。只有把 所有的树都围起来,花园才围得很好。

返回恰好位于围栏周边的树木的坐标

示例 1:

输入: points = [[1,1],[2,2],[2,0],[2,4],[3,3],[4,2]]
输出: [[1,1],[2,0],[3,3],[2,4],[4,2]]

示例 2:

输入: points = [[1,2],[2,2],[4,2]]
输出: [[4,2],[2,2],[1,2]]

注意:

  • 1 <= points.length <= 3000
  • points[i].length == 2
  • 0 <= xi, yi <= 100
  • 所有给定的点都是 唯一 的。

二、解题思路

这个问题是计算几何中的经典问题,称为“凸包问题”。凸包是包含所有给定点的最小凸多边形。在这个问题中,树木的位置是给定的点,我们需要找到这些点的凸包,这个凸包的顶点就是我们要找的树木坐标。

解决这个问题的常用算法有几种,比如“Graham扫描”和“Andrew扫描”。这里我将使用“Graham扫描”算法来解决这个问题,因为它相对简单且易于实现。

Graham扫描算法的步骤如下:

  1. 找到所有点中y坐标最小的点,如果有多个这样的点,则选择x坐标最小的点作为起点。
  2. 将所有点按照极角(相对于起点)进行排序。
  3. 从起点开始,按顺序遍历每个点,检查当前点是否使凸包的拐向从左到右。如果是,则将当前点添加到凸包中;如果不是,则移除前一个点,因为它不在凸包上。
  4. 重复步骤3,直到所有点都被处理。

三、具体代码

import java.util.*;

class Solution {
    public int[][] outerTrees(int[][] trees) {
        if (trees.length <= 1) return trees;

        // Step 1: Find the starting point
        int start = 0;
        for (int i = 1; i < trees.length; i++) {
            if (trees[i][1] < trees[start][1] || (trees[i][1] == trees[start][1] && trees[i][0] < trees[start][0])) {
                start = i;
            }
        }

        // Swap the starting point with the first point
        int[] temp = trees[0];
        trees[0] = trees[start];
        trees[start] = temp;

        // Step 2: Sort the points based on polar angle with respect to the starting point
        Arrays.sort(trees, 1, trees.length, (a, b) -> {
            int orientation = orientation(trees[0], a, b);
            if (orientation == 0) {
                return distance(trees[0], a) - distance(trees[0], b); // Sort by distance if collinear
            }
            return orientation;
        });

        // Step 3 and 4: Graham scan
        List<int[]> hull = new ArrayList<>();
        for (int[] p : trees) {
            while (hull.size() >= 2 && orientation(hull.get(hull.size() - 2), hull.get(hull.size() - 1), p) > 0) {
                hull.remove(hull.size() - 1); // Remove the last point
            }
            hull.add(p);
        }

        // Convert the hull list to array
        return hull.toArray(new int[hull.size()][]);
    }

    // Helper function to calculate the cross product of vectors OA and OB
    // A positive cross product indicates a counter-clockwise turn, 0 indicates a collinear point, and negative indicates a clockwise turn
    private int orientation(int[] o, int[] a, int[] b) {
        return (a[1] - o[1]) * (b[0] - a[0]) - (a[0] - o[0]) * (b[1] - a[1]);
    }

    // Helper function to calculate the squared distance between two points
    private int distance(int[] a, int[] b) {
        return (a[0] - b[0]) * (a[0] - b[0]) + (a[1] - b[1]) * (a[1] - b[1]);
    }
}

四、时间复杂度和空间复杂度

1. 时间复杂度
  • 寻找起始点:O(n),其中n是数组trees的长度。我们需要遍历所有点来找到y坐标最小(或x坐标最小,如果y坐标相同)的点。
  • 排序点:O(n log n),使用快速排序或其他基于比较的排序算法对点进行排序,根据极角(如果极角相同,则根据距离)。
  • Graham扫描:O(n),遍历排序后的点,对于每个点,我们可能需要从凸包中移除最后一个点,直到找到合适的点加入凸包。由于每个点最多被加入和移除一次,整个过程是线性的。

综上所述,总的时间复杂度是O(n log n),因为排序步骤是主导步骤。

2. 空间复杂度
  • 凸包列表:O(n),在最坏的情况下,所有点都在凸包上,因此我们需要存储所有点。
  • 辅助空间:O(1),除了凸包列表外,我们只使用了常数额外空间。

因此,总的空间复杂度是O(n),其中n是数组trees的长度。

五、总结知识点

  • 算法设计

    • 凸包问题:理解凸包的概念以及如何通过算法找到给定点的凸包。
    • Graham扫描算法:掌握Graham扫描算法的步骤,包括选择起始点、排序点、以及构建凸包。
  • 数据结构

    • 数组:使用数组来存储点的坐标。
    • 列表:使用ArrayList来动态存储构成凸包的点。
  • 排序

    • 自定义排序:使用Arrays.sort方法和自定义比较器(Comparator)来根据极角和距离对点进行排序。
  • 数学概念

    • 向量叉乘:计算向量叉乘以确定点的相对位置(顺时针、逆时针或共线)。
    • 距离计算:计算两点之间的欧几里得距离的平方,用于比较共线点之间的距离。
  • 算法逻辑

    • 循环和条件判断:使用循环和条件判断来遍历点集,并在构建凸包时进行逻辑判断。
    • 边界条件处理:检查点集长度,如果长度小于等于1,直接返回原点集。
  • 辅助函数

    • 辅助函数orientation:计算向量叉乘,用于判断点的相对位置。
    • 辅助函数distance:计算两点之间的距离的平方。
  • Java编程技巧

    • 数组操作:交换数组元素以将起始点置于数组首位。
    • 泛型:使用泛型ArrayList来存储int[]类型的点。
    • Lambda表达式:在自定义排序中使用Lambda表达式简化代码。

以上就是解决这个问题的详细步骤,希望能够为各位提供启发和帮助。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2284772.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

< OS 有关 > 阿里云 几个小时前 使用密钥替换 SSH 密码认证后, 发现主机正在被“攻击” 分析与应对

信息来源&#xff1a; 文件&#xff1a;/var/log/auth.log 因为在 sshd_config 配置文件中&#xff0c;已经定义 LogLevel INFO 部分内容&#xff1a; 2025-01-27T18:18:55.68272708:00 jpn sshd[15891]: Received disconnect from 45.194.37.171 port 58954:11: Bye Bye […

使用八爪鱼爬虫和Web Scraper抓取数据实战案例,附详细教程

最近有不少小伙伴咨询怎么抓取抖音视频或者评论的数据&#xff0c;他们多是自媒体或者商家&#xff0c;想要模仿爆火视频或者分析视频评论区的舆情信息&#xff0c;确实呀&#xff0c;现在抖音是流量高地&#xff0c;淘金的地方&#xff0c;真的是一个值得挖掘的宝藏。当然我一…

海外问卷调查渠道查如何设置:最佳实践+示例

随着经济全球化和一体化进程的加速&#xff0c;企业间的竞争日益加剧&#xff0c;为了获得更大的市场份额&#xff0c;对企业和品牌而言&#xff0c;了解受众群体的的需求、偏好和痛点才是走向成功的关键。而海外问卷调查才是获得受众群体痛点的关键&#xff0c;制作海外问卷调…

【C++数论】880. 索引处的解码字符串|2010

本文涉及知识点 数论&#xff1a;质数、最大公约数、菲蜀定理 LeetCode880. 索引处的解码字符串 给定一个编码字符串 s 。请你找出 解码字符串 并将其写入磁带。解码时&#xff0c;从编码字符串中 每次读取一个字符 &#xff0c;并采取以下步骤&#xff1a; 如果所读的字符是…

从ai产品推荐到利用cursor快速掌握一个开源项目再到langchain手搓一个Text2Sql agent

目录 0. 经验分享&#xff1a;产品推荐 1. 经验分享&#xff1a;提示词优化 2. 经验分享&#xff1a;使用cursor 阅读一篇文章 3. 经验分享&#xff1a;使用cursor 阅读一个完全陌生的开源项目 4. 经验分享&#xff1a;手搓一个text2sql agent &#xff08;使用langchain l…

Blazor-选择循环语句

今天我们来说说Blazor选择语句和循环语句。 下面我们以一个简单的例子来讲解相关的语法&#xff0c;我已经创建好了一个Student类&#xff0c;以此类来进行语法的运用 因为我们需要交互性所以我们将类创建在*.client目录下 if 我们做一个学生信息的显示&#xff0c;Gender为…

appium自动化环境搭建

一、appium介绍 appium介绍 appium是一个开源工具、支持跨平台、用于自动化ios、安卓手机和windows桌面平台上面的原生、移动web和混合应用&#xff0c;支持多种编程语言(python&#xff0c;java&#xff0c;Ruby&#xff0c;Javascript、PHP等) 原生应用和混合应用&#xf…

大数据Hadoop入门2

目录 第三部分&#xff08;Hadoop MapReduce和Hadoop YARN&#xff09; 1.课程内容-大纲-学习目标 2.理解先分再合、分而治之的思想 3.hadoop团队针对MapReduce的设计构思 4.Hadoop MapReduce介绍、阶级划分和进程组成 5.Hadoop MapReduce官方示例-圆周率PI评估 6.Hadoo…

21.Word:小赵-毕业论文排版❗【39】

目录 题目​ NO1.2 NO3.4 NO5.6 NO7.8.9 NO10.11.12 题目 NO1.2 自己的论文当中接收老师的修改&#xff1a;审阅→比较→源文档&#xff1a;考生文件夹&#xff1a;Word.docx→修订的文档&#xff1a;考生文件夹&#xff1a;教师修改→确定→接收→接收所有修订将合并之…

【go语言】并发编程

一、协程、线程、进程 在计算机编程中&#xff0c;进程、线程和协程都是用于并发执行任务的不同概念。他们的区别主要体现在创建、管理和调度的复杂度上&#xff0c;特别是在不同的编程语言中有不同的实现方式。下面是他们的详细区别和在 go 语言中的实现方式。 1.1 进程 定义…

算法1-1 模拟与高精度

目录 一 阶乘数码 二 麦森数 三 模拟题 一 阶乘数码 本题中n<1000,1000的阶乘为以下这么大&#xff0c;远超long的范围 402387260077093773543702433923003985719374864210714632543799910429938512398629020592044208486969404800479988610197196058631666872994808558901…

公式与函数的应用

一 相邻表格相乘 1 也可以复制 打印标题

ShenNiusModularity项目源码学习(7:数据库结构)

ShenNiusModularity项目默认使用mysql数据库&#xff0c;数据库连接字符串放到了ShenNius.Admin. Mvc、ShenNius.Admin.Hosting的appsettings.json文件内。   ShenNiusModularity项目为自媒体内容管理系统&#xff0c;支持常规管理、CMS管理、商城管理等功能&#xff0c;其数…

手撕Diffusion系列 - 第九期 - 改进为Stable Diffusion(原理介绍)

手撕Diffusion系列 - 第九期 - 改进为Stable Diffusion&#xff08;原理介绍&#xff09; 目录 手撕Diffusion系列 - 第九期 - 改进为Stable Diffusion&#xff08;原理介绍&#xff09;DDPM 原理图Stable Diffusion 原理Stable Diffusion的原理解释Stable Diffusion 和 Diffus…

论文笔记(六十三)Understanding Diffusion Models: A Unified Perspective(三)

Understanding Diffusion Models: A Unified Perspective&#xff08;三&#xff09; 文章概括 文章概括 引用&#xff1a; article{luo2022understanding,title{Understanding diffusion models: A unified perspective},author{Luo, Calvin},journal{arXiv preprint arXiv:…

修改maven的编码格式为utf-8

1.maven默认编码为GBK 注:配好MAVEN_HOME的环境变量后,在运行cmd. 打开cmd 运行mvn -v命令即可. 2.修改UTF-8为默认编码. 设置环境变量 变量名 MAVEN_OPTS 变量值 -Xms256m -Xmx512m -Dfile.encodingUTF-8 3.保存,退出cmd.重新打开cmd 运行mvn -v命令即可. 源码获取&…

从AD的原理图自动提取引脚网络的小工具

这里跟大家分享一个我自己写的小软件&#xff0c;实现从AD的原理图里自动找出网络名称和引脚的对应。存成文本方便后续做表格或是使用简单行列编辑生成引脚约束文件&#xff08;如.XDC .UCF .TCL等&#xff09;。 我们在FPGA设计中需要引脚锁定文件&#xff0c;就是指示TOP层…

【数据结构】(1)集合类的认识

一、什么是数据结构 1、数据结构的定义 数据结构就是存储、组织数据的方式&#xff0c;即相互之间存在一种或多种关系的数据元素的集合。 2、学习数据结构的目的 在实际开发中&#xff0c;我们需要使用大量的数据。为了高效地管理这些数据&#xff0c;实现增删改查等操作&…

解决使用Selenium时ChromeDriver版本不匹配问题

在学习Python爬虫过程中如果使用Selenium的时候遇到报错如下session not created: This version of ChromeDriver only supports Chrome version 99… 这说明当前你的chrome驱动版本和浏览器版本不匹配。 例如 SessionNotCreatedException: Message: session not created: This…

CAN波特率匹配

STM32 LinuxIMX6ull&#xff08;Linux&#xff09;基于can-utils测试