UVALive - 6912 Prime Switch (状压DP)

题目链接:传送门

【题意】有n个灯,m个开关,灯的编号从1~n,每个开关上有一个质数,这个开关同时控制编号为这个质数的倍数的灯,问最多有多少灯打开。

【分析】发现小于根号1000的质数有10个左右,然后大于根号1000的质数所控制的灯是不会重叠的,所以我们状压枚举小于31的质数,然后贪心后面的。

#include <cstdio>
#include <map>
#include <algorithm>
#include <vector>
#include <iostream>
#include <set>
#include <queue>
#include <string>
#include <cstdlib>
#include <cstring>
#include <cmath>
#define inf 0x3f3f3f3f
#define met(a,b) memset(a,b,sizeof a)
#define pb push_back
#define mp make_pair
#define rep(i,l,r) for(int i=(l);i<=(r);++i)
#define inf 0x3f3f3f3f
using namespace std;
typedef long long ll;
const int N = 1050;;
const int M = 165;
const int mod = 19260817;
const int mo=123;
const double pi= acos(-1.0);
typedef pair<int,int>pii;
int n,k,cas,cnt;
int a[N],b[20];
int vis[N];
int main(){
    int T;
    scanf("%d",&T);
    while(T--){
        cnt=0;
        scanf("%d%d",&n,&k);
        for(int i=0;i<k;i++){
            scanf("%d",&a[i]);
            if(a[i]<=31)b[cnt++]=a[i];
        }
        int ans=0;
        for(int i=0;i<(1<<cnt);i++){
            int now=0;
            met(vis,0);
            for(int j=0;j<cnt;j++){
                if(i&(1<<j)){
                    for(int s=b[j];s<=n;s+=b[j])vis[s]^=1;
                }
            }
            for(int j=0;j<k;j++){
                if(a[j]<=31)continue;
                int res=0;
                for(int s=a[j];s<=n;s+=a[j]){
                    if(vis[s])res--;
                    else res++;
                }
                if(res<=0)continue;
                for(int s=a[j];s<=n;s+=a[j]){
                    vis[s]^=1;
                }
            }
            for(int j=1;j<=n;j++)if(vis[j])now++;
            ans=max(ans,now);
        }
        printf("Case #%d: %d\n",++cas,ans);
    }
    return 0;
}
/*
4
10 2
2 5
21 4
2 3 5 7
100 1
5
100 3
3 19 7
*/

 

posted @ 2017-10-01 11:54  贱人方  阅读(496)  评论(0编辑  收藏  举报