• 【NOI2015】【寿司晚宴】【状压DP】


    Description

    为了庆祝 NOI 的成功开幕,主办方为大家准备了一场寿司晚宴。

    小 G 和小 W 作为參加 NOI 的选手,也被邀请參加了寿司晚宴。

    在晚宴上,主办方为大家提供了 n−1 种不同的寿司,编号 1,2,3,…,n−1。当中第 i 种寿司的美味度为 i+1 (即寿司的美味度为从 2 到 n)。
    如今小 G 和小 W 希望每人选一些寿司种类来品尝,他们规定一种品尝方案为不和谐的当且仅当:小 G 品尝的寿司种类中存在一种美味度为 x 的寿司,小 W 品尝的寿司中存在一种美味度为 y 的寿司,而 x 与 y 不互质。
    如今小 G 和小 W 希望统计一共同拥有多少种和谐的品尝寿司的方案(对给定的正整数 p 取模)。注意一个人能够不吃不论什么寿司。

    Input

    输入文件的第 1 行包括 2 个正整数 n,p。中间用单个空格隔开,表示共同拥有 n 种寿司。终于和谐的方案数要对 p 取模。

    Output

    输出一行包括 1 个整数,表示所求的方案模 p 的结果。

    Sample Input

    3 10000

    Sample Output

    9

    HINT

     2≤n≤500


    0<p≤1000000000
    题解:能够发现,选了一个数等于是选了它的质因子。首先n仅仅有500,所以小于根号500的质因子仅仅有8个。

    我们能够把这8个质因子压成二进制位。形成2^8个集合。

    对于每一个数。仅仅可能含有不超过1个大于根号500的质因子,我们按这个将每一个数分类。

    把每一个数存到一个结构体s里,s.kind表示这个数里大于根号500的质因子是什么。没有的话为1。s.se表示这个数小于根号500的质因子的包括情况,用二进制位压一下。

    然后我们按kind排序。dp就可以。
    dp的时候把kind同样的放在一起dp。

    设f[i][j]表示第一个人选了i这个质因数集合,第二个人选了j这个质因数集合的方案数。
    对于每一类数開始dp时。

    先把f数组复制两遍到p[1]和p[2];

    p[i][j][k]表示当前是第i个人进行操作。第1个人选的集合是j第二个人是k的方案数。
    用p数组进行更新。

    dp结束后要用p数组更新f数组。f=p1+p2-f; 由于两个p数组中都包括了选当前数的情况,然而这显然是不能够的,所以要减去之前的f。
    #include<iostream>
    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    int f[301][301],p[3][301][301],pp,ans;
    int prime[8]={2,3,5,7,11,13,17,19},n;
    struct use{
    	int kind,se;
    }s[600];
    bool cmp(use a,use b)
    {
    	if (a.kind!=b.kind) return a.kind<b.kind;
    	else return a.se<b.se;
    }
    int main()
    {
        freopen("dinner.in","r",stdin);
        freopen("dinner.out","w",stdout);
    	scanf("%d%d",&n,&pp);
        for (int i=1;i<=n;i++)
          {
            int temp;
            temp=i;
            for (int j=0;j<8;j++)
             if (temp%prime[j]==0)
               {
                 s[i].se|=1<<j;
    			 while (temp%prime[j]==0) temp/=prime[j];
               }
             s[i].kind=temp;  
          }
        sort(s+2,s+n+1,cmp);
        f[0][0]=1;
    	for (int i=2;i<=n;i++)
         {
         	if (i==2||s[i].kind==1||s[i].kind!=s[i-1].kind)
         	  {
         	  	 memcpy(p[1],f,sizeof f );
         	  	 memcpy(p[2],f,sizeof f );
         	  }
         	 for (int j=255;j>=0;j--)
         	   for (int k=255;k>=0;k--)
         	     {
         	     	if ((k&s[i].se)==0) p[1][j|s[i].se][k]=(p[1][j|s[i].se][k]+p[1][j][k])%pp;
         	        if ((j&s[i].se)==0) p[2][j][k|s[i].se]=(p[2][j][k|s[i].se]+p[2][j][k])%pp;
    			 }
           if (i==n||s[i].kind==1||s[i].kind!=s[i+1].kind)
             {
             	for (int j=0;j<=255;j++)
             	  for (int k=0;k<=255;k++)
             	    f[j][k]=((p[1][j][k]+p[2][j][k]-f[j][k])%pp+pp)%pp;
             }
    	 }
    	 ans=0;
    	 for (int i=0;i<=255;i++)
    	   for (int j=0;j<=255;j++)
    	     if ((i&j)==0) ans=(ans+f[i][j])%pp;
    	cout<<ans<<endl;
    }
  • 相关阅读:
    微信小程序promise解决onload异步
    小程序中使用 Less (VScode)
    Vue中使用less
    小程序获取 图片宽高
    ssh 登录出现Are you sure you want to continue connecting (yes/no)?解决方法
    SQL SERVER 收缩日志
    SQL SERVER-日期时间
    oracle判断查询结果是否为空
    修改Tomcat默认JDK版本
    Microsoft SQL Server Management Studio ------- 附加数据库 对于 服务器“xxx&amp;amp;quot;失败(错误码5120)
  • 原文地址:https://www.cnblogs.com/jhcelue/p/6797196.html
Copyright © 2020-2023  润新知