f(i,j,S)表示到(i,j),且经由的路径上的颜色集合为S的价值的最小值,从上方和左方转移过来即可。

要注意,内存不足,需要滚动数组优化,即使用了map,还是需要。

路径输出的时候,可以再跑一遍dp,这样就不用再开一个大数组了。

我的写法比较菜。卡了常数

#include<cstdio>
#include<algorithm>
#include<map>
using namespace std;
int w[401][401],c[401][401];
typedef pair<int,int> Point;
map<int,int>f[401][401];
typedef pair<int,Point> data;
typedef map<int,int>::iterator ITER;
map<int,data>g[401][401];
int n,K,W;
int calc(int x)
{
int res=0;
while(x)
{
res+=(x&1);
x>>=1;
}
return res;
}
Point path[1001];
int e;
int main()
{
scanf("%d%d%d",&n,&K,&W);
for(int i=1;i<=n;++i)
for(int j=1;j<=n;++j)
scanf("%d",&w[i][j]);
for(int i=1;i<=n;++i)
for(int j=1;j<=n;++j)
scanf("%d",&c[i][j]);
if(w[1][1]>W)
{
puts("-1");
return 0;
}
f[1][1][1<<(c[1][1]-1)]=w[1][1];
for(int i=1;i<=n;++i){
for(int j=1;j<=n;++j)
{
if(i>1 && j>1)
{
for(ITER it=f[i-1][j].begin();it!=f[i-1][j].end();++it)
{
int nS=(*it).first|(1<<(c[i][j]-1)),tmp=(*it).second+w[i][j];
if(f[i][j].find(nS)==f[i][j].end())
{
if(tmp<=W)
{
f[i][j][nS]=tmp;
g[i][j][nS]=data((*it).first,Point(i-1,j));
}
}
else if(f[i][j][nS]>tmp)
{
f[i][j][nS]=tmp;
g[i][j][nS]=data((*it).first,Point(i-1,j));
}
}
for(ITER it=f[i][j-1].begin();it!=f[i][j-1].end();++it)
{
int nS=(*it).first|(1<<(c[i][j]-1)),tmp=(*it).second+w[i][j];
if(f[i][j].find(nS)==f[i][j].end())
{
if(tmp<=W)
{
f[i][j][nS]=tmp;
g[i][j][nS]=data((*it).first,Point(i,j-1));
}
}
else if(f[i][j][nS]>tmp)
{
f[i][j][nS]=tmp;
g[i][j][nS]=data((*it).first,Point(i,j-1));
}
}
}
else if(i==1 && j>1)
{
for(ITER it=f[i][j-1].begin();it!=f[i][j-1].end();++it)
{
int nS=(*it).first|(1<<(c[i][j]-1)),tmp=(*it).second+w[i][j];
if(f[i][j].find(nS)==f[i][j].end())
{
if(tmp<=W)
{
f[i][j][nS]=tmp;
g[i][j][nS]=data((*it).first,Point(i,j-1));
}
}
else if(f[i][j][nS]>tmp)
{
f[i][j][nS]=tmp;
g[i][j][nS]=data((*it).first,Point(i,j-1));
}
}
}
else if(i>1 && j==1)
{
for(ITER it=f[i-1][j].begin();it!=f[i-1][j].end();++it)
{
int nS=(*it).first|(1<<(c[i][j]-1)),tmp=(*it).second+w[i][j];
if(f[i][j].find(nS)==f[i][j].end())
{
if(tmp<=W)
{
f[i][j][nS]=tmp;
g[i][j][nS]=data((*it).first,Point(i-1,j));
}
}
else if(f[i][j][nS]>tmp)
{
f[i][j][nS]=tmp;
g[i][j][nS]=data((*it).first,Point(i-1,j));
}
}
}
}
if(i>=3)
{
for(int j=1;j<=n;++j)
{
f[i-2][j].clear();
// g[i-2][j].
}
}
}
int ans=2147483647;
int anS;
for(ITER it=f[n][n].begin();it!=f[n][n].end();++it)
{
int tmp=calc((*it).first);
if(tmp<ans)
{
ans=tmp;
anS=(*it).first;
}
}
if(ans==2147483647)
puts("-1");
else
{
printf("%d\n",ans);
data U=data(anS,Point(n,n));
path[++e]=Point(n,n);
while(U.second!=Point(1,1))
{
U=g[U.second.first][U.second.second][U.first];
path[++e]=U.second;
}
for(int i=e;i>=1;--i)
printf("%d %d%c",path[i].first,path[i].second,i==1 ? '\n' : ' ');
}
return 0;
}

