洛谷[P3799 妖梦拼木棒]{暴力枚举} 奋斗的珂珂~

    技术2025-06-22  15

    洛谷[P3799 妖梦拼木棒] {暴力枚举}

    题目背景

    上道题中,妖梦斩了一地的木棒,现在她想要将木棒拼起来。

    题目描述

    有 n 根木棒,现在从中选 4 根,想要组成一个正三角形,问有几种选法?

    答案对 1 0 9 10^9 109+7 取模。

    输入格式

    第一行一个整数 n。

    第二行 n个整数,第 i 个整数 ai代表第 i 根木棒的长度。

    输出格式

    一行一个整数代表答案。

    输入输出样例

    输入 4 1 1 2 2 输出 1

    说明/提示

    数据规模与约定 对于 30% 的数据,保证 n≤5× 1 0 3 10^3 103。 对于 100% 的数据,保证 1≤n≤ 1 0 5 10^5 105 ,0≤ai ≤5× 1 0 3 10^3 103

    解题思路

    1、首先计算每一个数字的出现个数,有点映射的思想。 2、然后如果一个数字x出现大于等于两次则说明这个数字可能作为正三角形的两条边。 3、然后再假设j是选择的四个数中除相等的两个数之外较小的那个数,则其一定小于等于(x/2),此时另外一个数字就是x-j。然后就进行比遍历,寻找满足条件的结果,计算可能的值。

    注意

    正三角形就是等边三角形。

    取模运算时不要使用+=,要写成正常形式,否则会溢出。

    完整代码

    #include<bits/stdc++.h> using namespace std; const int maxn=1e5+10; const int mod=1e9+7; int a[maxn]; int sum[maxn]; int C(int x,int c) { if(c==1) return x; else return (x*(x-1)/2)%mod; } int main() { int m; int maxa; scanf("%d",&m); for(int i=1;i<=m;i++) { scanf("%d",&a[i]); sum[a[i]]++; maxa=max(maxa,a[i]);//找出最大的那个数,作为之后循环的边界 } int ans=0; for(int i=1;i<=maxa;i++) { if(sum[i]>=2)//如果个数大于两个说明可以选择其中的两个作为正三角形的两边 { int fang=C(sum[i],2); for(int j=1;j<=i/2;j++)//强制使得j是小的那一部分 { if(j!=i-j&&sum[j]>=1&&sum[i-j]>=1) { ans=(ans+fang*C(sum[j],1)*C(sum[i-j],1)%mod)%mod; } if(j==i-j&&sum[j]>=2) { ans=(ans+fang*C(sum[j],2)%mod)%mod;//注意相加不能使用+= } } } } printf("%d",ans); return 0; }
    Processed: 0.014, SQL: 9