首页 > 代码库 > usaco 2.2.4 生日派对灯(最近写题碰到的,虽然知道现在写这个有点晚了)

usaco 2.2.4 生日派对灯(最近写题碰到的,虽然知道现在写这个有点晚了)

经过分析,他看似很多的开灯的方法其实合并起来就只有八个。

首先,一个开关在执行的时候只能按一次(因为你就算按了两次就相当于一次也没有按)。

当一个都不按的时候  当然就只有一种:不按。

当按一下的时候:很明显(1)(2)(3)(4)四种。

当按两下的时候:(1,2)(1,3)(1,4)(2,3)(2,4)(3,4)。 然而(1,2)就相当于是3;(1,3)=2;(2,3)=1;所以按两下的就只有(1,4)(2,4)(3,4)。

当按三下的时候:(1,2,3)(1,2,4)(1,3,4)(2,3,4)....。然而都可以相当于 空(3,4)(2,4)(1,4)也就是什么都没有了 QWQ;

按四下:应该就可以发现其实也是能和前面的合并;

所以所有的策略中就只有8中了:空 (1)(2)(3)(4)(1,4)(2,4)(3,4)。

这就已经把按灯的套路合并完了。

-----------------------------接下来是计数器c的情况的分析----------------------------------------------------

当c==0的时候,很明显:不按。

当c==1的时候,也很明显:(1)(2)(3)(4) 四种。

当c==2的时候,就有七种情况了:(1)[按2和3合并为1((2)(3)的情况一样)](2)(3)(1,4)(2,4)(3,4)不按(这里没有(4),因为无论如何按两下都不可能把(4)给按出来)。

当c==3的时候,所有情况都可以了(你问我为什么??像第二步一样了)。

 

所以这道题很容易的就搞定了。

#include<iostream>
#include<cstdio>
#include<ctime>
#include<algorithm>
using namespace std;
int n,c,x;
int a[150],b[150],d[150];
long long q[150];
int t=0;
void init()
{
    cin>>n>>c;
    while((cin>>x)&&(x!=-1))
        a[x]=1;
    while((cin>>x)&&(x!=-1))
        b[x]=1;
}
void haha()
{
    for(int i=1;i<=n;i++)
        d[i]=1;
}
bool check()
{
    for(int i=1;i<=n;i++)
    {
        if((a[i])&&(!d[i])) return 0;
        if((b[i])&&(d[i]))  return 0;
    }
    return 1;
}
void print()
{
    for(int i=1;i<=n;i++)
        cout<<d[i];
    cout<<endl; t++;
}
void doit1()
{
    for(int i=1;i<=n;i++)
        d[i]=(d[i])?0:1;
}
void doit2()
{
    for(int i=1;i<=n;i+=2)
        d[i]=(d[i])?0:1;
}
void doit3()
{
    for(int i=2;i<=n;i+=2)
        d[i]=(d[i])?0:1;
}
void doit4()
{
    for(int i=1;i<=n;i+=3)
        d[i]=(d[i])?0:1;
}
int main()
{
        freopen("add.in","r",stdin);
    freopen("add.out","w",stdout);
    init();
    haha();
    switch(c)
    {
        case 0:
            if(check()) print();
            break;
        case 1:
            haha(); doit1();
            if(check()) print();
            haha(); doit2();
            if(check()) print();
            haha(); doit3();
            if(check()) print();
            haha(); doit4();
            if(check()) print();
            break;
        case 2:
            haha(); doit1();
            if(check()) print();
            haha(); doit3(); doit4();
            if(check()) print();
            haha(); doit2();
            if(check()) print();
            haha(); doit1(); doit4();
            if(check()) print();
            haha(); doit3();
            if(check()) print();
            haha(); doit2(); doit4();
            if(check()) print();
            haha();
            if(check()) print();
            break;
        default:
            haha(); doit1();
            if(check()) print();
            haha(); doit3(); doit4();
            if(check()) print();
            haha(); doit2();
            if(check()) print();
            haha();doit4();
            if(check()) print();
            haha(); doit1(); doit4();
            if(check()) print();
            haha(); doit3();
            if(check()) print();
            haha(); doit2(); doit4();
            if(check()) print();
            haha();
            if(check()) print();
            break;
    }
    if(t==0){ 
        cout<<"IMPOSSIBLE"<<endl;
    }
    return 0;
}
    

usaco 2.2.4 生日派对灯(最近写题碰到的,虽然知道现在写这个有点晚了)