zoukankan      html  css  js  c++  java
  • MMM 状压dp学习记

    状压dp学习记

    by scmmm

    开始日期

    2019/7/17

    前言

    状压dp感觉很好理解(本质接近于爆搜但是又有广搜的感觉),综合了dp的高效性(至少比dfs,bfs优),又能解决普通dp难搞定的问题(例如旅行商问题),又能体验到空间利用的高效性。

    Level 1.模板题

    [洛谷] P1896 [SCOI2005]互不侵犯

    state 指的是每一行的状态

    king 指的是这种情况下国王的个数
    难度★★,很好的一个入门题,dp部分:

    	for(i=1;i<=p;i++)
    	{
    		if(king[i]<=m) f[1][i][king[i]]=1;
    	}
    	for(i=2;i<=n;i++)
    	for(j=1;j<=p;j++)
    	for(k=1;k<=p;k++)
    	{
    		if(state[j]&state[k]) continue;
    		if((state[j]<<1)&state[k]) continue;
    		if((state[j]>>1)&state[k]) continue;
    		for(l=1;l<=m;l++)
    		{
    			if(king[k]+l>m) continue;
    			f[i][k][king[k]+l]+=f[i-1][j][l];
    		}
    	}
    
    }
    

    Level2. P2704 [NOI2001]炮兵阵地

    一道考察空间利用的好题,优化方案两个:

    1.预处理好每行存在的方案(最大60个),然后f[110,65,65]不会爆空间

    2.使用滚动数组,第一行位置1,第二行2,第三行0 结论 第i行为(i+3)%3 (+3是为了防止之后f[i-1]时出现负数)

    对于1.特别注意当前状态是state[i,j]而非j(我在这里被卡了好久)

    难度★★★,对暴力和预处理有个更好的认识,dp部分:

    for(i=1;i<=p[1];i++)
    for(j=1;j<=p[2];j++)
    {
    	if(state[1][i]&state[2][j]) continue;
    	int u=state[2][j],looker=0;
    	while(u) looker+=u%2,u=u>>1;
    	f[2][i][j]=max(f[2][i][j],f[1][0][i]+looker);
    }
    for(i=3;i<=n;i++)
    {
    	for(j=1;j<=p[i];j++)
    	for(l=1;l<=p[i-1];l++)
    	{
    		if(state[i-1][l]&state[i][j]) continue;
    		for(k=1;k<=p[i-2];k++)
    		{
    			if(state[i-1][l]&state[i-2][k]) continue;
    			if(state[i][j]&state[i-2][k]) continue;
    			int u=state[i][j],looker=0;
    			while(u) looker+=u%2,u=u>>1;
    			f[i][l][j]=max(f[i-1][k][l]+looker,f[i][l][j]);
    		}
    	}
    }
    
  • 相关阅读:
    TabControl
    Loading
    Dialog
    Combobox
    Markdown编辑器Editor.md使用方式
    XSS攻击
    跨域解决方案及实现
    angular4 自定义表单组件
    angular4 Form表单相关
    js 详解setTimeout定时器
  • 原文地址:https://www.cnblogs.com/zsx6/p/11206861.html
Copyright © 2011-2022 走看看