状压dp专题复习

(有些题过于水,我直接跳了)

技巧总结 :

1.矩阵状压上一行的选择情况 \(n * 2^n\)

D [BZOJ2734][HNOI2012]集合选数

蒻得不行的我觉得这是一道比较难的题,以至于我卡了很久

可以看出,所有会互相直接造成影响的数之间构成一张\(DAG\),边就是\(i->i*2,i->i*3\)

取出每一个连通块之后,就是一个独立集个数的问题

\(DAG\)还可以求独立集?

我们其实可以惊人得发现,这张\(DAG\)过于整齐,就是一个网格图,就是一张网格图上相邻的点不能取的问题

这个,状压矩阵即可


const int N=1e5+10,P=1e9+1; int n;
ll dp[20][1<<11];
int A[20]; ll Solve(int i){
int t=0;
A[0]=0;
dp[0][0]=1;
for(;i<=n;i*=2) {//网格图的列数
t++;
A[t]=0;
int c=0;
for(int j=i;j<=n;j*=3) A[t]|=1<<(c++); //取出网格图这一行的大小
rep(j,0,A[t]) dp[t][j]=0;
rep(S1,0,A[t-1]) {
int fl=1;
rep(j,0,c+1) if((S1&(1<<j)) && (S1&(1<<(j+1)))) fl=0;
if(!fl) continue;
int R=A[t]^(S1&A[t]);
for(reg int S2=R;;S2=(S2-1)&R) {
int fl=1;
rep(j,0,c-1) if((S2&(1<<j)) && (S2&(1<<(j+1)))) fl=0;
if(!fl) {
if(!S2) break;
continue;
}
(dp[t][S2]+=dp[t-1][S1])%=P;
if(!S2) break;
}
}
}
ll res=0;
rep(i,0,A[t]) res+=dp[t][i];
res%=P;
return res;
} int main(){
n=rd();
ll ans=1;
rep(i,1,n) {
if(i%2==0||i%3==0) continue;
//这是一个联通块
ans=ans*Solve(i)%P;
}
printf("%lld\n",ans);
}

\[\
\]

\[\
\]

G [BZOJ1097] [POI2007]旅游景点atr

预处理前面k个点之间的dis,然后就像是一个TSP一样,但是有限制

int d[30][30];
#include<bits/stdc++.h>
using namespace std; #define reg register
//typedef long long ll;
typedef int ll;
#define rep(i,a,b) for(reg int i=a,i##end=b;i<=i##end;++i)
#define drep(i,a,b) for(reg int i=a,i##end=b;i>=i##end;--i) char IO; int rd(){
int s=0,f=0;
while(!isdigit(IO=getchar())) if(IO=='-') f=1;
do s=(s<<1)+(s<<3)+(IO^'0');
while(isdigit(IO=getchar()));
return f?-s:s;
} const int N=20010,M=2e5+10; #pragma GCC optimize(3)
#pragma GCC optimize(2)
#define chk(a,b) ((a>b)&&(a=b)) int n,m,k; ll dis[N]; struct Node{
int x;
ll d;
int operator < (const Node __) const{
return x<__.x;
}
};
vector <Node> G[N]; int s[N<<2]; int pos[N]; inline void push(reg int x) {
int p=pos[x];
while(p>1) {
p>>=1;
if(dis[s[p<<1]]<dis[s[p<<1|1]]) s[p]=s[p<<1];
else s[p]=s[p<<1|1];
}
} inline int top(){
reg int x=s[1];
s[pos[x]]=0;
push(x);
return x;
} void Build(int p,int l,int r) {
if(l==r) {
pos[l]=p;
return;
}
reg int mid=(l+r)>>1;
Build(p<<1,l,mid);
Build(p<<1|1,mid+1,r);
}
//用线段树实现堆的功能 void GetDis(int st,int R) {
memset(dis,63,sizeof dis); dis[st]=0;
for(reg int i=1;i<=n;++i) s[pos[i]]=i;
push(st);
int cnt=0;
while(dis[s[1]]<=1e9) {
reg int u=top();
if(u<=R && (++cnt>=R) ) return;
rep(i,0,G[u].size()-1) {
int v=G[u][i].x,w=G[u][i].d;
if(dis[v]<=dis[u]+w) continue;
dis[v]=dis[u]+w;
push(v);
}
}
} int fa[35];
int dp[1<<20][22];
int Log[1<<21]; int tmp[35]; int main(){
n=rd(),m=rd();
k=rd()+1;
rep(i,0,21) Log[1<<i]=i;
rep(i,1,m) {
int u=rd(),v=rd(),w=rd();
G[u].push_back((Node){v,w});
G[v].push_back((Node){u,w});
}
int q=rd();
rep(i,1,q) {
int u=rd(),v=rd();
fa[v]|=1<<(u-2);
}
Build(1,1,n);
rep(i,1,k) {
GetDis(i,k);
rep(j,1,k) d[i][j]=dis[j];
}
GetDis(n,n);
int A=(1<<(k-1))-1;
memset(dp,63,sizeof dp);
dp[0][1]=0;
for(reg int S=0;S<A;++S) {
reg int T=S;
int cnt=0;
rep(j,0,k-2) if(~S&(1<<j)) if((fa[j+2]&S)==fa[j+2]) tmp[++cnt]=j+2;
while(T) {
reg int i=Log[T&-T]+2;
T&=T-1;
if(dp[S][i]>1e9) continue;
rep(k,1,cnt) {
int j=tmp[k];
reg int NS=S|(1<<(j-2));
chk(dp[NS][j],dp[S][i]+d[i][j]);
}
}
do {
reg int i=1;
if(dp[S][i]>1e9) continue;
rep(k,1,cnt) {
int j=tmp[k];
reg int NS=S|(1<<(j-2));
chk(dp[NS][j],dp[S][i]+d[i][j]);
}
} while(0);
}
ll ans=1e9;
rep(i,1,k) ans=min(ans,dp[A][i]+dis[i]);
printf("%d\n",ans);
}

