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

洛谷[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
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;
 } 

你可能感兴趣的:(暴力枚举)