为了使这条路的环境更加优美,政府想把这些雕塑分成若干组。并通过在组与组之间种上一些树。来吸引很多其它的游客来巴厘岛。
同一组中的全部雕塑必须位于这条路的连续一段上。
当中:
标签:api 大小 二进制 i++ problems lin lib data memset
为了使这条路的环境更加优美,政府想把这些雕塑分成若干组。并通过在组与组之间种上一些树。来吸引很多其它的游客来巴厘岛。
同一组中的全部雕塑必须位于这条路的连续一段上。
当中:
输入的第一行包括三个用空格分开的整数 N,A,B。
输出一行一个数,表示最小的终于优美度。
)
子任务 1 (9 分)
贪心+DP
要使终于的答案最小,能够直观产生一种贪心的想法,从高到低枚举答案的每一位。假设能取0则取0。否则取1。
然后主要问题转化为怎样推断终于答案的某一位是否能取0。当然要保证前面全部位不变的前提下。
我们考虑用DP解决问题。
如果当前枚举到第pos位。令f[i][j]表示前i个数分成j组,满足前pos-1位,当前这一位是否能填0。
则f[i][j]=true当且仅当存在k满足f[k][j-1]=true且(sum[i]-sum[k])|ans==ans且(sum[i]-sum[k])&(1<<pos-1)==0。然后推断f[n][i]中是否有等于true的项。a≤i≤b。
可是这个复杂度是O(n^3logM),对于最后一组数据会TLE。
考虑到最后一组数据的特殊性:a等于1,也就是组数没有下界。
所以我们能够去掉DP的第二维,即用g[i]表示前i个数满足条件的最少组数。然后推断g[n]和b的大小就能够了。
注意1<<pos-1要写成1ll<<pos-1。
#include<iostream> #include<cstdlib> #include<cmath> #include<cstdio> #include<cstring> #include<algorithm> #define F(i,j,n) for(int i=j;i<=n;i++) #define D(i,j,n) for(int i=j;i>=n;i--) #define ll long long #define maxn 2005 #define inf 1000000000 using namespace std; int n,a,b,len,g[maxn]; ll ans,sum[maxn]; bool f[maxn][maxn]; inline int read() { int x=0,f=1;char ch=getchar(); while (ch<'0'||ch>'9'){if (ch=='-') f=-1;ch=getchar();} while (ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();} return x*f; } inline void solve1() { D(pos,len,1) { memset(f,false,sizeof(f)); f[0][0]=true; F(i,1,n) F(j,1,i) F(k,j-1,i-1) if (f[k][j-1]) { ll tmp=sum[i]-sum[k]; if (((tmp>>pos)|ans)==ans&&(tmp&(1ll<<(pos-1)))==0){f[i][j]=true;break;} } bool flag=false; F(i,a,b) if (f[n][i]){flag=true;break;} ans<<=1; if (!flag) ans|=1; } } inline void solve2() { D(pos,len,1) { F(i,1,n) g[i]=inf; F(i,1,n) F(j,0,i-1) { ll tmp=sum[i]-sum[j]; if (((tmp>>pos)|ans)==ans&&(tmp&(1ll<<(pos-1)))==0) g[i]=min(g[i],g[j]+1); } ans<<=1; if (g[n]>b) ans|=1; } } int main() { n=read();a=read();b=read(); F(i,1,n) sum[i]=sum[i-1]+read(); for(ll tmp=sum[n];tmp;tmp>>=1) len++; if (a!=1) solve1(); else solve2(); printf("%lld\n",ans); return 0; }
标签:api 大小 二进制 i++ problems lin lib data memset
原文地址:http://www.cnblogs.com/wzjhoutai/p/7241429.html