Timus 1515

系统 1136 0
      
        #include 
      
      
        <
      
      
        iostream
      
      
        >
      
      
        
using namespace std;

#define MAX 10000

int origin[ 101 ] = { 0 };

typedef
struct range_st {
int l,r;
} range_st,
* range_t;

int ranges_len = 0 ;
range_st ranges[MAX];
range_st temp[MAX];

void union_range(range_st rg) {
int i,j,union_count;
for (i = 0 ;i < ranges_len && ranges[i].r + 1 < rg.l;i ++ ) ; // find the first range that can union rg
if (i == ranges_len) // no such range found
ranges[ranges_len ++ ] = rg;
else if (ranges[i].r < rg.r) {
ranges[i].r
= rg.r;
}

union_count
= 0 ;
for (j = i + 1 ;j < ranges_len;j ++ ) {
if (ranges[i].r + 1 >= ranges[j].l) { // self-union occur
if (ranges[i].r < ranges[j].r)
ranges[i].r
= ranges[j].r;

union_count
++ ;
}
}

ranges_len
-= union_count;
}

void update_range( int n) {
int temp_len = 0 ;
range_st rg;
for ( int i = 0 ;i < ranges_len;i ++ ) {
rg.l
= ranges[i].l + n;
rg.r
= ranges[i].r + n;
temp[temp_len
++ ] = rg;
}

for ( int i = 0 ;i < temp_len;i ++ )
union_range(temp[i]);
}

void print_range() {
for ( int i = 0 ;i < ranges_len;i ++ )
printf(
" (%d,%d) " , ranges[i].l, ranges[i].r);
printf(
" \n " );
}

int main() {

int i,j,k;
int m,n,d,t,ret;
range_st rg;
int N;
cin
>> N;
for (i = 0 ;i < N;i ++ )
cin
>> origin[i];

rg.l
= rg.r = 0 ;
ranges[ranges_len
++ ] = rg; // init range (0,0)

for (i = 0 ;i < N;i ++ ) {
n
= origin[i];

update_range(n);

// print_range();

if (ranges_len > 1 )
break ;
}

ret
= ranges[ 0 ].r + 1 ;

cout
<< ret << endl;

return 0 ;
}

囧死的一题目,给出N个数(N<=100),求一个最小的数,这个数不能是这N个数的任何组合的求和数。

暴力的思维让我去计算所以组合数,根据前i个数生成的所有和数,去计算第i+1个数能够生成的和数,然后把这两堆和数做合并,这可以正确地求出所有可能的和数,但就死活ME,因为数量太大了。注意给出的N个数,每个数最大值是10^6。

在使用第i个数的时候,其实就可以从集合中遍历,查看是否存在一个数少于Ni并且不在集合中,如果是,那么这个就是答案,但写的代码过不了,一直WA 3。

后来换了个思维,通过在草稿上写了些例子,认为这题目应该有很高效的计算方法才是,结果就得出了最后AC的代码。属于0开销代码。

作出的改变是把生成的和数集合中,连续的和数表示成范围,这样处理数据的数量级就大减,并且当发现存在两个不连续的范围后,就能马上得出答案。

Timus 1515


更多文章、技术交流、商务合作、联系博主

微信扫码或搜索:z360901061

微信扫一扫加我为好友

QQ号联系: 360901061

您的支持是博主写作最大的动力,如果您喜欢我的文章,感觉我的文章对您有帮助,请用微信扫描下面二维码支持博主2元、5元、10元、20元等您想捐的金额吧,狠狠点击下面给点支持吧,站长非常感激您!手机微信长按不能支付解决办法:请将微信支付二维码保存到相册,切换到微信,然后点击微信右上角扫一扫功能,选择支付二维码完成支付。

【本文对您有帮助就好】

您的支持是博主写作最大的动力,如果您喜欢我的文章,感觉我的文章对您有帮助,请用微信扫描上面二维码支持博主2元、5元、10元、自定义金额等您想捐的金额吧,站长会非常 感谢您的哦!!!

发表我的评论
最新评论 总共0条评论