[洛谷 3448][POI2006]MIS-Teddies

题目链接

https://www.luogu.com.cn/problem/P3448

题解

细节不少的 DP 题。

我们设 \(f(i,j,k,l,m,n)\) 表示使用 \(A1,A2,B1,B2\) 型玩具的个数分别为 \(i,j,k,l\),且最后两个玩具类型分别为 \(m,n\) 时的合法方案数。

转移时只需枚举接下来放哪个玩具即可。

然而直接设状态空间会超过空间限制。注意到第一维可以滚动,我们就可以借此优化空间。

#include <cstdio>
#define MOD 1000000
int f[2][45][45][45][5][5];
int a[5];
bool check(int i,int j,int k,int l,int m,int n)
{
 static int tmp[5];
 tmp[0]=i,tmp[1]=j,tmp[2]=k,tmp[3]=l;
 for(int i=0;i<=3;i++)
  if(tmp[i]==0&&(m==i||n==i))return false;
 return true;
}
bool get(int x,int y)
{
 return x&(1<<y);
}
bool check(int i,int j,int k,int l,int m,int n,int o)
{
 static int tmp[5];
 tmp[0]=i,tmp[1]=j,tmp[2]=k,tmp[3]=l;
 tmp[o]++;
 for(int i=0;i<=3;i++)
  if(tmp[i]>a[i])return false;
 for(int i=0;i<=1;i++)
  if(get(m,i)==get(n,i)&&get(n,i)==get(o,i)&&m!=4)return false;
 return true;
}
void calc(int i,int j,int k,int l,int m,int n,int o)
{
 static int tmp[5];
 tmp[0]=i,tmp[1]=j,tmp[2]=k,tmp[3]=l;
 int res=f[i&1][j][k][l][m][n];
 tmp[o]++;
 int&p=f[tmp[0]&1][tmp[1]][tmp[2]][tmp[3]][n][o];
 p+=res;
 if(p>=MOD)p-=MOD;
}
void clear(int i)
{
 for(int j=0;j<=a[1];j++)
  for(int k=0;k<=a[2];k++)
   for(int l=0;l<=a[3];l++)
    for(int m=0;m<=4;m++)
     for(int n=0;n<=4;n++)
      f[i][j][k][l][m][n]=0;
}
int main()
{
 for(int i=0;i<=3;i++)
  scanf("%d",&a[i]);
 f[0][0][0][0][4][4]=1;
 for(int i=0;i<=a[0];i++)
 {
  clear(!(i&1));
  for(int j=0;j<=a[1];j++)
   for(int k=0;k<=a[2];k++)
    for(int l=0;l<=a[3];l++)
     for(int m=0;m<=4;m++)
      for(int n=0;n<=4;n++)
      {
       int p=i+j+k+l;
       if((p==0&&(m!=4||n!=4))||(p==1&&m!=4))continue;
       if(p>=2&&(m==4||n==4))continue;
       if(!check(i,j,k,l,m,n))continue;
       for(int o=0;o<=3;o++)
        if(check(i,j,k,l,m,n,o))
         calc(i,j,k,l,m,n,o);
      }
 }
 int ans=0;
 for(int i=0;i<=4;i++)
  for(int j=0;j<=4;j++)
   ans+=f[a[0]&1][a[1]][a[2]][a[3]][i][j];
 printf("%d\n",ans%MOD);
 return 0;
}

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注

此站点使用Akismet来减少垃圾评论。了解我们如何处理您的评论数据