C# 给定欧氏平面中的一组线可以形成的三角形的数量

         给定欧氏平面中的一组线可以形成的三角形的数量(Number of Triangles that can be formed given a set of lines in Euclidean Plane)

        给定欧氏平面上的 n 条不同直线的集合 L = {l 1 , l 2 , ………, l n }。第i 条直线由形式为 a i x + b i y = c i的方程给出。求出可以使用集合 L 中的直线形成的三角形的数量。请注意,没有两对直线会在同一点相交。 
注意:此问题未提及直线不能平行,这使得问题难以解决。

例子: 

输入:a[] = {1, 2, 3, 4} 
       b[] = {2, 4, 5, 5} 
       c[] = {5, 7, 8, 6}
       
输出:2

可以形成的三角形数量为:2

输入:a[] = {1, 2, 3, 2, 4, 1, 2, 3, 4, 5} 
       b[] = {2, 4, 6, 3, 6, 5, 10, 15, 20, 25} 
       c[] = {3, 5, 11, 10, 9, 17, 13, 11, 7, 3}
       
输出:30

可以形成的三角形数量为:30

朴素算法

朴素算法可以描述为: 

    1、从集合 L 中选取 3 条任意线。

    2、现在检查是否可以使用选定的 3 条线形成三角形。这可以通过检查它们是否都是平行线来轻松完成。

    3、如果可以形成三角形,则增加计数器。 
 
时间复杂度:有n C 3 (如图:)个三元组线。对于每个三元组,我们必须进行 3 次比较才能检查任何 2 条线是否不平行,这意味着检查可以在 O(1) 时间内完成。这使得朴素算法成为 O(n 3 ),如图: 

高效算法

这也可以在 O(n log n) 中实现。高效算法背后的逻辑如下所述。

我们将集合 L 划分为各种子集。子集的形成基于斜率,即特定子集中的所有线具有相同的斜率,即它们彼此平行。

让我们考虑三个集合(例如 A、B 和 C)。对于特定集合(例如 A),属于该集合的线都是彼此平行的。如果我们有 A、B 和 C,我们可以从每个集合中挑选一条线来得到一个三角形,因为这些线都不会平行。通过制作子集,我们确保没有两条平行的线被一起挑选。

现在如果我们只有这3 个子集, 

三角形的数量 =(从 A 中选取一条线的方式数量)* 
                      (从 B 中选取一条线的方式数量)* 
                      (从 C 中选取一条线的方式数量)
                   = m1*m2*m3

这里 m1 是具有第一个斜率的元素的数量(在集合 A 中)
这里 m2 是具有第一个斜率的元素的数量(在集合 B 中)
这里 m3 是具有第一个斜率的元素的数量(在集合 C 中)

类似地,如果我们有4 个子集,我们可以扩展这个逻辑来得到, 
三角形数量 = m1*m2*m3 + m1*m2*m4 + m1*m3*m4 + m2*m3*m4

对于大于 3 的子集数量,如果我们有“k”个子集,我们的任务是找到每次取 3 个子集元素数量的总和。这可以通过维护一个计数数组来完成。我们创建一个计数数组,其中计数i表示第i 个平行线子集的数量。 

我们逐一计算以下值。

sum1 = m1 + m2 + m3 ..... 
sum2 = m1*m2 + m1*m3 + ... + m2*m3 + m2*m4 + ... 
sum3 = m1*m2*m3 + m1*m2*m4 + ...... m2*m3*m4 + .... 
sum3 给出我们的最终答案

示例代码: 

// C# program to find the number of 
// triangles that can be formed 
// using a set of lines in Euclidean 
// Plane 
using System.Collections.Generic; 
using System;  
 
class GFG{
     
static double EPSILON = 1.0842e-19;
 
// Double variables can't be checked precisely 
// using '==' this function returns true if 
// the double variables are equal 
static bool compareDoubles(double A, double B) 

    double diff = A - B; 
    return (diff < EPSILON) && 
          (-diff < EPSILON); 

 
// This function returns the number of 
// triangles for a given set of lines 
static int numberOfTringles(int []a, int []b, 
                            int []c, int n) 

     
    // Slope array stores the slope of lines 
    List<double> slope = new List<double>();
    for(int i = 0; i < n; i++) 
        slope.Add((double)(a[i] * 1.0) / b[i]); 
 
    // Slope array is sorted so that all lines 
    // with same slope come together 
    slope.Sort(); 
 
    // After sorting slopes, count different 
    // slopes. k is index in count[]. 
    int []count = new int [n];
    int k = 0; 
     
    // Count of current slope 
    int this_count = 1; 
     
    for(int i = 1; i < n; i++) 
    { 
        if (compareDoubles((double)slope[i],
                           (double)slope[i - 1])) 
            this_count++; 
        else
        { 
            count[k++] = this_count; 
            this_count = 1; 
        } 
    } 
    count[k++] = this_count; 
 
