[GDOI2014]拯救莫莉斯 状压DP
题面:
莫莉斯·乔是圣域里一个叱咤风云的人物,他凭借着自身超强的经济头脑,牢牢控制了圣域的石油市场。
圣域的地图可以看成是一个n*m的矩阵。每个整数坐标点(x , y)表示一座城市( 1\le x\le n,1\le y\le m1≤x≤n,1≤y≤m )。两座城市间相邻的定义为:对于城市(Ax, Ay)和城市(Bx, By),满足 (Ax - Bx)^2 + (Ay - By)^2 = 1(Ax−Bx)2+(Ay−By)2=1 。
由于圣域的石油贸易总量很大,莫莉斯意识到不能让每笔石油订购单都从同一个油库里发货。为了提高效率,莫莉斯·乔决定在其中一些城市里建造油库,最终使得每一个城市X都满足下列条件之一:
1.该城市X内建有油库,
2.某城市Y内建有油库,且城市X与城市Y相邻。
与地球类似,圣域里不同城市间的地价可能也会有所不同,所以莫莉斯想让完成目标的总花费尽可能少。如果存在多组方案,为了方便管理,莫莉斯会选择建造较少的油库个数。
n * m <= 50, m < n
简单题意:
有一个带权值的矩阵,取一个方格的代价为它的权值,取一个方格时可以给它自己和上下左右的格子打上标记,求最小代价(代价相同取最少数量的方格)使得整个矩阵都被标记。
题解:
观察到n * m <=50,m < n,也就是说m最大也只能是7,看见这么小的数,,,显然这就是个状压啊!
第一眼貌似就是很套路的状压,,,
不过貌似还是有不同的,
唯一的不同在于:一行受上一行和下一行的同时影响,也就是说当前行可以不满足全都标记,因为后来还可以有别的城市来标记它,
因此枚举i和i-1和i-2行的状态,合法条件为:必须使得i-1合法,因此i-1要是再不合法的话以后都不能合法了,,,
同时将f初始化为极大值,这样的话就无需判断i-2是否合法,因为不合法的话将不会被更新,那么值就是inf,也就不会被当做决策了
但是观察到我们并不方便临时计算每个状态的各种数据,因此我们先预处理一遍,处理出每一行的每一个状态对应的城市数和代价,
分别记为num[i][j].num 和 num[i][j].cost
设f[i][j][k].cost 为第i行状态为j,第i-1行状态为k的最小代价,f[i][j][k].num表示在最小代价的基础上的最少城市数,
那么我们的合法条件显然为:if(((k | j | l | (k << 1) | (k >> 1)) & all) == all),其中k为i-1行,j为i行,l为i-2行,
all为2 ^ m - 1 , 也就是2进制下的 111111(m个1)
& all用于消除高于m位的1,以免对ans产生影响。
如果==all就表示合法,因为| j 和| l 表示用上下的城市来标记k一次,然后k << 1 和k >> 1就是用左右的城市来标记一次
那么有转移方程:
if(f[i-1][k][l] + num[i][j] <= f[i][j][k])
f[i][j][k]=f[i-1][k][l] + num[i][j];
其中
node operator + (node a,node b)
{
a.cost += b.cost;
a.num += b.num;
return a;
} bool operator <= (node a,node b)
{
if(a.cost < b.cost) return true;
else if(a.cost == b.cost && a.num < b.num) return true;
else return false;
}
最后统计ans的时候的合法条件为第n行合法。
#include<bits/stdc++.h>
using namespace std;
#define R register int
#define AC 300
#define inf 80000000
#define ac 50
int n,m,all;
int s[ac][ac];
struct node{
int cost,num;
}f[ac][AC][AC],num[ac][AC],ans;
/*因为n * m <= 50 , m < n,所以m实际上是很小的,m <= sqrt(50),
因此预处理出对于每一行,任意状态下的油库个数和代价,在代价相同的基础上取油库最少
一定要注意状态从0开始枚举!*/
node operator + (node a,node b)
{
a.cost += b.cost;
a.num += b.num;
return a;
} bool operator <= (node a,node b)
{
if(a.cost < b.cost) return true;
else if(a.cost == b.cost && a.num < b.num) return true;
else return false;
} void pre()
{
scanf("%d%d",&n,&m);
all=( << m) - ;
for(R i=;i<=n;i++)
for(R j=;j<=m;j++)
scanf("%d",&s[i][j]);
for(R i=;i<=n;i++)//枚举行,看做常数?
for(R j=;j<=all;j++)//枚举状态
for(R k=;k<=m;k++)//看做常数?
if(j & ( << (m - k))) //预处理,,,但是看上去复杂度很高啊
{
num[i][j].cost+=s[i][k];
num[i][j].num++;
}
ans.cost=inf , ans.num=inf;
} void work()
{
for(R i=;i<=all;i++) f[][i][]=num[][i];
for(R i=;i<=all;i++)
for(R j=;j<=all;j++)
{
f[][i][j].cost = inf;
if(((j | i | (j << ) | (j >> )) & all) == all)
f[][i][j]=f[][j][] + num[][i];
}
for(R i=;i<=n;i++)//枚举行
for(R j=;j<=all;j++)//枚举当前行
for(R k=;k<=all;k++)//枚举上一行状态
{
f[i][j][k].cost = inf;
f[i][j][k].num = inf;
for(R l=;l<=all;l++)//枚举上上行状态
if(((k | j | l | (k << ) | (k >> )) & all) == all)//全都更新一遍,至于如何解决二次传播的方法,,,用原版就好了啊
if(f[i-][k][l] + num[i][j] <= f[i][j][k])
f[i][j][k]=f[i-][k][l] + num[i][j];
}
for(R i=;i<=all;i++)
{
for(R j=;j<=all;j++)
if(((i | j | (i << ) | (i >> )) & all) == all)
if(f[n][i][j] <= ans) ans=f[n][i][j];
}
printf("%d %d\n",ans.num,ans.cost);
}//因为受两行影响,而且是上下两行,,,,所以当前行不用满足,保证上一行满足即可???
//因为上一行还不满足的话就满足不了了
int main()
{
// freopen("in.in","r",stdin);
pre();
work();
// fclose(stdin);
return ;
}
[GDOI2014]拯救莫莉斯 状压DP的更多相关文章
- 拯救莫莉斯 状压dp
题目大意:每个点有费用,要求选出花费最少的一些点,使得全部点都满足:他被选或与他相邻的任意点被选. 没看清数据范围233333 和翻格子游戏一样,考虑上中下三行,可行才能转移 f[i][j][k]表示 ...
- [GDOI2014]拯救莫莉斯
题目描述 莫莉斯·乔是圣域里一个叱咤风云的人物,他凭借着自身超强的经济头脑,牢牢控制了圣域的石油市场. 圣域的地图可以看成是一个n*m的矩阵.每个整数坐标点(x , y)表示一座城市吗,两座城市间相邻 ...
- 【[GDOI2014]拯救莫莉斯】
可能我的状态比较鬼畜,应该没有人这么写 设\(dp[i][j][k]\)表示在第\(i\)行,放置油库的状态为\(j\),实际上周围已经有油库或者本身有油库的状态为\(k\)的时候的最小花费 由于我们 ...
- 拯救莫莉斯[GDOI2014]
时间限制:1s 内存限制:256MB 问题描述 莫莉斯·乔是圣域里一个叱咤风云的人物,他凭借着自身超强的经济头脑,牢牢控制了圣域的石油市场. 圣域的地图可以看成是一个n*m的矩阵.每个整数坐标 ...
- [ GDOI 2014 ] 拯救莫莉斯
\(\\\) \(Description\) 有一个 \(N\times M\) 的网格,每个格点都有权值,图是四连通的. 现在选择一个点集,使得每个格点要么被选中,要么连通的点之一被选中. 求这个点 ...
- luogu3888 GDOI2014拯救莫里斯 (状压dp)
题目描述 莫莉斯·乔是圣域里一个叱咤风云的人物,他凭借着自身超强的经济头脑,牢牢控制了圣域的石油市场. 圣域的地图可以看成是一个n*m的矩阵.每个整数坐标点(x , y)表示一座城市\(( 1\le ...
- BZOJ 1087: [SCOI2005]互不侵犯King [状压DP]
1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 3336 Solved: 1936[Submit][ ...
- nefu1109 游戏争霸赛(状压dp)
题目链接:http://acm.nefu.edu.cn/JudgeOnline/problemShow.php?problem_id=1109 //我们校赛的一个题,状压dp,还在的人用1表示,被淘汰 ...
- poj3311 TSP经典状压dp(Traveling Saleman Problem)
题目链接:http://poj.org/problem?id=3311 题意:一个人到一些地方送披萨,要求找到一条路径能够遍历每一个城市后返回出发点,并且路径距离最短.最后输出最短距离即可.注意:每一 ...
随机推荐
- 更改steam的游戏库
用记事本打开steam/steamapps/libraryfolders.vdf,然后按照格式添加条目 "LibraryFolders"{ "TimeNextStatsR ...
- macOS 10.13 High Sierra PHP开发环境配置
命令:sudo rm /usr/local/mysql sudo rm -rf /usr/local/mysql* sudo rm -rf /Library/StartupItems/MySQLCOM ...
- Spring缓存穿透问题修复
本文来自网易云社区. 本剧情纯属真实,犹如雷同实乃缘分. 发生 事情的发生在某天早上,天气怎样反正是忘了,只记得当时监控平台大量的数据库错误报警. 作为后端开发,当看到日志中大量的db连接获取失败,心 ...
- dubbo入门(一)
1.简介 Dubbo由阿里巴巴开源,是一个分布式服务框架,致力于提供高性能和透明化的RPC(远程过程调用)远程服务调用方案,以及SOA服务治理方案.如果没有分布式的需求,Dbubbo是不需要的,其本质 ...
- 使用redux-actions优化actions管理
redux-actions的api很少,有三个createAction(s) handleASction(s) combineActions 主要用到createAction去统一管理actio ...
- 「日常训练」Jin Yong’s Wukong Ranking List(HihoCoder-1870)
题意与分析 2018ICPC北京站A题. 题意是这样的,给定若干人的武力值大小(A B的意思是A比B厉害),问到第几行会出现矛盾. 这题不能出现思维定势,看到矛盾就是矛盾并查集--A>B.A&g ...
- Python :编写条件分支代码的技巧
『Python 工匠』是什么? 我一直觉得编程某种意义是一门『手艺』,因为优雅而高效的代码,就如同完美的手工艺品一样让人赏心悦目. 在雕琢代码的过程中,有大工程:比如应该用什么架构.哪种设计模式.也有 ...
- POSTMan 快速上手(一图带你玩 Postman )
POSTMan 快速上手(一图带你玩 Postman ):
- 【shell 每日一练6】初始化安装Mysql并修改密码
一.简单实现mysql一键安装 参考:[第二章]MySQL数据库基于Centos7.3-部署 此脚本前提条件是防火墙,selinux都已经设置完毕: [root@web130 ~]# cat Inst ...
- 【halcon】算子
算子 rgb1_to_gray 灰度化 threshold:英文是阈的意思 二值化算子 Connection Compute connected components of a region. ...