\[\
\]

\[\
\]

H [BZOJ2004] [Hnoi2010]Bus 公交线路

\(n\)都\(10^9\)了,还不矩阵吗?

\(dp[S]\)表示前\(p\)位哪些点放了车,不过状态显然保证\(popcount(S)==k\)

然后由于状态最多其实是\(C_{10}^{5}=252\)所以可以跑矩阵


bool be; int n,p,k; int dp[1<<10];
int cnt[1<<10];
int tmp[1<<10];
int A; int st[300],sc,id[1<<10];
int f[1][300],ans[1][300]; int B; struct Mat{
int a[300][300];
void init(){ memset(a,0,sizeof a); }
void Get1(){ rep(i,1,sc) a[i][i]=1; }
Mat operator * (const Mat x) const {
Mat res;
for(reg int i=1;i<=sc;++i) {
for(reg int j=1;j<=sc;++j) {
ll t=0;
for(reg int o=1;o<=sc;++o) t+=a[i][o]*x.a[o][j];
res.a[i][j]=t%P;
}
}
return res;
}
}res,x; void Solve(){
A=(1<<p)-1;
rep(i,1,A) cnt[i]=cnt[i&(i-1)]+1;
rep(S,0,A) if(cnt[S]==k) ++sc,id[st[sc]=S]=sc;
rep(S,0,A) if(cnt[S]==k) {
if(S&1) {
x.a[id[S]][id[(S>>1)|(1<<(p-1))]]++;
} else {
rep(i,0,p-1) if(S&(1<<i)) {
int NS=((S^(1<<i))>>1)|(1<<(p-1));
x.a[id[S]][id[NS]]++;
}
}
}
int T=0;
for(reg int j=p-1;j>=p-k;j--) T|=1<<j;
f[0][id[T]]=1;
res.Get1();
n-=k;
int t=n;
while(t) {
if(t&1) res=res*x;
x=x*x;
t>>=1;
}
rep(i,0,0) rep(j,1,sc) rep(o,1,sc) (ans[i][o]+=f[i][j]*res.a[j][o])%=P;
T=0;
rep(j,p-k,p-1) T|=1<<j;
printf("%d\n",ans[0][id[T]]);
} bool ed; int main(){
//printf("%.2lf\n",(&ed-&be)/1024.0/1024.0);
n=rd(),k=rd(),p=rd();
Solve();
}

\[\
\]

\[\
\]

L [BZOJ3195] [Jxoi2012]奇怪的道路

题目限定了距离,所以直接dp选了几条边,之前的点每个点的边数是不是奇数

 

const int N=80,P=1000000007;

int n,m,k;

ll dp[31][1<<8][31];
ll C[N][N]; int main(){
n=rd(),m=rd(),k=rd();
k=min(k,n);
C[0][0]=1;
rep(i,1,N-1) {
C[i][0]=1;
rep(j,1,N-1) C[i][j]=(C[i-1][j-1]+C[i-1][j])%P;
}
dp[0][0][0]=1;
rep(i,1,k-1) {
int A=(1<<i)-1;
rep(S,0,A) {
int t=0;
rep(j,0,i-1) if(S&(1<<j)) t++;
rep(R,0,A) {
int NS=(S^R)|((t&1)<<i);
rep(j,0,m) {
for(reg int d=j+t;d<=m;d+=2) {
(dp[i][NS][d]+=C[(d-j-t)/2+i-1][i-1]*dp[i-1][R][j]%P)%=P;
}
}
}
}
}
int A=(1<<k)-1;
rep(i,k,n-1) {
rep(S,0,A) {
int t=0;
rep(j,0,k-1) if(S&(1<<j)) t++;
rep(R,0,A) if((R&1)==(S&1)) {
int NS=((S^R)>>1)|((t&1)<<(k-1));
rep(j,0,m) {
for(reg int d=j+t;d<=m;d+=2) {
(dp[i][NS][d]+=C[(d-j-t)/2+k-1][k-1]*dp[i-1][R][j]%P)%=P;
}
}
}
}
}
ll ans=dp[n-1][0][m];
printf("%lld\n",ans);
}

