题目大意

给定一个$n\times m$的网格$(n,m\leq 20)$,每个格子都是$S\space \#\space B\space x\space .$中第一个。

$S$表示起点,保证有且仅有一个。

$\#$表示障碍,不能通过,$.$表示空地,可以通过

$B$表示炸弹,$x$是一个数字,每个数子代表着一个宝藏,每个宝藏有对应的价值(可以为负)。

炸弹和宝藏的数量不超过$8$个

现在你要规划一条从$S$出发,每次只能沿着上下左右四个方向,只能经过除了空地和起点的可以自交的闭合回路,使得这条回路缩圈起来的格子中不含有炸弹,将圈起来的格子中所有宝藏之和加起来记为$sum$,路径长度记为$len$,求$\max(sum-len)$。

举个例子

这遍是原图中的最优解。

题解

我们先想办法解决如何判断一个点是否在回路内。

考虑一个点$(i,j)$(第$i$行第$j$列),从它出发向左下作一条方向无限贴近于正下的射线,通过这条射线与回路相交次数的奇偶性来判断,即枚举所有$k(k>i)$,计算$sum$表示回路经过$(k,j),(k,j-1)$两个点之间的次数和。

若$sum$为奇数,那么在形内,否则在形外。

接下来就简单了,由于炸弹和宝藏的和不超过$8$个,我们直接将炸弹看做价值为$-INF$的宝藏,然后每个宝藏是否在形内进行装状态压缩。

这样就可以从起点出发进行$BFS$了,$F_{(i,j,k)}$表示从$S$出发,到达$(i,j)$,所有宝藏是否在形内的情况为$k$的最短步数。最后只需枚举状态进行计算取$\max$即可。

最终复杂度为$O(nmk2^k)$,我图着省事把其中一个$k$改成了$n$也能过。

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#define LL long long
#define M 30
#define bas 1000
#define INF 10000000
using namespace std;
int read(){
int nm=0,fh=1; char cw=getchar();
for(;!isdigit(cw);cw=getchar()) if(cw=='-') fh=-fh;
for(;isdigit(cw);cw=getchar()) nm=nm*10+(cw-'0');
return nm*fh;
}
const int dx[]={0,0,-1,1},dy[]={-1,1,0,0};
int n,m,q[INF],dis[M][M][550],pos[M][M],T,B,tot,val[M];
int xt[M],yt[M],xb[M],yb[M],hd,tl,Sx,Sy,ans;
char ch[M];
void ins(int x,int y,int sta){q[tl++]=(x*bas+y)*bas+sta;}
void tk(int &x,int &y,int &sta){sta=q[hd]%bas,q[hd]/=bas,y=q[hd]%bas,q[hd]/=bas,x=q[hd],hd++;}
int main(){
n=read(),m=read(),memset(dis,0x3f,sizeof(dis));
for(int i=1;i<=n;i++){
scanf("%s",ch+1);
for(int j=1;j<=m;j++){
if(ch[j]=='.') continue;
if(ch[j]=='B') B++,xb[B]=i,yb[B]=j,pos[i][j]=++tot,val[tot]=-INF;
else if(ch[j]=='#') pos[i][j]=-1;
else if(ch[j]=='S') ins(i,j,0),Sx=i,Sy=j,dis[i][j][0]=0;
else ++T,xt[ch[j]-'0']=i,yt[ch[j]-'0']=j,pos[i][j]=++tot;
}
}
for(int i=1;i<=T;i++) val[pos[xt[i]][yt[i]]]=read();
while(hd<tl){
int x,y,now; tk(x,y,now);
for(int j=0;j<4;j++){
int xx=x+dx[j],yy=y+dy[j],rem=now;
if(pos[xx][yy]||xx<1||yy<1||xx>n||yy>m) continue;
if(y!=yy){
for(int j=1,nw=max(y,yy);j<xx;j++)
if(pos[j][nw]>0) rem^=(1<<(pos[j][nw]-1));
}
if(dis[xx][yy][rem]<INF) continue;
dis[xx][yy][rem]=dis[x][y][now]+1,ins(xx,yy,rem);
}
}
for(int i=1;i<(1<<tot);i++){
int res=0;
for(int dt=1;dt<=tot;dt++) if((i>>(dt-1))&1) res+=val[dt];
ans=max(ans,res-dis[Sx][Sy][i]);
} printf("%d\n",ans); return 0;
}

  