    // Calculating sum1 (Sum of all slopes) 
    // sum1 = m1 + m2 + ... 
    int sum1 = 0; 
    for(int i = 0; i < k; i++) 
        sum1 += count[i]; 
 
    // Calculating sum2. sum2 = m1*m2 + m2*m3 + ... 
    int sum2 = 0; 
     
    // Needed for sum3 
    int []temp = new int [n]; 
    for(int i = 0; i < k; i++) 
    { 
        temp[i] = count[i] * (sum1 - count[i]); 
        sum2 += temp[i]; 
    } 
    sum2 /= 2; 
 
    // Calculating sum3 which gives
    // the final answer 
    // m1 * m2 * m3 + m2 * m3 * m4 + ... 
    int sum3 = 0; 
     
    for(int i = 0; i < k; i++) 
        sum3 += count[i] * (sum2 - temp[i]);
         
    sum3 /= 3; 
 
    return sum3; 

 
// Driver code 
public static void Main() 

     
    // lines are stored as arrays of a, b 
    // and c for 'ax+by=c' 
    int []a = { 1, 2, 3, 4 }; 
    int []b = { 2, 4, 5, 5 }; 
    int []c = { 5, 7, 8, 6 }; 
 
    // n is the number of lines 
    int n = a.Length;
 
     Console.WriteLine("The number of triangles " +
                       "that can be formed are: " + 
                       numberOfTringles(a, b, c, n));

}
 
// This code is contributed by Stream_Cipher 

输出: 

可形成的三角形数量为:2

时间复杂度:代码中的所有循环都是 O(n)。因此,此实现中的时间复杂度由用于对斜率数组进行排序的排序函数决定。这使得算法为 O(nlogn)。

辅助空间:O(n)

如果您喜欢此文章,请收藏、点赞、评论,谢谢,祝您快乐每一天。

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

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

相关文章

从CRUD到高级功能:EF Core在.NET Core中全面应用(三)

目录 IQueryable使用 原生SQL使用 实体状态跟踪 全局查询筛选器 并发控制使用 IQueryable使用 在EFCore中IQueryable是一个接口用于表示可查询的集合&#xff0c;它继承自IEnumerable但具有一些关键的区别&#xff0c;使得它在处理数据库查询时非常有用&#xff0c;普通集…

淘宝关键词页面爬取绘图进行数据分析

对爬虫、逆向感兴趣的同学可以查看文章&#xff0c;一对一小班V教学&#xff1a;https://blog.csdn.net/weixin_35770067/article/details/142514698 关键词页面爬取代码 from DrissionPage import WebPage, ChromiumOptions from DataRecorder import Recorder import time …

【C++提高篇】—— C++泛型编程之模板基本语法和使用的详解

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、模板的概念二、函数模板2.1 函数模板的使用2.2 函数模板注意事项2.3 普通函数与函数模板的区别2.4 普通函数与函数模板的调用规则2.5 模板的局限性 三、类模…

用JAVA写算法之输入输出篇

本系列适合原来用C语言或其他语言写算法&#xff0c;但是因为找工作或比赛的原因改用JAVA语言写算法的同学。当然也同样适合初学算法&#xff0c;想用JAVA来写算法题的同学。 常规方法&#xff1a;使用Scanner类和System.out 这种方法适用于leetcode&#xff0c;以及一些面试手…

uniapp中h5的微应用解决办法

考虑过用wujie&#xff0c;参考官网Vue组件封装 | 无界的教程&#xff0c;虽然没报错&#xff0c;但是子应用的vue节点根本没挂载上&#xff0c;不知道什么原因&#xff0c;如下图所示 后面采用iframe方式将子应用导入进来&#xff1a; 父应用&#xff1a; <template>&…

数据分析及应用:经营分析中的综合指标解析与应用

目录 1. 市场份额(Market Share) 2. 客户获取成本(Customer Acquisition Cost, CAC) 3. 客户生命周期价值(Customer Lifetime Value, CLV) 4. 客户留存率(Customer Retention Rate, CRR) 5. 净推荐值(Net Promoter Score, NPS) 6. 转化率(Conversion Rate) …

【大数据】Flink + Kafka 实现通用流式数据处理详解

目录 一、前言 二、流式数据处理场景介绍 2.1 流式数据处理概述 2.1.1 流式数据处理场景介绍 2.2 流式数据处理技术栈 2.2.1 数据采集 2.2.2 数据处理 2.2.3 数据存储 2.2.4 数据展示 2.3 流式数据处理场景面临的问题和挑战 三、通用的流式数据处理场景解决方案 3.1…

【Java 学习】深度剖析Java多态:从向上转型到向下转型,解锁动态绑定的奥秘,让代码更优雅灵活

&#x1f4ac; 欢迎讨论&#xff1a;如对文章内容有疑问或见解&#xff0c;欢迎在评论区留言&#xff0c;我需要您的帮助&#xff01; &#x1f44d; 点赞、收藏与分享&#xff1a;如果这篇文章对您有所帮助&#xff0c;请不吝点赞、收藏或分享&#xff0c;谢谢您的支持&#x…