【滚动数组】【状压dp】Gym - 100956F - Colored Path的更多相关文章

  1. 状压dp Gym - 100676G

    http://codeforces.com/gym/100676 题目大意: 给你n个科目,m个关系,例如A->B,表示要学习B科目,一定要把A科目学习掉.同理,如果还有C->B,那么,B ...

  2. zoj3545Rescue the Rabbit (AC自动机+状压dp+滚动数组)

    Time Limit: 10 Seconds      Memory Limit: 65536 KB Dr. X is a biologist, who likes rabbits very much ...

  3. Codeforces Gym 100015F Fighting for Triangles 状压DP

    Fighting for Triangles 题目连接: http://codeforces.com/gym/100015/attachments Description Andy and Ralph ...

  4. Codeforces Gym 100610 Problem K. Kitchen Robot 状压DP

    Problem K. Kitchen Robot Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/gym/10061 ...

  5. POJ 3311 Hie with the Pie (状压DP)

    dp[i][j][k] i代表此层用的状态序号 j上一层用的状态序号 k是层数&1(滚动数组) 标准流程 先预处理出所有合法数据存在status里 然后独立处理第一层 然后根据前一层的max推 ...

  6. ZOJ3802 Easy 2048 Again (状压DP)

    ZOJ Monthly, August 2014 E题 ZOJ月赛 2014年8月 E题 http://acm.zju.edu.cn/onlinejudge/showProblem.do?proble ...

  7. 状压DP uvalive 6560

    // 状压DP uvalive 6560 // 题意:相邻格子之间可以合并,合并后的格子的值是之前两个格子的乘积,没有合并的为0,求最大价值 // 思路: // dp[i][j]:第i行j状态下的值 ...

  8. ZOJ 3723 (浙大月赛)状压DP

    A了一整天~~~终于搞掉了. 真是血都A出来了. 题目意思很清楚,肯定是状压DP. 我们可以联系一下POJ 1185  炮兵阵地,经典的状压DP. 两道题的区别就在于,这道题的攻击是可以被X挡住的,而 ...

  9. hdu_4529_郑厂长系列故事——N骑士问题(状压DP)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=4529 题意:中文,不解释 题解:状压DP,dp[i][j][k][s]表示第i行当前用了j个骑士,i- ...

随机推荐

  1. meta 常用标签总结

    声明:并非原创 meta元素工有3个可选属性(http-equiv.name.scheme)和一个必选属性(content),content定义与http-equiv或name属性相关的元信息 可选属 ...

  2. Telepro工具注册码

    Teleport Pro v1.54 注册码 Teleport Pro v1.54姓名(Name):3ddown.com序列号(Serial):161594593

  3. CentOS添加字体

    到Windows XP或者Vista下复制字体到CentOS 1.到Windows XP或者Vista下复制字体到CentOS 雅黑:msyh 黑体:SimHei 宋体:SimSun 华文细黑:STX ...

  4. angularjs ng-switch

    <p> <a href="#" ng-click="toggle()">Toggle Section</a> </p& ...

  5. zf-关于业务量统计柱形图(上月份的没显示出来的解决办法)

    首先要想到是存储过程里面除了问题,导致没有显示出来 因为本年度和本季度 是能显示出来的 所以后台代码是没问题的 存储过程里 有个tj_type  这个tj_type有3个值 1 代表本年度 2 代表本 ...

  6. android 以不规则图片为边框切割另外图片

    转自:http://blog.sina.com.cn/s/blog_474928c90101dkvf.html 最近工作上遇到了一个将一个图片按照相框进行裁剪的问题,花了一个下午研究了下,在此整理一下 ...

  7. HDU 1815 Building roads

    二分答案 + 2-SAT验证 POJ 稳过,HDU C++ 超时,G++ 550ms左右AC #include<cstdio> #include<cstring> #inclu ...

  8. OpenCV学习笔记(27)KAZE 算法原理与源码分析(一)非线性扩散滤波

    http://blog.csdn.net/chenyusiyuan/article/details/8710462 OpenCV学习笔记(27)KAZE 算法原理与源码分析(一)非线性扩散滤波 201 ...

  9. java采用Apache FileUpload组件实现上传

    可见:FileItemSteam(servletFileUpload.getItemIterator(httpServletRequest))速度要快于FileItem(servletFileUplo ...

  10. 2016"百度之星" - 资格赛(Astar Round1) Problem A

    保存前缀乘,询问的时候输出c[ri]/c[li-1]即可,因为是除法,所以计算一下c[li-1]的逆元. #include <stdio.h> #include <math.h> ...