非常好的dp,非常考dp的能力

很显然是个计数问题,那么很显然要么是排列组合,要么是递推,这道题很显然递推的面更大一些。

那么我们来设计一下状态:

设状态f[i][j][k][p]表示目前到了第i个点,这i个点中有j个白点是奇数条好的路径的结尾,k个黑点是奇数条好的路径的结尾,p个白点是偶数条好的路径的结尾的方案数

可能这个状态本身不是特别好懂,我们详细解释一下:

这样的图的个数会取决于好的路径的条数,而好的路径的条数又可以分成两类:以黑点为结尾和以白点为结尾

那么对于每一个黑点或白点,他只有两种可能:有奇数条好的路径在这结尾和有偶数条好的路径在这结尾。

这样我们就将整个图的点分为了4类:白点是奇数条好的路径的结尾,黑点是奇数条好的路径的结尾,白点是偶数条好的路径的结尾,黑点是偶数条好的路径的结尾。

我们将其中三种扔进状态里,而第四个就可以算出来。

接着我们考虑转移:

分类转移:如果一个节点是白色,那么一个合法的路径一定会从上一个黑点转过来。

而且如果想要影响这个白点奇偶性,上面一定会连上一个奇数黑点,因为偶数黑点对这个白点的奇偶性没有影响。

我们讨论这个白点是奇数白点还是偶数白点:如果这个白点是奇数白点,那么从上面的奇数黑点转过来时,要连1个,3个...奇数个,这样才能保证这个白点是奇数的。

那么如果设之前奇数黑点的数量为k,总方案数即为

利用这一点,乘原来方案数转移即可

那么这个白点是偶数白点的转移也同理,总方案数即为

同理转移即可

最后,考虑其他点:由于别的点对这个白点的奇偶性没有影响,所以对于别的点可以随便连,也就是2^别的点的个数,乘这个东西即可

那么黑点也就同理了。

贴代码:

#include <cstdio>
#include <cmath>
#include <cstring>
#include <cstdlib>
#include <iostream>
#include <algorithm>
#include <queue>
#include <stack>
#define ll long long
#define mode 1000000007
using namespace std;
ll dp[55][55][55][55];//到了第几个点,奇数白点,奇数黑点,偶数白点
int n,typ;
int c[55];
ll mul[55];
int main()
{
scanf("%d%d",&n,&typ);
mul[0]=1;
for(int i=1;i<=n;i++)
{
scanf("%d",&c[i]);//0黑1白-1任意
mul[i]=(mul[i-1]<<1)%mode;
}
ll ret=0;
if(c[1]==0||c[1]==-1)
{
dp[1][0][1][0]=1;
if(typ==1&&n==1)
{
ret++;
}
}
if(c[1]==1||c[1]==-1)
{
dp[1][1][0][0]=1;
if(typ==1&&n==1)
{
ret++;
}
}
for(int i=2;i<=n;i++)
{
for(int j=0;j<=i;j++)//奇数白
{
for(int k=0;k+j<=i;k++)//奇数黑
{
for(int p=0;j+k+p<=i;p++)//偶数白
{
int t=i-j-k-p;//偶数黑
if(c[i]==1||c[i]==-1)
{
ll s=0;
if(j+p)
{
if(j)
{
if(!k)
{
s+=dp[i-1][j-1][k][p];
}else
{
s+=mul[k-1]*dp[i-1][j-1][k][p]%mode;
}
s%=mode;
}
if(p)
{
if(k)
{
s+=mul[k-1]*dp[i-1][j][k][p-1]%mode;
}
s%=mode;
}
}
s*=mul[j+p+t-1];
s%=mode;
dp[i][j][k][p]+=s;
dp[i][j][k][p]%=mode;
}
if(c[i]==0||c[i]==-1)
{
ll s=0;
if(k+t)
{
if(k)
{
if(!j)
{
s+=dp[i-1][j][k-1][p];
}else
{
s+=mul[j-1]*dp[i-1][j][k-1][p]%mode;
}
s%=mode;
}
if(t)
{
if(j)
{
s+=mul[j-1]*dp[i-1][j][k][p]%mode;
}
s%=mode;
}
}
s*=mul[k+p+t-1];
s%=mode;
dp[i][j][k][p]+=s;
dp[i][j][k][p]%=mode;
}
if(i==n)
{
if((j+k)%2==typ)
{
ret+=dp[i][j][k][p];
ret%=mode;
}
}
}
}
}
}
printf("%lld\n",ret);
return 0;
}

CF979E的更多相关文章

  1. ZR8.2 DP

    DP 1CF1101D 我们发现,最终答案一定和质因数有关 我们发现\(w_i <= 2*10^5\)级别的树,他的素因子的个数不会非常多(\(<=10\)) 然后我们就设 gcd是\(d ...

随机推荐

  1. 【.net ajax显示后台返回值】

    1..net ajax显示后台返回值 <script>        $(document).ready(function () {            $("#btn&quo ...

  2. 遇到以前跑一次却没问题的问题,直接maven install 再跑

    遇到以前跑一次却没问题的问题,直接maven install 再用tomcat 跑 比如,xml 路径找不到类,奇葩的报错啊

  3. python正则下载图片

    import urllib.request import re # 打开图片路径 def open_url(url): #设置请求路径 req = urllib.request.Request(url ...

  4. Java 进制间的转换

    package com.touch.onlinedu; public class Test { public static void main(String[] args) { // 1 : 0001 ...

  5. Kafka监控KafkaOffsetMonitor【转】

    1.概述 前面给大家介绍了Kafka的背景以及一些应用场景,并附带上演示了Kafka的简单示例.然后,在开发的过程当中,我们会发现一些问题,那就是消息的监控情况.虽然,在启动Kafka的相关服务后,我 ...

  6. requests库入门04-http基本认证

    因为后续样例中GitHub都需要提供认证,所以先写关于基本认证的 http的请求中,有一些请求是需要通过授权认证之后才会响应,授权认证就是检查用户名和密码的过程.http有一个基本认证方式,在认证的过 ...

  7. JQuery 拾遗

    jquery基本上依赖百度,不熟悉的jquery操作记录于此: 1.判断元素的显示隐藏:$("#XXX").is(':visible'). 2. jquery取所有属性以什么开头 ...

  8. PHP操作MongoDB 数据库

    最近有个项目,需要用php操作mongoDb数据,整理如下 1,连接MongoDB数据库 $conn = new Mongo(); 其他链接方式 //$conn=new Mongo(); #连接本地主 ...

  9. Threading.local

    在多线程环境下,每个线程都有自己的数据.一个线程使用自己的局部变量比使用全局变量好,因为局部变量只有线程自己能看见,不会影响其他线程,而全局变量的修改必须加锁. Threading.local可以创建 ...

  10. 题解 SP26045 【GCDMAT2 - GCD OF MATRIX (hard)】

    承接一下洛咕上的题解,这里基本就是谈谈优化,放个代码的 我们发现这里的常数主要来自于除法,那么我们优化除法次数,把所有的 \(n/1...n/s\) (\(s=\sqrt n\))存下来,然后归并排( ...