首页 > 其他分享 >Codeforces Round 959 (Div. 1 + Div. 2) C. Hungry Games题解

Codeforces Round 959 (Div. 1 + Div. 2) C. Hungry Games题解

时间:2024-09-24 20:51:40浏览次数:10  
标签:959 idx int 题解 mid Div include dp

Codeforces Round 959 (Div. 1 + Div. 2) C. Hungry Games题解

题目链接
zh

大致题意: 给定一个长度为n的数组,并且给出一对l,r表示一个区间,如果 ∑ i = l r a [ i ] \sum_{i=l}^{r}a[i] ∑i=lr​a[i]不为0的话称这样的一个区间是好区间。相加的过程中遵循如下操作:当累加值小于等于x不执行操作,如果大于x,那么累加值就变为0。

思路分析: 本题可以使用DP来求解。我们定义 d p [ i ] dp[i] dp[i]表示以i为区间左端点满足题意区间的数量。根据a的取值发现,累加过程中,在 g < = x g<=x g<=x 时,g是逐渐增大的。那么如果存在的这样一个 r r r 使得 g g g 的值第一次超过而变成0,那么之后的状态可以通过 d p [ r + 1 ] dp[r+1] dp[r+1] 来转移。
记这样的 r r r 为idx。那么可以得到状态转移方程。
d p [ i ] = d p [ i d x ] + i d x − i dp[i]=dp[idx]+idx-i dp[i]=dp[idx]+idx−i
这样的 r r r 可以通过二分或者直接使用 u p p e r b o u n d upper_bound upperb​ound函数来求得。
***代码如下: ***

#include <iostream>
#include <cstring>
#include <map>
#include <cmath>
#include <numeric>
#include <queue>
#include <vector>
#include <unordered_map>
#include <algorithm>
using namespace std;
const int INF=0x3f3f3f3f;
using pii=pair<int,int>;
const int N=2e5+10;
long long a[N];
int dp[N];
int main()
{
    cin.tie(0);
    cout.tie(0);
    ios::sync_with_stdio(false);
    int t;
    cin>>t;
    while(t--)
    {
        memset(dp,0,sizeof dp);
        int n,x;
        cin>>n>>x;
        for(int i=1;i<=n;i++)
        {
            cin>>a[i];
            a[i]+=a[i-1];
        }
        for(int i=n;i>=1;i--)
        {
            int l=i,r=n;
            int res=n+1;
            while(l<=r)
            {
                int mid=(l+r)/2;
                if(a[mid]-a[i-1]>x)
                {
                    r=mid-1;
                    res=mid;
                }
                else l=mid+1;
            }
            dp[i]=dp[res+1]+res-i;
        }
        long long ans=0;
        for(int i=1;i<=n;i++) ans+=dp[i];
        cout<<ans<<endl;
    }     
    return 0;
}

标签:959,idx,int,题解,mid,Div,include,dp
From: https://blog.csdn.net/Nsy1448361880/article/details/142497196

相关文章

  • 力扣题解2207
    大家好,欢迎来到无限大的频道。今日继续给大家带来力扣题解。题目描述(中等)​:字符串中最多数目的子序列给你一个下标从 0 开始的字符串 text 和另一个下标从 0 开始且长度为 2 的字符串 pattern ,两者都只包含小写英文字母。你可以在 text 中任意位置插入 一个......
  • 【洛谷】P2261 [CQOI2007] 余数求和 的题解
    【洛谷】P2261[CQOI2007]余数求和的题解洛谷传送门题解这还是蓝题,这还是省选qaq题目看着很简单,但是真的很考验思路,思路对了,代码不到555分钟写完。刚开始做的时......
  • 题解:SP1741 TETRIS3D - Tetris 3D
    题意维护一个\(D\timesS\)的平面,每个点有一个高度。要求支持一个操作:查询一个矩形区域的最大值,并将该区域更新为最大值加上给定的数。分析发现\(D,S\leq10^3\),考虑使用二维线段树维护。二维线段树,顾名思义,就是在普通线段树的每一个节点上维护一棵线段树。在本题中,外层节......
  • 题解:P10950 太鼓达人
    分析显然答案包含长度为\(K\)的所有\(01\)串,每个串和前一个的重叠长度为\(K-1\),所以每个串对长度的贡献为\(1\)。因此该串的长度为所有\(01\)串的个数,即\(2^K\)。考虑第二个如何解决。发现每个位置的状态只有\(0\)和\(1\),考虑爆搜。显然直接搜的复杂度为\(O(2^......
  • 题解:P5184 [COCI2009-2010#2] PASIJANS
    分析考虑贪心,每次尽量选最小的字符。显然是每次选字典序最小的弹栈。我们要比较的是每个栈的字典序,但是朴素比较是\(O(L)\)的,考虑将它优化到\(O(1)\)。这个时候我们可以先离散化然后套路地将所有串拼一起跑SA。记得在每个串之间加分割符。这样每次比较字典序就变成了\(......
  • 题解:P6351 [PA2011] Hard Choice
    题意维护一张无向图,要求支持以下操作:切断一条边。查询两个点是否有有两条完全不同的路径相连。分析因为断边操作不好维护,考虑离线后将断边变为加边。因此,我们只需要维护加边操作即可。考虑使用LCT。首先,因为涉及到边权,套路地用节点代替边。如果某一条边连接的两个点......
  • ABC245G Foreign Friends 题解 / 二进制分组
    ABC245GForeignFriends题解回顾一下二进制分组。题目大意给定一张\(N\)个点\(M\)条边的无向图,及\(L\)个特殊点。每个点有颜色\(C_i\)。求每个点到离他最近的与他颜色不同特殊点的距离。Solve两个点颜色不同,等价于他们的颜色在二进制下至少有一位不同。所以我们考......
  • MySQL GROUP BY 分区大小写问题解析
    在数据库操作中,GROUPBY是一个常用的SQL语句,用于根据一个或多个列的值对结果集进行分组。然而,在使用MySQL时,你可能会遇到一个常见问题:大小写敏感性。本文将探讨MySQL中GROUPBY的大小写敏感性问题,并提供一些解决方案。什么是大小写敏感性?在计算机科学中,大小写敏感性是指......
  • Codeforces Round 974 (Div.3) 题解
    CodeforcesRound974(Div.3)题解A.RobinHelps模拟按照题意模拟即可。voidShowball(){intn,k;cin>>n>>k;intcur=0,ans=0;for(inti=0;i<n;i++){intx;cin>>x;if(x>=k)cur+=x;elseif(!x){if(cur>=1)cu......
  • AT_jsc2021_g Spanning Tree 题解
    感觉自己稍微有一点唐了。思路我们首先可以把一定要连的边连起来。这样就变成了一个无向图生成树计数问题。如何求解。使用矩阵树定理!我们可以求出基尔霍夫矩阵,然后跑一遍行列式就可以了。时间复杂度:\(O(n^3)\)。Code#include<bits/stdc++.h>usingnamespacestd;con......