题面:

莫莉斯·乔是圣域里一个叱咤风云的人物,他凭借着自身超强的经济头脑,牢牢控制了圣域的石油市场。

圣域的地图可以看成是一个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的更多相关文章

  1. 拯救莫莉斯 状压dp

    题目大意:每个点有费用,要求选出花费最少的一些点,使得全部点都满足:他被选或与他相邻的任意点被选. 没看清数据范围233333 和翻格子游戏一样,考虑上中下三行,可行才能转移 f[i][j][k]表示 ...

  2. [GDOI2014]拯救莫莉斯

    题目描述 莫莉斯·乔是圣域里一个叱咤风云的人物,他凭借着自身超强的经济头脑,牢牢控制了圣域的石油市场. 圣域的地图可以看成是一个n*m的矩阵.每个整数坐标点(x , y)表示一座城市吗,两座城市间相邻 ...

  3. 【[GDOI2014]拯救莫莉斯】

    可能我的状态比较鬼畜,应该没有人这么写 设\(dp[i][j][k]\)表示在第\(i\)行,放置油库的状态为\(j\),实际上周围已经有油库或者本身有油库的状态为\(k\)的时候的最小花费 由于我们 ...

  4. 拯救莫莉斯[GDOI2014]

    时间限制:1s     内存限制:256MB 问题描述 莫莉斯·乔是圣域里一个叱咤风云的人物,他凭借着自身超强的经济头脑,牢牢控制了圣域的石油市场. 圣域的地图可以看成是一个n*m的矩阵.每个整数坐标 ...

  5. [ GDOI 2014 ] 拯救莫莉斯

    \(\\\) \(Description\) 有一个 \(N\times M\) 的网格,每个格点都有权值,图是四连通的. 现在选择一个点集,使得每个格点要么被选中,要么连通的点之一被选中. 求这个点 ...

  6. luogu3888 GDOI2014拯救莫里斯 (状压dp)

    题目描述 莫莉斯·乔是圣域里一个叱咤风云的人物,他凭借着自身超强的经济头脑,牢牢控制了圣域的石油市场. 圣域的地图可以看成是一个n*m的矩阵.每个整数坐标点(x , y)表示一座城市\(( 1\le ...

  7. BZOJ 1087: [SCOI2005]互不侵犯King [状压DP]

    1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3336  Solved: 1936[Submit][ ...

  8. nefu1109 游戏争霸赛(状压dp)

    题目链接:http://acm.nefu.edu.cn/JudgeOnline/problemShow.php?problem_id=1109 //我们校赛的一个题,状压dp,还在的人用1表示,被淘汰 ...

  9. poj3311 TSP经典状压dp(Traveling Saleman Problem)

    题目链接:http://poj.org/problem?id=3311 题意:一个人到一些地方送披萨,要求找到一条路径能够遍历每一个城市后返回出发点,并且路径距离最短.最后输出最短距离即可.注意:每一 ...

随机推荐

  1. Vue框架核心之数据劫持

    本文来自网易云社区. 前瞻 当前前端界空前繁荣,各种框架横空出世,包括各类mvvm框架横行霸道,比如Angular.Regular.Vue.React等等,它们最大的优点就是可以实现数据绑定,再也不需 ...

  2. C#新特性记录

    C#6.0新特性笔记 Getter专属赋值 可以在构造函数中,给只有get的属性赋初始值. class Point { public int x { get; } public Point() { x ...

  3. 「日常训练」 Counting Cliques(HDU-5952)

    题意与分析 题源:2016ACM/ICPC沈阳现场赛. 这题让我知道了什么是团,不过最恶心的还是这题的数据了,卡了无数次- - 解决方法是维护一个G数组,不能去遍历邻接矩阵.至少我改了这么一个地方就过 ...

  4. The 2018 ACM-ICPC Asia Qingdao Regional Contest K XOR Clique

    K XOR Clique BaoBao has a sequence a​1​​,a​2​​,...,a​n​​. He would like to find a subset S of {1,2,. ...

  5. Git 与 GitHub

    Git 这个年代,不会点Git真不行啦,少年别问问什么,在公司你就知道了~ Git是一个协同开发的工具,主要作用是进行版本控制,而且还能自动检测代码是否发生变化. 一. 安装 下载地址:https:/ ...

  6. 【SpringCloud】 第九篇: 服务链路追踪(Spring Cloud Sleuth)

    前言: 必需学会SpringBoot基础知识 简介: spring cloud 为开发人员提供了快速构建分布式系统的一些工具,包括配置管理.服务发现.断路器.路由.微代理.事件总线.全局锁.决策竞选. ...

  7. C 关键字 标示符 注释

    一 关键字 1. 什么是关键字 关键字就是C语言提供的有特殊含义的符号 也叫做"保留字" C语言一共提供了32个关键字 这些关键字都被C语言赋予了特殊含义 auto double ...

  8. linux学习总结----shell编程

    ## 环境变量 ## 全局变量 ``` 常见的全局环境变量 PATH 指令的搜索路径 HOME 用户的家目录 LOGNAME 登录名 SHELL 脚本的类型 使用全局环境变量 echo $PATH 自 ...

  9. 悲剧文本(Broken Keyboard (a.k.a. Beiju Text),UVA 11988)

    题目描述: 题目思路: 1.使用链表来重新定位各个字符 2.用数组实现链表 3.开一个数组list[i]来存储字符数组下一个字符的位置 #include <iostream> #inclu ...

  10. python3-声音处理

    先来说下二进制读写文件,这需要struct库 #二进制文件读写 import struct a= b=- # print(struct.pack("h",b)) # print(s ...