[bzoj1187][HNOI2007]神奇游乐园_插头dp
bzoj-1187 HNOI-2007 神奇游乐园
题目大意:经历了一段艰辛的旅程后,主人公小P乘坐飞艇返回。在返回的途中,小P发现在漫无边际的沙漠中,有一块狭长的绿地特别显眼。往下仔细一看,才发现这是一个游乐场,专为旅途中疲惫的人设计。娱乐场可以看成是一块大小为n×m的区域,且这个n×m的区域被分成n×m个小格子,每个小格子中就有一个娱乐项目。然而,小P并不喜欢其中的所有娱乐项目,于是,他给每个项目一个满意度。满意度为正时表示小P喜欢这个项目,值越大表示越喜欢。为负时表示他不喜欢,这个负数的绝对值越大表示他越不喜欢。为0时表示他对这个项目没有喜恶。小P决定将飞艇停在某个小格中,然后每步他可以移动到相邻的上下左右四个格子的某个格子中。小P希望找一条路径,从飞艇所在格出发,最后又回到这个格子。小P有一个习惯,从不喜欢浪费时间。因此,他希望经过每个格子都是有意义的:他到一个地方后,就一定要感受以下那里的惊险和刺激,不管自己是不是喜欢那里的娱乐项目。而且,除了飞艇所在格,其他的格子他不愿意经过两次。小P希望自己至少要经过四个格子。在满足这些条件的情况下,小P希望自己玩过的娱乐项目的满意度之和最高。你能帮他找到这个最高的满意度之和吗?
数据范围:$2\le n\le 100$,$2\le m\le 6$。
想法:
不看题的情况下,题目的这个数据范围比较像状压或者全排列什么玩意。
看了题之后就是让你求一个最大的环。
哦....
插头dp呀~
需要左括号右括号和无三种插头。
需要括号的原因是只要一个环。
与那个插头dp例题“求曼哈顿路径”不一样的是,左右括号不一定非得要在最后一个格子合上。
代码:
#include <bits/stdc++.h>
using namespace std;
int m,f[110][7][130],a[110][7],b[7],w[2200],v[130],tot;
inline void update(int &a,int b) {a=(a>b ? a : b);}
void dfs(int p,int c,int now)
{
if(c<0||c>m-p+1) return;
if(p>m)
{
w[now]=++tot,v[tot]=now;
return;
}
dfs(p+1,c,now);
dfs(p+1,c+1,now+b[p]);
dfs(p+1,c-1,now+2*b[p]);
}
inline int Rgt(int v,int p)
{
int i,c=0;
for(i=p;i<=m;i++)
{
if(v/b[i]%3==1) c++;
if(v/b[i]%3==2) c--;
if(!c) return i;
}
return -1;
}
inline int Lft(int v,int p)
{
int i,c=0;
for(i=p;~i;i--)
{
if(v/b[i]%3==2) c++;
if(v/b[i]%3==1) c--;
if(!c) return i;
}
return -1;
}
int main()
{
int n,i,j,k,p,q,ans=-1 << 30;
scanf("%d%d",&n,&m);
for(i=1;i<=n;i++)
for(j=1;j<=m;j++)
scanf("%d",&a[i][j]);
b[0]=1;
for(i=1;i<=m;i++) b[i]=b[i-1]*3;
dfs(0,0,0);
memset(f,0xc0,sizeof(f));
f[1][0][w[0]]=0;
for(i=1;i<=n;i++)
{
for(j=1;j<=m;j++)
{
for(k=1;k<=tot;k++)
{
p=v[k]/b[j-1]%3,q=v[k]/b[j]%3;
if(!p&&!q) update(f[i][j][k],f[i][j-1][k]);
if(!p&&!q&&i<n&&j<m) update(f[i][j][w[v[k]+b[j-1]+2*b[j]]],f[i][j-1][k]+a[i][j]);
if(!p&&q)
{
if(j<m) update(f[i][j][k],f[i][j-1][k]+a[i][j]);
if(i<n) update(f[i][j][w[v[k]+q*(b[j-1]-b[j])]],f[i][j-1][k]+a[i][j]);
}
if(p&&!q)
{
if(i<n) update(f[i][j][k],f[i][j-1][k]+a[i][j]);
if(j<m) update(f[i][j][w[v[k]+p*(b[j]-b[j-1])]],f[i][j-1][k]+a[i][j]);
}
if(p==1&&q==1) update(f[i][j][w[v[k]-b[j-1]-b[j]-b[Rgt(v[k],j)]]],f[i][j-1][k]+a[i][j]);
if(p==2&&q==2) update(f[i][j][w[v[k]-2*b[j-1]-2*b[j]+b[Lft(v[k],j-1)]]],f[i][j-1][k]+a[i][j]);
if(p==2&&q==1) update(f[i][j][w[v[k]-2*b[j-1]-b[j]]],f[i][j-1][k]+a[i][j]);
if(p==1&&q==2&&!(v[k]-b[j-1]-2*b[j])) update(ans,f[i][j-1][k]+a[i][j]);
}
}
for(j=1;j<=tot;j++)
if(v[j]%3==0)
f[i+1][0][j]=f[i][m][w[v[j]/3]];
}
printf("%d\n",ans);
return 0;
}
小结:插头dp是不是多做做题就好了。
[bzoj1187][HNOI2007]神奇游乐园_插头dp的更多相关文章
- BZOJ1187 [HNOI2007]神奇游乐园(插头dp)
麻麻我会写插头dp了! 推荐陈丹琦论文:https://wenku.baidu.com/view/3e90d32b453610661ed9f4bd.html 破题调一年 #include <cs ...
- [HNOI2007]神奇游乐园(插头DP)
题意:n*m的矩阵内值有正有负,找一个四连通的简单环(长度>=4),使得环上值的和最大. 题解:看到2<=m<=6和简单环,很容易想到插头DP,设f[i][j][k]表示轮廓线为第i ...
- 洛谷P3190 [HNOI2007]神奇游乐园(插头dp)
传送门 大概是算第一道自己做出来的插头dp? (虽然都是照着抄板子的) (虽然有个地方死活没调出来最后只能看题解才发现自己错在哪里的) 我就当你们都会插头dp了…… 因为必须得是一条路径,所以扫描线上 ...
- bzoj 1187: [HNOI2007]神奇游乐园【插头dp】
要判边界!!要判边界!!要判边界!!if(j!=m)!!! 我真是zz横着转移要判断到底能不能向右边出边-- 然后剩下的和1814差不多,九十因为不要求经过所有格子,所以左右括号随时可以合并,但是注意 ...
- [bzoj1187][HNOI2007]神奇游乐园
来自FallDream的博客,未经允许,请勿转载,谢谢, 经历了一段艰辛的旅程后,主人公小P乘坐飞艇返回.在返回的途中,小P发现在漫无边际的沙漠中,有一块狭长的绿地特别显眼.往下仔细一看,才发现这是一 ...
- P3190-[HNOI2007]神奇游乐园【插头dp】
正题 题目链接:https://www.luogu.com.cn/problem/P3190 题目大意 \(n*m\)的网格上有权值,求一条权值和最大的不交回路. \(1\leq n\leq 100, ...
- [BZOJ1187]神奇游乐园(插头DP)
Description 题意给定一个矩阵,每个格子有权值,在[-1000.1000]内,求一条回路使得回路经过权值和最大,每个格子最多经过一次 2≤n≤100,2≤m≤6 Code #include ...
- 【BZOJ1187】[HNOI2007]神奇游乐园 插头DP
[BZOJ1187][HNOI2007]神奇游乐园 Description 经历了一段艰辛的旅程后,主人公小P乘坐飞艇返回.在返回的途中,小P发现在漫无边际的沙漠中,有一块狭长的绿地特别显眼.往下仔细 ...
- bzoj 1187: [HNOI2007]神奇游乐园 插头dp
1187: [HNOI2007]神奇游乐园 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 668 Solved: 337[Submit][Statu ...
随机推荐
- ext笔记
命名 The top-level namespaces and the actual class names should be CamelCased. Everything else shoul ...
- Ukulele 天空之城
- Emmet:HTML/CSS代码快速编写神器--20150422
Emmet的前身是大名鼎鼎的Zen coding,如果你从事Web前端开发的话,对该插件一定不会陌生.它使用仿CSS选择器的语法来生成代码,大大提高了HTML/CSS代码编写的速度,比如下面的演示: ...
- 【dp】拔河比赛
01背包:感谢ZCK大佬 题目描述 学校举行拔河比赛,所有的人被分成了两组,每个人必须(且只能够)在其中的一组,要求两个组的人数相差不能超过1,且两个组内的所有人体重加起来尽可能地接近. 输入 输入中 ...
- syslog命令
更多请关注 Linux命令大全 syslog 介绍 syslog是Linux系统默认的日志守护进程.默认的syslog配置文件是/etc/syslog.conf文件.程序,守护进程和内核提供了访问系统 ...
- Springboot(一)-IDEA搭建springboot项目(demo)
jdk版本:1.8.0_162 1.打开IDEA-file-new-project-Spring Initializer,JDK和URL选默认,next (这一步如果是不能联网的话,可以选择直接创建m ...
- 有关linux的GPG签名验证错误的解决方法。
GPG签名验证错误:由于没有公钥,下列签名无法进行验证: NO_PUBKEY 6AF0E1940624A220 找了下原因,虽然不知道原理,不过大概意思还是能才出来的,解决方法如下: gpg --ke ...
- R语言基础-list matrix array
列表可以包含多种类型,如数字/字符/向量/data.frame/list # 创建含一个向量元素的list list1 = list(c(1,2,3)) # list2有三个元素 list2 = li ...
- HDU 3506 DP 四边形不等式优化 Monkey Party
环形石子合并问题. 有一种方法是取模,而如果空间允许的话(或者滚动数组),可以把长度为n个换拓展成长为2n-1的直线. #include <iostream> #include <c ...
- Given a linked list, return the node where the cycle begins. If there is no cycle, returnnull. Follow up: Can
Given a linked list, return the node where the cycle begins. If there is no cycle, returnnull. Follo ...