题目传送门(内部题23)


输入格式

第一行有$2$个整数$n,m$。
接下来有$n$行,每行$m$个整数,表示$a$数组。
接下来有$n$行,每行$m$个整数,表示$b$数组。


输出格式

一行一个整数表示答案。


样例

样例输入:

3 3
0 6 8
1 6 1
0 6 8
0 1 2
3 4 5
0 6 7

样例输出:

21


数据范围与提示

样例解释:

最优路径$(2,3)\rightarrow (3,2)\rightarrow (3,3)$。
最优答案$5+6+7+|3-2|+|2-3|+|3-3|+|3-2|=21$。

数据范围:

对于所有数据,$1\leqslant n,m\leqslant 2\times {10}^3,0\leqslant a_i\leqslant {10}^6,0\leqslant b_i\leqslant {10}^6$。
保证至少存在一个地区$a[i][j]>0$,所有$a[i][j]=0$的地区满足$b[i][j]=0$。


题解

首先,为使国王心情更舒畅,我们所要规划的路线一定是在离散化之后$a[i][j]$每次只增加$1$的一条路线,那么你可能会想到建边跑最长路,但是显然这样就$TLE$掉了。

所以我们考虑$DP$,定义$dp[i][j]$表示到达点$(i,j)$的最大吸引度之和。

那么我们可以暴力转移,枚举每一个点,最劣时间复杂度是$\Theta(n^2m^2)$,还是不行。

所以考虑进行优化。

考虑对于点$(i,j)$,要从点$(i',j')$转移得来,那么将分为下面四种情况:

可能从左上,右上,左下,右下四个方向进行转移。

那么你可能会想到用二维树状数组,时间复杂度:$\Theta(n\times m\times \log n \times \log m)$,这样你就拿到了$80$分。

那么考虑满分算法。

分别维护四个最大值:

  $(1,1)−(i,j):dp[i'][j']−i−j$的最大值。
  $(1,j)−(i,m):dp[i'][j']−i+j$的最大值。
  $(i,1)−(n,j):dp[i'][j']+i−j$的最大值。
  $(i,j)−(n,m):dp[i'][j']+i+j$的最大值。

转移的时候从这$4$个最大值中转直接转移过来即可。

那你可能会存在疑问,如果$(i',j')$在$(i,j)$的左上方,但是我们却从维护的左下角中的最大之中转移过来了怎么办?

我会告诉你,这种状况是一定不会发生的,仔细计算一下即可发现从右下转移一定没有从左上转移得到的值更大。

到此这道题就轻松解决了。

对于下面我的代码,我把所有有$a[i][j]$的点压入了队列,然后进行排序,所以$dp$只有一维。

时间复杂度:$\Theta(n\times m)$。

期望得分:$100$分。

实际得分:$100$分。


代码时刻

#include<bits/stdc++.h>
using namespace std;
struct node{long long x,y,v,w;}que[4000001];
int n,m;
long long a[2001][2001],b[2001][2001];
long long num,prem[4],maxn[4];
long long dp[4000001];
long long ans;
bool cmp(node a,node b){return a.v<b.v;}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
scanf("%lld",&a[i][j]);
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
{
scanf("%lld",&b[i][j]);
if(a[i][j])que[++num]=(node){i,j,a[i][j],b[i][j]};
}
sort(que+1,que+num+1,cmp);
int lft;
dp[1]=que[1].w;
maxn[0]=max(maxn[0],dp[1]+que[1].x+que[1].y);
maxn[1]=max(maxn[1],dp[1]-que[1].x+que[1].y);
maxn[2]=max(maxn[2],dp[1]+que[1].x-que[1].y);
maxn[3]=max(maxn[3],dp[1]-que[1].x-que[1].y);
for(int i=2;i<=num;i++)
{
if(que[i].v!=que[i-1].v)
{
lft=i;
break;
}
dp[i]=que[i].w;
maxn[0]=max(maxn[0],dp[i]+que[i].x+que[i].y);
maxn[1]=max(maxn[1],dp[i]-que[i].x+que[i].y);
maxn[2]=max(maxn[2],dp[i]+que[i].x-que[i].y);
maxn[3]=max(maxn[3],dp[i]-que[i].x-que[i].y);
}
for(int i=lft;i<=num;i++)
{
if(que[i].v!=que[i-1].v)
for(int j=0;j<4;j++)
{
prem[j]=maxn[j];
maxn[j]=0;
}
dp[i]=max(max(prem[0]-que[i].x-que[i].y,prem[1]+que[i].x-que[i].y),max(prem[2]-que[i].x+que[i].y,prem[3]+que[i].x+que[i].y))+que[i].w;
maxn[0]=max(maxn[0],dp[i]+que[i].x+que[i].y);
maxn[1]=max(maxn[1],dp[i]-que[i].x+que[i].y);
maxn[2]=max(maxn[2],dp[i]+que[i].x-que[i].y);
maxn[3]=max(maxn[3],dp[i]-que[i].x-que[i].y);
}
for(int i=1;i<=num;i++)
ans=max(ans,dp[i]);
cout<<ans<<endl;
return 0;
}

rp++

