AtCoder Grand Contest 067 F - Yakiniku Restaurants

题目传送门:https://arc067.contest.atcoder.jp/tasks/arc067_d

题目大意:

有\(N\)家烧烤店,在直线上按顺序排列,第\(i\)家烧烤店和第\(i+1\)家烧烤店的距离为\(A_i\)。你有\(M\)张烧烤券,在第\(i\)家烧烤店使用第\(j\)张券可以获得\(B_{i,j}\)的快乐,你可以在某家烧烤店使用多张券。你现在可以从某个烧烤店开始,使用所有的券,使得你的快乐值减去所走路程最大

我们考虑每个\(B_{i,j}\)的贡献,我们找到第一个一个\(B_{L,j}>B_{i,j}\)且\(L<i\),然后\(R\)类似,那么\(B_{i,j}\)对答案有贡献需要决策左端点在\((L,i]\)中,右端点在\([i,R)\)中

于是我们可以设\(f_{l,r}\)表示决策在\([l,r]\)的收益,对于每个\(B_{i,j}\),我们对\(f_{(L,i],[i,R)}\)加上\(B_{i,j}\)的贡献,可以证明,对于某张券\(j\),\(B_{1\sim n,j}\)对答案的贡献矩阵没有交集,因此我们可以用二维差分解决,最后还原\(f\)即可

/*problem from Wolfycz*/
#include<cmath>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define inf 0x7f7f7f7f
using namespace std;
typedef long long ll;
typedef unsigned int ui;
typedef unsigned long long ull;
inline char gc(){
    static char buf[1000000],*p1=buf,*p2=buf;
    return p1==p2&&(p2=(p1=buf)+fread(buf,1,1000000,stdin),p1==p2)?EOF:*p1++;
}
inline int frd(){
    int x=0,f=1; char ch=gc();
    for (;ch<'0'||ch>'9';ch=gc())   if (ch=='-')    f=-1;
    for (;ch>='0'&&ch<='9';ch=gc()) x=(x<<3)+(x<<1)+ch-'0';
    return x*f;
}
inline int read(){
    int x=0,f=1; char ch=getchar();
    for (;ch<'0'||ch>'9';ch=getchar())  if (ch=='-')    f=-1;
    for (;ch>='0'&&ch<='9';ch=getchar())    x=(x<<3)+(x<<1)+ch-'0';
    return x*f;
}
inline void print(int x){
    if (x<0)    putchar('-');
    if (x>9)    print(x/10);
    putchar(x%10+'0');
}
const int N=5e3,M=2e2;
int L[M+10][N+10],R[M+10][N+10],B[M+10][N+10],stack[N+10];
ll sum[N+10][N+10],A[N+10];
int main(){
    int n=read(),m=read(); ll Ans=0;
    for (int i=2;i<=n;i++)  A[i]=read()+A[i-1];
    for (int i=1;i<=n;i++)
        for (int j=1;j<=m;j++)
            B[j][i]=read();
    for (int i=1;i<=m;i++){
        for (int j=1,top=0;j<=n;j++){
            while (top&&B[i][stack[top]]<B[i][j])   top--;
            L[i][j]=top?stack[top]+1:1;
            stack[++top]=j;
        }
        for (int j=n,top=0;j>=1;j--){
            while (top&&B[i][stack[top]]<B[i][j])   top--;
            R[i][j]=top?stack[top]-1:n;
            stack[++top]=j;
        }
        for (int j=1;j<=n;j++){
            sum[L[i][j]][j]+=B[i][j];
            sum[L[i][j]][R[i][j]+1]-=B[i][j];
            sum[j+1][j]-=B[i][j];
            sum[j+1][R[i][j]+1]+=B[i][j];
        }
    }
    for (int i=1;i<=n;i++){
        for (int j=1;j<=n;j++)  sum[i][j]+=sum[i][j-1];
        for (int j=1;j<=n;j++)  sum[i][j]+=sum[i-1][j];
        for (int j=i;j<=n;j++)  Ans=max(Ans,sum[i][j]-A[j]+A[i]);
    }
    printf("%lld\n",Ans);
    return 0;
}

原文地址:https://www.cnblogs.com/Wolfycz/p/10071802.html

时间: 2024-08-30 01:36:09

AtCoder Grand Contest 067 F - Yakiniku Restaurants的相关文章

AtCoder Grand Contest 006 F - Blackout

Description 在 \(n*n\) 的棋盘上给出 \(m\) 个黑点,若 \((x,y)\),\((y,z)\) 都是黑点,那么 \((z,x)\) 也会变成黑点,求最后黑点的数量 题面 Solution 把点 \((x,y)\) 看作一条从 \(x\) 到 \(y\) 的有向边 我们分析性质: 如果存在一个自环,那么这个点所在的连通块就会变成一个完全图 原因是和这个点有单向边的点都会变成双向边,有双向边之后就会形成自环,那么就可以一直重复这个过程,就变成了完全图 我们想办法判断图中有没