JAVA-Exploit编写(7)--http-request库文件上传使用续篇

目录 1.http-request简介 2. 依赖导入 3.靶场实战 3.1 简单使用 3.2 增加判断机制 3.3 文件上传成功后的利用 3.4 正则匹配上传文件的返回的路径 1.http-request简介 http-request 是一个库 里面提供很多方法&#xff0c;使得很容易就可以构造http请求,相比于之前使用的…

抖音小程序一键获取手机号

前端代码组件 <button v-if"!isFromOrderList"class"get-phone-btn" open-type"getPhoneNumber"getphonenumber"onGetPhoneNumber">一键获取</button>// 获取手机号回调onGetPhoneNumber(e) {var that this tt.login({f…

NavVis手持激光扫描帮助舍弗勒快速打造“数字孪生”工厂-沪敖3D

在全球拥有近100家工厂的舍弗勒&#xff0c;从2016年开启数字化运营进程&#xff0c;而当前制造、库存、劳动力和物流的数字化&#xff0c;已无法支持其进一步简化工作流程&#xff0c;亟需数字化物理制造环境&#xff0c;打造“数字孪生”工厂。 NavVis为其提供NavVis VLX 3…

2024年博客之星主题创作|Android 开发:前沿技术、跨领域融合与就业技能展望

目录 引言 一、推动 Android 应用创新的核心力量 1.1 人工智能与机器学习的崛起 1.2 增强现实&#xff08;AR&#xff09;与虚拟现实&#xff08;VR&#xff09;的应用扩展 1.3 5G技术的推动 1.4 跨平台开发技术的成熟 1.4.1 React Native 1.4.2 Flutter 1.4.3 Taro …

云原生周刊:K8s 生产环境架构设计及成本分析

开源项目推荐 KubeZoneNet KubeZoneNet 旨在帮助监控和优化 Kubernetes 集群中的跨可用区&#xff08;Cross-Zone&#xff09;网络流量。这个项目提供了一种简便的方式来跟踪和分析 Kubernetes 集群中跨不同可用区的通信&#xff0c;帮助用户优化集群的网络架构、提高资源利用…

[Qt] Box Model | 控件样式 | 实现log_in界面

目录 1、样式属性 &#xff08;1&#xff09;盒模型&#xff08;Box Model&#xff09; 2、控件样式示例 &#xff08;1&#xff09;按钮 &#xff08;2&#xff09;复选框 &#xff08;3&#xff09;单选框 &#xff08;4&#xff09;输入框 &#xff08;5&#xff09…

C++并发编程之提高C++多线程应用可测试性的思想和方法

提高C多线程应用的可测试性是一个重要的课题&#xff0c;因为多线程应用程序通常比单线程应用程序更复杂&#xff0c;更容易出现难以复现的并发问题。为了确保多线程应用的可靠性和正确性&#xff0c;可以采用以下思想和方法来提高其可测试性。 1. 模块化设计 将多线程应用分…

Golang Gin系列-6:Gin 高级路由及URL参数

在本章中&#xff0c;我们将深入研究使用Gin框架的高级路由和URL参数。我们将介绍如何创建和使用路由组、应用中间件、提取路径参数、处理查询字符串、处理静态文件以及使用HTML模板。 路由分组 为什么要使用路由组&#xff1f; 使用路由组有助于保持代码结构整洁有序。当路由…

Javascript IndexedDB(Dexie.js) React 存储大量(5-50M)的完整示例

将 IndexedDB 整合到 React 应用中,可以为你的应用提供高效的客户端存储解决方案,特别适用于需要存储大量结构化数据或离线功能的场景。为了简化 IndexedDB 的使用,通常推荐使用封装好的库,如 Dexie.js 或 idb。这些库提供了更简洁和友好的 API,方便在 React 中进行集成。…

android wifi AsyncChannel(WifiManager和WifiP2pManager)

AynscChannel的讲解 [Android]AsyncChannel介绍-CSDN博客 WifiP2pManager里的channel的使用理解 WifiP2pManager.java public void createGroup(Channel c, ActionListener listener) {checkChannel(c);c.mAsyncChannel.sendMessage(CREATE_GROUP, WifiP2pGroup.NETWORK_ID_PE…

【2025】拥抱未来 砥砺前行

2024是怎样的一年 2024在历史画卷上是波澜壮阔的一年&#xff0c;人工智能的浪潮来临&#xff0c;涌现出无数国产大模型。 22年11月ChatGPT发布&#xff0c;它的出现如同在平静湖面上投下一颗巨石&#xff0c;激起了层层波澜&#xff0c;短短五天用户数就达到了100万&#xff0…

FreeRTOS系统移植

前言 学习RTOS之前最重要的就是要学会将系统移植到单片机中&#xff0c;这里可以直接使用cubemx生成移植好的工程&#xff0c;也可以下载库来进行自己移植&#xff0c;这里我选择下载库来自己移植&#xff0c;因为这样可以配合Linux开发stm32单片机程序。 一、下载系统代码 …