题目大概说给一张地图,地图每个格子都有0到9中的某一个数字。现在要在一个格子放炸弹,炸弹爆炸后水柱有两种扩展方式,一种是上、下、左、右,另一种是左上、右下、右上、左下,且四个方向的长度都一样。问放哪个格子怎么爆炸使得水柱覆盖的格子上的数字乘积最大,结果模1e9+7。

这题不会做。。

  • 首先,各个格子的值取对数,这个为了比大小,因为需要模数,通过取对数缩小值。另外也把乘法转化成加法。这是个挺经典的技巧。
  • 接下来,水柱的话肯定不能延长到0,不然功亏一篑,那么利用DP求出各个格子向8各个方向能延长多长,即dp[dir][x][y]。转移就是格子(x,y)不为0,从dp[dir][x+d[dir]][y+d[dir]]+1这儿转移,否则值为0。具体我是用记忆话搜索实现的。
  • 然后知道各个格子8个方向能延长最长长度,上、下、左、右四个方向能够延长的最小值就是第一种水柱的拓展最优的长度,而左上、右下、右上、左下四个方向的最小值就是第二种了。
  • 如何快速得出各个方向水柱覆盖的数字和——利用前缀和!这也是个挺经典的技巧。而取对数把乘法转化成加法,这就使得能利用前缀和的差求区间和。那么预处理出各个方向的前缀和就能在O(1)得出区间和了,即水柱覆盖到的格子的和。
  • 最后就是通过枚举每个格子,得到各个格子放炸弹最多能得到的数字和,更新答案。

注意最后的答案不要直接求次幂还原。。会有精度问题。。应该要记录下来,在原地图中把各个数字累乘出答案。。

 #include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
using namespace std; int n;
double a[][];
int b[][];
//上、下、右、左、左上、右下、右上、左下
int dx[]={-,,,,-,,-,};
int dy[]={,,,-,-,,,-}; int d[][][];
int dp(int dir,int x,int y){
if(a[x][y]<) return d[dir][x][y]=;
if(d[dir][x][y]!=-) return d[dir][x][y];
int nx=x+dx[dir],ny=y+dy[dir],res=;
if(nx>= && nx<=n && ny>= && ny<=n) res+=dp(dir,nx,ny);
return d[dir][x][y]=res;
} double sum[][][];
int rec1[][],rec2[][]; int main(){
scanf("%d",&n);
for(int i=; i<=n; ++i){
for(int j=; j<=n; ++j){
scanf("%1d",&b[i][j]);
if(b[i][j]==) a[i][j]=-;
else a[i][j]=log2(b[i][j]);
}
}
for(int i=; i<=n; ++i){
for(int j=; j<=n; ++j){
sum[][i][j]=sum[][i][j-]+a[j][i];
}
}
for(int i=; i<=n; ++i){
for(int j=; j<=n; ++j){
sum[][i][j]=sum[][i][j-]+a[i][j];
}
}
for(int i=n; i>=; --i){
int x=i,y=;
for(int j=; j<=n-i+; ++j){
sum[][n-i+][j]=sum[][n-i+][j-]+a[x][y];
rec1[x][y]=j;
++x; ++y;
}
}
for(int i=; i<=n; ++i){
int x=,y=i;
for(int j=; j<=n-i+; ++j){
sum[][n+i-][j]=sum[][n+i-][j-]+a[x][y];
rec1[x][y]=j;
++x; ++y;
}
}
for(int i=; i<=n; ++i){
int x=,y=i;
for(int j=; j<=i; ++j){
sum[][i][j]=sum[][i][j-]+a[x][y];
rec2[x][y]=j;
++x; --y;
}
}
for(int i=; i<=n; ++i){
int x=i,y=n;
for(int j=; j<=n-i+; ++j){
sum[][n+i-][j]=sum[][n+i-][j-]+a[x][y];
rec2[x][y]=j;
++x; --y;
}
}
memset(d,-,sizeof(d));
for(int k=; k<; ++k){
for(int i=; i<=n; ++i){
for(int j=; j<=n; ++j){
dp(k,i,j);
}
}
}
double ans=-;
int ansx,ansy,ansdir=-,anslen;
for(int i=; i<=n; ++i){
for(int j=; j<=n; ++j){
if(a[i][j]<) continue; int len=min(min(d[][i][j],d[][i][j]),min(d[][i][j],d[][i][j]));
double res=;
res+=sum[][j][i]-sum[][j][i-len];
res+=sum[][j][i+len-]-sum[][j][i];
res+=sum[][i][j-]-sum[][i][j-len];
res+=sum[][i][j+len-]-sum[][i][j];
if(ans<res){
ans=res;
ansx=i; ansy=j; ansdir=; anslen=len;
} len=min(min(d[][i][j],d[][i][j]),min(d[][i][j],d[][i][j]));
res=;
res+=sum[][n-i+j][rec1[i][j]]-sum[][n-i+j][rec1[i][j]-len];
res+=sum[][n-i+j][rec1[i][j]+len-]-sum[][n-i+j][rec1[i][j]];
res+=sum[][i+j-][rec2[i][j]-]-sum[][i+j-][rec2[i][j]-len];
res+=sum[][i+j-][rec2[i][j]+len-]-sum[][i+j-][rec2[i][j]];
if(ans<res){
ans=res;
ansx=i; ansy=j; ansdir=; anslen=len;
}
}
}
if(ansdir==-){
printf("");
return ;
}
long long res=;
if(ansdir==){
for(int i=ansx-anslen+; i<=ansx+anslen-; ++i){
res*=b[i][ansy];
res%=;
}
for(int i=ansy-anslen+; i<=ansy+anslen-; ++i){
if(i==ansy) continue;
res*=b[ansx][i];
res%=;
}
}else{
int x=ansx-anslen+,y=ansy-anslen+;
for(int i=; i<anslen*; ++i){
res*=b[x][y];
res%=;
++x; ++y;
}
x=ansx-anslen+; y=ansy+anslen-;
for(int i=; i<anslen*; ++i){
if(x==ansx && y==ansy){
++x; --y;
continue;
}
res*=b[x][y];
res%=;
++x; --y;
}
}
printf("%lld",res);
return ;
}