状压dp专题复习的更多相关文章

  1. ZOJ 3777 - Problem Arrangement - [状压DP][第11届浙江省赛B题]

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3777 Time Limit: 2 Seconds      Me ...

  2. HDU 1565 - 方格取数(1) - [状压DP][网络流 - 最大点权独立集和最小点权覆盖集]

    题目链接:https://cn.vjudge.net/problem/HDU-1565 Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32 ...

  3. 状压DP复习

    深感自己姿势水平之蒻……一直都不是很会状压DP,NOIP又特别喜欢考,就来复习一发…… 题目来源 Orz sqzmz T1 [BZOJ4197][NOI2015]寿司晚宴 (做过)质因数分解最大的质因 ...

  4. 状压DP复习笔记

    前言 复习笔记第4篇.CSP RP++. 引用部分为总结性内容. 0--P1433 吃奶酪 题目链接 luogu 题意 房间里放着 \(n\) 块奶酪,要把它们都吃掉,问至少要跑多少距离?一开始在 \ ...

  5. kuangbin专题十二 HDU1074 Doing Homework (状压dp)

    Doing Homework Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)To ...

  6. 动态规划专题(一)——状压DP

    前言 最近,决定好好恶补一下我最不擅长的\(DP\). 动态规划的种类还是很多的,我就从 状压\(DP\) 开始讲起吧. 简介 状压\(DP\)应该是一个比较玄学的东西. 由于它的时间复杂度是指数级的 ...

  7. 算法复习——状压dp

    状压dp的核心在于,当我们不能通过表现单一的对象的状态来达到dp的最优子结构和无后效性原则时,我们可能保存多个元素的有关信息··这时候利用2进制的01来表示每个元素相关状态并将其压缩成2进制数就可以达 ...

  8. FZU 1025 状压dp 摆砖块

    云峰菌曾经提到过的黄老师过去讲课时的摆砖块 那时百度了一下题目 想了想并没有想好怎么dp 就扔了 这两天想补动态规划知识 就去FZU做专题 然后又碰到了 就认真的想并且去做了 dp思想都在代码注释里 ...

  9. HDU 3920Clear All of Them I(状压DP)

    HDU 3920   Clear All of Them I 题目是说有2n个敌人,现在可以发n枚炮弹,每枚炮弹可以(可以且仅可以)打两个敌人,每一枚炮弹的花费等于它所行进的距离,现在要消灭所有的敌人 ...

随机推荐

  1. MySQL--用户管理 pymysql 索引

    目录 用户管理 创建mysql账户 权限管理 涉及到的表 pymysql 索引 语法 结论 用户管理 主要是为了控制权限,让不同的人只能操作只属于只记得那部分数据 创建mysql账户 账户中涉及三个数 ...

  2. Springboot jpa多数据源

    1.SpringBootApplication package com.xx.xxx; import org.springframework.beans.factory.annotation.Auto ...

  3. android中实现service动态更新UI界面

    案例:通过service向远程服务器发送请求,根据服务器返回的结果动态更新主程序UI界面,主程序可实时关闭或重启服务. 注册BroadcastReceiver 在主程序activity中注册一个Bro ...

  4. Junit测试。

    Junit使用: 白盒测试 步骤: 1.定义测试类. 2.定义测试方法:可以单独运行. 3.给方法加@Test,导入junit依赖环境. 判定结果: 红色:失败  绿色:成功. 一般不看输出,而是使用 ...

  5. k8s之Deployment 声明式地升级应用(五)

    Deployment 声明式地升级应用 现在你已经知道如何将应用程序组件打包进容器,将他们分组到pod中,并为它们提供临时或者持久存储,将密钥或配置文件注入,并可以使pod之间互相通信.这就是微服务化 ...

  6. manjaro手动安装Redis

    以前都是用的Windows系统,最近有被win10搞得有点烦,就入了manjaro的坑,windows下部分软件在manjaro安装记录,留个记录. 我的系统信息 下面开始正式干活. 一.准备步骤 下 ...

  7. 详解CentOS6.7部署Tomcat及主配置文件

    Java程序实现部署及应用 POSIX :可移植操作系统,编程操作系统接口规范,实现跨平台编译运行. API:应用程序编程接口 ABI:应用程序二进制接口 描述了应用程序和操作系统之间,一个应用和它的 ...

  8. uboot是什么

    u-boot是一种普遍用于嵌入式系统中的Bootloader,Bootloader是在操作系统运行之前执行的一小段程序,通过它,我们可以初始化硬件设备.建立内存空间的映射表,从而建立适当的软硬件环境, ...

  9. 洛谷P2216 理想的正方形(单调队列)

    洛谷P2216 理想的正方形 题目链接 思路: 直接暴力显然不可行,可以发现每一个矩形向右边扩展时是一列一列增加,于是可以想到单调队列,用数组来维护当前每列的最大值.因为行也有限制,所以还要用一个单调 ...

  10. python正则表达式(7)--flag修饰符、match对象属性

    正则表达式—修饰符 正则表达式可以包含一些标志修饰符来控制匹配模式,用在正则表达式处理函数中的flag参数中,为可选参数. (1) re.I 全写(re.IGNORECASE) 表示使匹配时,忽略大小 ...