HDU 4466 Triangle(计数)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4466

题意:给出一根长度为n的铁丝。将其分成若干段并将每段折成一个三角形,使得三角形都相似。有多少种分法?

思路:首先计算周长为M的三角形有多少个,设为f[M]。设三角形的三边a,b,c满足a<=b<=c,那么分两种情况:

(1)b=c,此时c的上限为(M-1)/2,下限为ceil(M/3)=(M+2)/3,所以此时的三角形个数为(M-1)/2-(M+2)/3+1;

(2)b!=c,那么b<=c-1,因为a+b>c>c-1,因此一般来说有多少个三角形(a,b,c-1)就有多少个三角形(a,b,c),但是此时要减去a+b=c的情况。三角形(a,b,c-1)的个数就是f[M-1]。此时若a+b=c,即M-1=a+b+c-1=c+c-1,即M=2c,因此M必须为偶数。a+b=c=M/2,使得a+b=M/2的有序(a<=b)二元组有M/2/2。

根据这两个可以得到计算f的递推关系。下面我们看怎么得到周长为M且三边满足Gcd(a,b,c)=1的三角形个数?这个可用于筛素数类似的方法得到。那么对于n,将其分成若干份,每份相等,那么不同的分法就是隔板法。

 

int f[N],Pow2[N];

void init()
{
    int i,j;
    f[3]=1;
    for(i=4;i<N;i++)
    {
        f[i]=f[i-1]+(i-1)/2-(i+2)/3+1;
        f[i]%=mod;
        if(i%2==0) (f[i]-=i/2/2)%=mod;
        if(f[i]<0) f[i]+=mod;
    }
    Pow2[0]=1;
    for(i=1;i<N;i++) Pow2[i]=Pow2[i-1]*2%mod;
    for(i=3;i<N;i++) for(j=2;j*i<N;j++)
    {
        (f[j*i]-=f[i])%=mod;
        if(f[i]<0) f[i]+=mod;
    }
}

int n;

int main()
{
    init();
    int num=0;
    Rush(n)
    {
        i64 ans=0,i;
        for(i=1;i*i<=n;i++) if(n%i==0)
        {
            ans+=(i64)f[i]*Pow2[n/i-1];
            if(i*i!=n) ans+=(i64)f[n/i]*Pow2[i-1];
            ans%=mod;
        }
        if(ans<0) ans+=mod;
        printf("Case %d: %I64d\n",++num,ans);
    }
}

  

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值