CF221C Circling Round Treasures的更多相关文章

  1. CF 375C Circling Round Treasures [DP(spfa) 状压 射线法]

    C - Circling Round Treasures 题意: 在一个$n*m$的地图上,有一些障碍,还有a个宝箱和b个炸弹.你从(sx,sy)出发,走四连通的格子.你需要走一条闭合的路径,可以自交 ...

  2. Circling Round Treasures CodeForces - 375C

    C. Circling Round Treasures time limit per test 1 second memory limit per test 256 megabytes input s ...

  3. Codeforces 375C Circling Round Treasures - 最短路 - 射线法 - 位运算

    You have a map as a rectangle table. Each cell of the table is either an obstacle, or a treasure wit ...

  4. 【CF375C】Circling Round Treasures

    Portal --> CF375C Solution 一个有趣的事情:题目中有很大的篇幅在介绍如何判断一个位置在不在所围的多边形中 那么..给了方法当然就是要用啊 ​ 首先是不能包含\('B'\ ...

  5. Circling Round Treasures(codeforces 375c)

    题意:要求在一张网格图上走出一条闭合路径,不得将炸弹包围进去,使围出的总价值减去路径长度最大. /* 类似于poj3182的做法,只不过出现了多个点,那么就用状态压缩的方法记录一个集合即可. */ # ...

  6. Codeforces 375C - Circling Round Treasures(状压 dp+最短路转移)

    题面传送门 注意到这题中宝藏 \(+\) 炸弹个数最多只有 \(8\) 个,故考虑状压,设 \(dp[x][y][S]\) 表示当前坐标为 \((x,y)\),有且仅有 \(S\) 当中的物品被包围在 ...

  7. Codeforces 375

    A 7的所有的余数都可以用1,6,8,9排列得到,然后搞一下就可以了. B 可以用类似于单调队列的东西搞.具体看代码: /* * Problem: B. Maximum Submatrix 2 * A ...

  8. Treasures and Vikings(两次搜索)

    Treasures and Vikings https://www.luogu.org/problemnew/show/P4668 题意翻译 你有一张藏宝图,藏宝图可视为 N×MN×M 的网格.每个格 ...

  9. SQL Server 随机数,随机区间,随机抽取数据rand(),floor(),ceiling(),round(),newid()函数等

    在查询分析器中执行:select rand(),可以看到结果会是类似于这样的随机小数:0.36361513486289558,像这样的小数在实际应用中用得不多,一般要取随机数都会取随机整数.那就看下面 ...

随机推荐

  1. ios json结构

    NSString *itemJson = [NSString stringWithFormat:@"{\"Id\":\"%@\",\"Cha ...

  2. 服务器端获取表单数据的编码解码问题(servlet)

    首先需要明确指出的是,这里的服务器是指tomcat. 在页面没有明确指定编码的情况下,客户端通过input标签和字符串向服务器传递两个值param1和param2.如果直接使用request.getP ...

  3. poj2421

    Constructing Roads Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 22048   Accepted: 93 ...

  4. Angular中的$cacheFactory的作用和用法

    1.Angular中的$cacheFactory的作用:    (1)put(key,value); 在缓存对象中插入一个键值对(key,value). (2)get(key); 在缓存对象中通过指定 ...

  5. 洛谷 2216 [HAOI2007]理想的正方形

    题目戳这里 一句话题意 给你一个a×b的矩形,求一个n×n的子矩阵,矩阵里面的最大值和最小值之差最小. Solution 这个题目许多大佬都是单调队列,但是我不是很会,只好用了比较傻逼的方法: 首先我 ...

  6. Django 之Form组件

    Django之From组件 扩展:Django 之 ModelForm组件 Form组件功能 Django的Form主要具有一下几大功能 生成HTML标签 验证用户数据(显示错误信息) HTML Fo ...

  7. python 的for else语句

    for中间不是break出来的,是正常循环完跳出循环的会执行else内的语句 while else语句也是如此 这个以前的常见语言没有,特此记录

  8. JavaWeb:前端开发基础

    JavaWeb:前端开发基础 内联元素和块级元素 说明: 联元素和块级元素都是html中的范畴,块元素和内联元素的主要差异是块元素是从新的一行开始.而内联元素一般显示在一行上.但是可以通过css的di ...

  9. 高性能javascript学习总结(1)--加载与运行

    一.脚本的位置         我们知道,一个<script>标签可以放在 HTML 文档的<head>或<body>标签中,但是浏览器是怎么加载和执行这些java ...

  10. 3.16课·········C#小结

    //附加//C#源码,被C#编译器,编译成执念代码(IL)//int16=short.....±32000//int32=int.......±21亿//int64=long......±922亿亿3 ...