【dp】luoguP4796 关于图 想不到是状压dp (┬_┬)

luoguP4796

题意:给定一张N个点M条边的无向图,每个点有一个颜色,所有点的颜色共有 K 种,编号为 1…K。求图上有多少条长度至少为 2 的简单路径,满足路径上的每一个点的颜色互不相同。路径上的点的连接顺序不同看作不同的两条路径。

数据范围:1?N,M?100000,1?K?5

一直朝着搜索的角度去思考,找不到什么明确的方法。

看了题解。。知道需要状压,但是想不到是状压dp。。。。。。。。。。。。。我dp果然太弱了(;′⌒`)

做法是:

设每个点的颜色是c[i],则对i这个点有一个起始状态 (1<<c[i]),所以初始化f[i][1<<c[i]]=1,然后就是,状态转移。

洛谷题解说得好:

  “ 

    一种先枚举点,再枚举状态进行转移

    一种先枚举状态,再枚举点进行转移

    前者会发现,某一状态的点可能会因为其它点还未被遍历,导致方案数未统计完全

    比如遍历到点i,点i+1可以转移到i点,但点i+1的状态还没处理出来,导致i+1的方案不能统计到i点上

    所以我们使用后者进行状态转移

  ”

所以 从状态1到状态(1<<k),枚举状态,然后把每个状态转移到连接到的点的状态,这样子必定是 每个状态转移的时候,每个点都考虑到,比枚举点更优。这种思路我很难想到,因为习惯性是枚举点的。

所以有:

for(int sta=1;sta<(1<<k);sta++)
    {
        for(int i=1;i<=n;i++)
        {
            if(f[i][sta])
            {
                if(cal(sta)>1)ans+=f[i][sta];
                for(int j=0;j<p[i].size();j++)
                {
                    if(sta&(1<<c[p[i][j]]))continue;
                    f[p[i][j]][sta|(1<<c[p[i][j]])]+=f[i][sta];
                }
            }
        }
    }

原文地址:https://www.cnblogs.com/kkkek/p/11959369.html

时间: 2024-09-29 15:55:14

【dp】luoguP4796 关于图 想不到是状压dp (┬_┬)的相关文章

[POJ2404]Jogging Trails(中国旅行商问题)(一般图的匹配——状压DP)

题目:http://poj.org/problem?id=2404 题意:有个n(n<=15)的点和m条无向边,每条边都有自己的权值.现在你要从某个点出发,每条边可以经过多次但要保证每条边至少走一次.现在你要找出一个方案,使得经过所有边的权值和最小,输出最小的权值和. 分析: 首先容易想到的是如果这个图G的每个点的度数都为偶数,那么G是欧拉图,那么一定存在欧拉回路,那么ans=∑每条边权值 如果图G不是欧拉图,那么必有偶数个奇度数的点. 如果我们把每条边都看作有无数条的话,那么原问题就是等价于走

状压dp,区间dp,矩阵快速幂

DP 首先先回忆一下dp,dp叫做记忆化搜索,是一种可以把暴力搜索中重复的部分重复利用,从而到达减小复杂度的目的.比如最应该熟悉的背包模型,如果你把选择的过程看成一步一步的,那么在这么多的搜索路径中一定有着很多很多的重复部分,dp就是一种把重复的部分加以利用的方法.相信大家都已经在以前的练习中已经明白了dp是什么样的思路了,接下来的两种dp会在大家已经了解经典的背包dp等模型下展开. 状态压缩dp: 首先先讲一个状压dp最最经典的模型,求哈密尔顿路径问题,也叫做旅行商问题:给你一张图,你要在每个

hdu 4906 状压dp

/* ID: neverchanje PROG: LANG: C++11 */ #include<vector> #include<iostream> #include<cstring> #include<string> #include<algorithm> #include<cmath> #include<cstdio> #include<set> #include<queue> #includ

cf16E Fish(状压DP)

题意: N只FISH.每个回合会有一只FISH吃掉另一个FISH.直到池塘里只剩一只FISH. 给出aij:第i只FISH吃掉第J只FISH的概率. 问每一只FISH是最后存活者的概率. Input The first line contains integer n (1 ≤ n ≤ 18) — the amount of fish in the lake. Then there follow n lines with n real numbers each — matrix a. aij (0

Travel(HDU 4284状压dp)

题意:给n个城市m条路的网图,pp在城市1有一定的钱,想游览这n个城市(包括1),到达一个城市要一定的花费,可以在城市工作赚钱,但前提有工作证(得到有一定的花费),没工作证不能在该城市工作,但可以走,一个城市只能工作一次,问pp是否能游览n个城市回到城市1. 分析:这个题想到杀怪(Survival(ZOJ 2297状压dp) 那个题,也是钱如果小于0就挂了,最后求剩余的最大钱数,先求出最短路和 Hie with the Pie(POJ 3311状压dp) 送披萨那个题相似. #include <

HDU 5765 Bonds(状压DP)

[题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=5765 [题目大意] 给出一张图,求每条边在所有边割集中出现的次数. [题解] 利用状压DP,计算不同的连通块,对于每条边,求出两边的联通块的划分方案数,就是对于该点的答案. [代码] #include <cstdio> #include <algorithm> #include <cstring> using namespace std; int n,m,T,Cas=1

poj1185 状压dp

poj1185   状压dp 炮兵阵地 Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 20940   Accepted: 8104 Description 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原(用"P"表示),如下图.在每一格平原地形上最多可以布置一支炮兵部队(山地上不能够部署炮兵部队

poj185--炮兵阵地(状压dp)

炮兵阵地 Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 20169   Accepted: 7805 Description 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原(用"P"表示),如下图.在每一格平原地形上最多可以布置一支炮兵部队(山地上不能够部署炮兵部队):一支炮兵部队在地图上的攻击

CodeForces 21D Traveling Graph 状压dp+欧拉回路

题目链接:点击打开链接 题意: 给定n个点m条边的无向图 求从1点开始经过每条边至少一次最后回到1点的最小路程 显然就是找一条路径可重复的欧拉回路 思路: 首先对于欧拉回路的结论是:所有点的度数都为偶数 因为所有边至少经过一次,那么可以把题意转换成加最少多少条边使得图满足以上结论 而加的边目的是为了把奇度数转成偶度数,先floyd一下得到任意点间加边的最小花费 dp[i]表示状态i下度数都为偶数的最小花费. 状压dp,把i状态下,所有未选择的点中挑2个奇度数的转移即可. #include <cs