当前位置:首页 > 洛谷 > 洛谷P2789题解:递归算法与避免重复计算的技巧

洛谷P2789题解:递归算法与避免重复计算的技巧

5个月前 (07-11)

洛谷P2789题解:递归算法与避免重复计算的技巧 洛谷题解 递归算法 组合数学 分治思想 C++ 第1张

一、题目解读

洛谷P2789题要求计算n条直线在平面上两两相交产生的交点总数。题目强调交点不重复,需考虑平行线情况。关键点在于如何高效枚举所有可能的交点组合,并排除重复结果。

二、解题思路

采用递归算法,核心思想是“分治+标记”。通过枚举每条线与其他线的平行关系,将问题分解为子问题:当前线与其他线平行或相交。为避免重复统计,使用全局数组A标记已存在的交点数,仅当新交点数未被记录时才累加。

三、解题步骤

1. 输入处理:获取总直线数n,初始化sum=0,A数组全0。

2. 递归函数plan(p,m):

○ 终止条件:p=0时,若m未出现,更新sum并标记A[m]。

○ 递归逻辑:枚举平行线数r(p到1),计算r条平行线与剩余p-r条线的交点数r*(p-r),递归调用plan(p-r, m+r*(p-r))。

3. 主函数:调用plan(n,0),输出sum。

四、代码与注释

#include <iostream>
#include <algorithm>

// 全局变量
int sum = 0;          // 总交点数之和
int p, n, r;          // p为剩余线数,n为总直线数,r为平行线数
bool A[100000] = {0}; // A[i]=1表示i个交点已存在,避免重复统计

// 递归函数:计算交点数
// p:剩余线数(当前待处理的线数)
// m:当前已计算的交点数
void plan(int p, int m) {
    // 终止条件:当剩余线数为0时,记录当前交点数
    if (p == 0) {
        if (A[m] == 0) { // 若该交点数未出现过,计入sum
            sum++;
        }
        A[m] = 1; // 标记为已存在
    } else {
        // 枚举平行线数量r(从p到1)
        for (int r = p; r >= 1; r--) {
            // 计算r条平行线与剩余p-r条线的交点数:r*(p-r)
            int newM = m + r * (p - r);
            // 递归处理剩余p-r条线
            plan(p - r, newM);
        }
    }
}

int main() {
    std::cin >> n; // 输入总直线数n
    plan(n, 0);    // 从n条线开始递归,初始交点数为0
    std::cout << sum << std::endl; // 输出总交点数之和
    return 0;
}

五、总结

本解法巧妙利用递归将复杂枚举转化为子问题求解,结合标记数组高效去重。核心在于理解平行线产生的交点数公式r*(p-r),并通过分治策略逐层递归。时间复杂度受递归深度影响,但通过避免重复计算显著优化结果。该思路对组合数学类问题具有参考价值。


原创内容 转载请注明出处

分享给朋友:

相关文章

力扣第1991题:寻找数组的中心索引 如何找到左右和相等的中心索引

力扣第1991题:寻找数组的中心索引 如何找到左右和相等的中心索引

题目解读给定一个整数数组,我们需要找到一个中心索引,使得该索引左侧所有元素的和等于右侧所有元素的和。如果不存在这样的索引,则返回-1。中心索引的定义不包含在左右两侧的和计算中。这个问题考察对数组遍历和...

【深度优先搜索实战】力扣547题:省份数量问题的图论解法

【深度优先搜索实战】力扣547题:省份数量问题的图论解法

题目解读‌我们面对的是一个典型的图论问题:给定一个城市的连接矩阵,需要计算其中相互连通的城市群(省份)数量。这个问题可以抽象为无向图中的连通分量计算,每个城市代表图中的一个节点,城市之间的连接关系代表...

力扣1137题:动态规划解泰波那契数 高效求解第N项的秘密

力扣1137题:动态规划解泰波那契数 高效求解第N项的秘密

一:重新解读题目泰波那契数列是一个充满数学趣味的递推序列:从第3项开始,每个数均为前三个数的和(即Tₙ₊₃ = Tₙ + Tₙ₊₁ + Tₙ₊₂)。当给定整数n时,需要高效计算出第n项的值。面对此类递...

力扣933题:队列的妙用:如何高效统计最近请求

力扣933题:队列的妙用:如何高效统计最近请求

题目重解:我们需要设计一个能统计最近3000毫秒内请求次数的系统。每当新的请求到来时,它会带有时间戳t,我们需要返回过去3000毫秒内(包括当前)发生的请求总数。这就像是在时间轴上维护一个滑动窗口,只...

牛客14496题解:括号最大深度问题(栈思想与代码优化)

牛客14496题解:括号最大深度问题(栈思想与代码优化)

一、题目解读牛客14496题要求计算给定括号字符串中的最大深度。例如,对于字符串 "(()())",最大深度为2。题目考察对括号嵌套结构的理解,以及如何通过编程找到最深嵌套层次。二...

力扣3112题解法:带时间限制的最短路径问题解析(C++代码)

力扣3112题解法:带时间限制的最短路径问题解析(C++代码)

一、题目解读力扣3112题要求解决带时间限制的最短路径问题:给定一个有向图,节点具有消失时间,需计算从起点到各节点的最短路径,且路径总时间不能超过节点的消失时间。题目难点在于需在传统最短路径算法(如D...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。