[CSP-S模拟测试]:biology(DP)的更多相关文章

  1. noi2019模拟测试赛(四十七)

    noi2019模拟测试赛(四十七) T1与运算(and) 题意: ​ 给你一个序列\(a_i\),定义\(f_i=a_1\&a_2\&\cdots\&a_i\),求这个序列的所 ...

  2. [考试反思]1109csp-s模拟测试106:撞词

    (撞哈希了用了模拟测试28的词,所以这次就叫撞词吧) 蓝色的0... 蓝色的0... 都该联赛了还能CE呢... 考试结束前15分钟左右,期望得分300 然后对拍发现T2伪了写了一个能拿90分的垃圾随 ...

  3. [考试反思]1003csp-s模拟测试58:沉淀

    稳住阵脚. 还可以. 至少想拿到的分都拿到了,最后一题的确因为不会按秩合并和线段树分治而想不出来. 对拍了,暴力都拍了.挺稳的. 但是其实也有波折,险些被卡内存. 如果内存使用不连续或申请的内存全部使 ...

  4. [考试反思]0814NOIP模拟测试21

    前两名是外校的240.220.kx和skyh拿到了190的[暴力打满]的好成绩. 我第5是170分,然而160分就是第19了. 在前一晚上刚刚爆炸完毕后,心态格外平稳. 想想前一天晚上的挣扎: 啊啊啊 ...

  5. csp-s模拟测试98

    csp-s模拟测试98 $T1$??不是我吹我轻松手玩20*20.$T2$装鸭好像挺可做?$T3$性质数据挺多提示很明显? $One$ $Hour$ $Later$ 这$T1$什么傻逼题真$jb$难调 ...

  6. csp-s模拟测试97

    csp-s模拟测试97 猿型毕露.水题一眼秒,火题切不动,还是太菜了. $T1$看了一会儿感觉$woc$期望题$T1??$假的吧??. $T2$秒. $T3$什么玩意儿. 40 01:24:46 00 ...

  7. csp-s模拟测试95

    csp-s模拟测试95 去世场祭. $T1$:这不裸的除法分块吗. $T2$:这不裸的数据结构优化$Dp$吗. $T3$:这不裸的我什么都不会搜索骗$30$分吗. 几分钟后. 这除法分块太劲了..(你 ...

  8. csp-s模拟测试93

    csp-s模拟测试93 自闭场. $T1$想到$CDQ$,因为复杂度少看见一个$0$打了半年还用了$sort$直接废掉,$T2$,$T3$直接自闭暴力分都没有.考场太慌了,心态不好. 02:07:34 ...

  9. csp-s模拟测试92

    csp-s模拟测试92 关于$T1$:最短路这一定建边最短路. 关于$T2$:傻逼$Dp$这一定线段树优化$Dp$. 关于$T3$:最小生成树+树P+换跟一定是这样. 深入(?)思考$T1$:我是傻逼 ...

随机推荐

  1. WPF样式统一之DevExpress设置窗体,控件为Office风格

    DevExpress相信不少人用过,虽然人家不是免费的,但是用过的应该都知道,确实是花了心血的C#插件,下面来介绍下在DevExpress下如何统一设置自己的WPF程序为经典Windows风格. 窗体 ...

  2. servlet--response、request

    请求响应流程图 response 1       response概述 response是Servlet.service方法的一个参数,类型为javax.servlet.http.HttpServle ...

  3. Java相关面试题总结+答案(四)

    [反射] 57. 什么是反射? 反射是在运行状态中,对于任意一个类,都能够知道该类的所有属性和方法,对于任意一个对象,都能够获得该对象的任一属性和方法:这种动态获取信息以及动态调用对象的方法的功能称之 ...

  4. 《剑指offer》面试题9 斐波那契数列 Java版

    书中方法一:递归,这种方法效率不高,因为可能会有很多重复计算. public long calculate(int n){ if(n<=0){ return 0; } if(n == 1){ r ...

  5. linux mysql修改数据表结构语法

    MySQL修改表的语法=========================增加列[add 列名]=========================①alter table 表名 add 列名 列类型 列 ...

  6. Panabit镜像功能配合wireshark抓包的方法

    Panabit镜像功能配合wireshark抓包的方法 Panabit的协议识别都是基于数据包的特征,因此捕获数据包样本是我们进行识别第一步要做的事情.下面就和大家说一下如何捕获网络应用的数据包. 到 ...

  7. 20180329-layoutSubviews的调用机制

    如果你想强制更新布局,不要直接调用此方法,你可以调用setNeedsLayout方法,如果你想立即显示你的views,你需要调用layoutIfNeed方法 layoutSubviews作用: lay ...

  8. Spring Boot 2 Webflux的全局异常处理

    https://www.jianshu.com/p/6f631f3e00b9 本文首先将会回顾Spring 5之前的SpringMVC异常处理机制,然后主要讲解Spring Boot 2 Webflu ...

  9. 基于Xilinx Zynq的计算处理平台

    基于Xilinx Zynq XC7Z045 FFG 900的高性能计算模块 本模块基于Xilinx公司的FPGA XC7Z045 FFG 9000 芯片, 支持64bitDDR3, 容量2GByte: ...

  10. sass-RGB颜色函数-RGB()颜色函数

    在 Sass 的官方文档中,列出了 Sass 的颜色函数清单,从大的方面主要分为 RGB , HSL 和 Opacity 三大函数,当然其还包括一些其他的颜色函数,比如说 adjust-color 和 ...