(new online judge)1322蓝桥杯2017初赛 包子凑数

P1322 - [蓝桥杯2017初赛]包子凑数 - New Online Judge 


题目分两步:(1)判断结果是否为INF;(2)如果不是INF,统计数量。考点是“数论gcd+简单DP”。

1. 什么时候答案不是INF

什么时候答案不是INF?也就是说,除了少数一些整数无法组合得到,其他所有的整数都能得到。

首先看2个数a、b的情况,结论是:若gcd(a,b)!=1,则答案不是INF。

以两个数4、5为例,任取x个4和y个5(x,y≥0),它们能组合得到的数是:

4x+5y=c

若把c看成常数,这是一个二元一次方程。

二元一次方程ax+by=c有整数解的定理:设a,b是整数且gcd(a,b)=d,如果d不能整除c,那么方程ax+by=c没有整数解,如果d能整除c,那么存在无穷多个整数解。

显然,如果gcd(a,b)=1,由于1能整除所有整数,此时ax+by能得到所有整数。
在例子4x+5y=c中,因为gcd(4,5)=1,那么不管c是什么整数,都存在整数解x、y ,也就是说答案不是INF。

不过,x、y的解可能是负整数,而本题要求x、y是非负整数。

下面证明:当c很大时,肯定有x、y的非负整数解。

当gcd(a,b)=1时,ax+by=c的通解是:

x=x0+bn
y=y0−an

其中n是任意整数,而x0、y0是一个特解,它显然满足:ax0+by0=c

两式分别乘以a、b,得:

ax=ax0+abn
by=by0−abn

取n是一个正整数,有ax=ax0+abn≥0,即x是非负的。而:

by=c−ax0−abn

当c很大时,by也是非负的,即y是非负的。

以上证明了c很大时,存在x、y的非负整数解。

以上讨论了给定2个数的情况,若给定多个数a、b、e、f…可以推导出结论:gcd(a,b,e,f, . . . )=1时,答案是非INF。

2. 统计

很简单。例如a[0],把它所有的倍数i=a[0]、2*a[0]、3*a[0]、. . . . . .都算一遍。a[1]、a[2]…也一样。

用dp[ i ]=1表示第i个整数被计算出来了,最后统计没有被算过的dp[ i ] 的个数即可。

代码:

#include <bits/stdc++.h>
using namespace std;
int ans,t,dp[100001],a[100001],n;
int main()
{
  cin>>n;
  for(int i = 0; i < n; i++) cin>>a[i];
  t = a[0];
  for(int i = 1; i < n; i++) t = __gcd(t,a[i]);
  if(t != 1)
  {
    cout<<"INF";
    return 0;
  }
  for(int i = 0; i < n; i++)
  {
    dp[a[i]] = 1;
    for(int j = 1; j <= 10000; j++)
      if(dp[j] == 1)
        dp[a[i] + j] = 1;
  }
  for(int i = 1; i <= 10000; i++)
    if(dp[i] == 0)
      ans++;
  cout<<ans;
  return 0;
}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值