【AtCoder】ARC067 F - Yakiniku Restaurants 单调栈+矩阵差分

[题目]F - Yakiniku Restaurants [题意]给定n和m,有n个饭店和m张票,给出Ai表示从饭店i到i+1的距离,给出矩阵B(i,j)表示在第i家饭店使用票j的收益,求任选起点和终点的最大(收益-代价).n<=5000,m<=200. [算法]单调栈+矩阵差分 [题解]直接枚举区间,很难同时计算m张票,我们反过来考虑每个B(i,j)的贡献. 对于B(i,j),令x为满足x<i,B(x,j)>B(i,j)的最大的x,令y为满足y>i,B(y,j)>B(

AtCoder Grand Contest 025 Problem D

www.cnblogs.com/shaokele/ AtCoder Grand Contest 025 Problem D Time Limit: 2 Sec Memory Limit: 1024 MB Description Takahashi is doing a research on sets of points in a plane. Takahashi thinks a set \(S\) of points in a coordinate plane is a good set w

AtCoder Grand Contest 024 Problem E(动态规划)

www.cnblogs.com/shaokele/ AtCoder Grand Contest 024 Problem E Time Limit: 2 Sec Memory Limit: 1024 MB Description Find the number of the possible tuples of sequences (\(A_0,A_1,-,A_N\)) that satisfy all of the following conditions, modulo \(M\): ? Fo

AtCoder Grand Contest 011

AtCoder Grand Contest 011 upd:这篇咕了好久,前面几题是三周以前写的... AtCoder Grand Contest 011 A - Airport Bus 翻译 有\(n\)个乘客到达了飞机场,现在他们都要坐车离开机场.第\(i\)个乘客到达的时间是\(T_i\),一个乘客必须在\([T_i,T_i+k]\)时刻做到车,否则他会生气.一辆车最多可以坐\(C\)个人.问最少安排几辆车可以让所有人都不生气. 题解 从前往后贪心即可. #include<iostream

AtCoder Grand Contest 014

AtCoder Grand Contest 014 A - Cookie Exchanges 有三个人,分别有\(A,B,C\)块饼干,每次每个人都会把自己的饼干分成相等的两份然后给其他两个人.当其中有一个人的饼干数量是奇数的时候停止,求会进行几次这样子的操作,或者会永远进行下去. 首先无解的情况一定是三个数都是相等的偶数. 否则直接暴力模拟就行了.(盲猜答案不会很大) 证明一下答案的范围:不妨令\(A\le B\le C\),那么最大值和最小值之间的差就是\(C-A\),那么执行完一次操作之后

AtCoder Grand Contest 016

AtCoder Grand Contest 016 A - Shrinking 你可以进行一个串的变换,把一个长度为\(n\)的串\(S\)可以变成长度为\(n-1\)的串\(T\),其中\(T_i\)要么是\(S_i\)要么是\(S_{i+1}\). 现在问你最少进行多少次这个操作,能够使最终得到的\(T\)只由一个字符构成. \(|S|\le 100\) 首先枚举最终字符是哪一个.那么首先在\(S\)末尾加上一个这个字符,那么这个最小步数等于对于所有位置而言,离它最近的枚举的字符到这个位置的

Atcoder Grand Contest 018 E - Sightseeing Plan

Atcoder Grand Contest 018 E - Sightseeing Plan 枚举从第二个矩形的 \((x_1,y_1)\) 进入,\((x_2,y_2)\) 出来,那么中间可以选的点的数量是 \(x_2+y_2-x_1-x_2+1\) ,也就是说对于每一条合法路线,从 \((x_1,y_1)\) 进入的贡献为 \(-x_1-x_2\) ,从 \((x_2,y_2)\) 出来的贡献为 \(x_2+y_2+1\) ,枚举一下第二个矩形边界上的点,我们只需要分别计算某个点到第一个矩形

【Atcoder Grand Contest 020 E】 Encoding Subsets

Atcoder Grand Contest 020 E 题意:给一个\(0-1\)字符串,如果其中有一段重复,就可以表示成\((\)这一块的表示\(\times\)出现次数\()\). 问这个字符串的所有子集中有多少种表示方法. 思路:考虑\(dp(s)\)表示字符串\(s\)的答案. 那么我们得考虑第一个表示成的位置是什么. ①第一位就是表示的第一位,不参与循环.那么转移到\(dp(s.substr(1))\),并且如果这位是\(1\),那么乘上\(2\),因为这位可能是\(0\). ②一个前