Codeforces 677E Vanya and Balloons(DP + 一些技巧)的更多相关文章

  1. Codeforces 677E Vanya and Balloons

    Vanya and Balloons 枚举中心去更新答案, 数字过大用log去比较, 斜着的旋转一下坐标, 然后我旋出来好多bug.... #include<bits/stdc++.h> ...

  2. Codeforces 677D - Vanya and Treasure - [DP+优先队列BFS]

    题目链接:http://codeforces.com/problemset/problem/677/D 题意: 有 $n \times m$ 的网格,每个网格上有一个棋子,棋子种类为 $t[i][j] ...

  3. Codeforces 677D Vanya and Treasure 暴力+BFS

    链接 Codeforces 677D Vanya and Treasure 题意 n*m中有p个type,经过了任意一个 type=i 的各自才能打开 type=i+1 的钥匙,最初有type=1的钥 ...

  4. codeforces 492E. Vanya and Field(exgcd求逆元)

    题目链接:codeforces 492e vanya and field 留个扩展gcd求逆元的板子. 设i,j为每颗苹果树的位置,因为gcd(n,dx) = 1,gcd(n,dy) = 1,所以当走 ...

  5. [BZOJ 3625] [Codeforces 438E] 小朋友的二叉树 (DP+生成函数+多项式开根+多项式求逆)

    [BZOJ 3625] [Codeforces 438E] 小朋友的二叉树 (DP+生成函数+多项式开根+多项式求逆) 题面 一棵二叉树的所有点的点权都是给定的集合中的一个数. 让你求出1到m中所有权 ...

  6. Codeforces Round #355 (Div. 2) D. Vanya and Treasure dp+分块

    题目链接: http://codeforces.com/contest/677/problem/D 题意: 让你求最短的从start->...->1->...->2->. ...

  7. Codeforces 900 E. Maximum Questions (DP,技巧)

    题目链接:900 E. Maximum Questions 题意: 给出一个长度为n只含有a和b还有'?'的串s,且'?'可以被任意替换为a或b.再给出一个字符串t (奇数位上为a,偶数位上为b,所以 ...

  8. E. Vanya and Balloons Codeforces Round #355 (Div. 2)

    http://codeforces.com/contest/677/problem/E 题意:有n*n矩形,每个格子有一个值(0.1.2.3),你可以在矩形里画一个十字(‘+’形或‘x’形),十字的四 ...

  9. codeforces 721C (拓排 + DP)

    题目链接:http://codeforces.com/contest/721/problem/C 题意:从1走到n,问在时间T内最多经过多少个点,按路径顺序输出. 思路:比赛的时候只想到拓排然后就不知 ...

随机推荐

  1. hdu1115(计算多边形几何重心)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1115 题意:给出一些点,求这些点围成的多边形的重心: 思路: 方法1:直接分别求所有点的x坐标的平均值 ...

  2. NYOJ之字符串逆序输出

    aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAswAAAJaCAIAAAC0jIYTAAAgAElEQVR4nO3du27rSpbGcb+Ecz2IU+ ...

  3. MyEclipse生成WAR包并在Tomcat下部署发布(转发)

    从来没有想过web项目还能打包的,但是有要求,就不得不去实现,在网上找了一下,发现挺简单的. 首先是使用MyEclipse将web项目打包,如下图所示. 右键选中项目,选择export. 然后选择J2 ...

  4. hdu2030 汉字统计

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2030 解题思路:主要考察汉字的编码方式, 汉字机内码在计算机的表达方式的描述是,使用二个字节,汉字的每 ...

  5. 【JAVA集合框架之List】

    一.List接口概述. List有个很大的特点就是可以操作角标. 下面开始介绍List接口中相对于Collection接口比较特别的方法.在Collection接口中已经介绍的方法此处就不再赘述. 1 ...

  6. 攻城狮在路上(壹) Hibernate(十二)--- Hibernate的检索策略

    本文依旧以Customer类和Order类进行说明.一.引言: Hibernate检索Customer对象时立即检索与之关联的Order对象,这种检索策略为立即检索策略.立即检索策略存在两大不足: A ...

  7. phpcms v9 常用调用标签(全)

    本文介绍phpcms v9中模板标签使用说明. {template ) {==}   {/,,)}     loop是data的时候用{thumb($v[thumb],,)} 分页标签------{$ ...

  8. bbed的使用--查看数据文件信息 & sid信息

    1.得到文件的块大小和数据块个数 在Linux和Unix上,oracle提供了一个小工具dbfsize用于查看文件块大小 (可以参看[ID:360032.1]How to detect and fix ...

  9. jbox使用总结

    jbox是一个不错的插件 当使用get打开新页面的时候,可以使用h.对像ID来获得对像ID的值 Js代码 js代码: /** * @description: test * @author: BrinP ...

  10. AngularJS - 指令入门

    指令,我将其理解为AngularJS操作HTML element的一种途径. 由于学习AngularJS的第一步就是写内置指令ng-app以指出该节点是应用的根节点,所以指令早已不陌生. 这篇日志简单 ...