[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 ...
随机推荐
- PAT (Basic Level) Practise (中文)-1034. 有理数四则运算(20)
PAT (Basic Level) Practise (中文)-1034. 有理数四则运算(20) http://www.patest.cn/contests/pat-b-practise/1034 ...
- VUE +element el-table运用sortable 拖拽table排序,实现行排序,列排序
Sortable.js是一款轻量级的拖放排序列表的js插件(虽然体积小,但是功能很强大) 项目需求是要求能对element中 的table进行拖拽行排序 这里用到了sorttable Sortable ...
- python 多线程 压测 mysql
#!/usr/bin/env python # encoding: utf-8 #@author: 东哥加油 #@file: sthread.py #@time: 2018/9/17 17:07 im ...
- js中小数精度问题
js中小数的取值为近似值,可能比实际值大,也可能比实际值小,进行“四舍五入”得到的 例如:alert(0.1+0.2);值为0.300000004 alert(0.2+0.7);值为1.899 ...
- aggregate和annotate使用
aggregate和annotate方法的使用场景 Django的aggregate和annotate方法属于高级查询方法,主要用于组合查询,是Django高手们必需要熟练掌握的.当我们需要对查询集( ...
- Python面向对象(类之间的关系)(三)
类与类之间的关系 在我们的世界中事物和事物之间总会有一些联系. 在面向对象中. 类和类之间也可以产生相关的关系 1. 依赖关系 执行某个动作的时候. 需要xxx来帮助你完成这个操作. 此时的关系是最轻 ...
- Python基本运算符和流程控制
常量 常量即不可改变的量,在Python中不存在常量,我们只能逻辑上规定一个常量并不去修改它,通常用全大写字母表示. 基本运算符之二 算术运算 运算符 说明 ** 幂运算 *, /, //, % 乘. ...
- Python3与SQLServer、Oracle、MySql的连接方法
环境: python3.4 64bit pycharm2018社区版 64bit Oracle 11 64bit SQLServer· Mysql 其中三种不同的数据库安装在不同的服务器上,通过局域网 ...
- hdu 5437
Alisha’s Party Time Limit: 3000/2000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others) ...
- PAT Basic 1070
1070 结绳 给定一段一段的绳子,你需要把它们串成一条绳.每次串连的时候,是把两段绳子对折,再如下图所示套接在一起.这样得到的绳子又被当成是另一段绳子,可以再次对折去跟另一段绳子串连.每次串连后,原 ...