首页 > 其他分享 >CF Div3 962补题 E-F

CF Div3 962补题 E-F

时间:2024-07-29 13:06:28浏览次数:18  
标签:子串 int mid CF leq 补题 ans Div3

CF Div3 962补题 E-F

E. Decode

链接:

Problem - E - Codeforces

简要题意:

给你一个长度为 \(n\) 的二进制字符串\(s\) 。对于每一对整数\((l, r)\) \((1 \leq l \leq r \leq n)\) 中,数出 \((x, y)\) \((l \leq x \leq y \leq r)\) 这样的整数对的个数。 \((l \leq x \leq y \leq r)\) 中的 \(\mathtt{0}\) 的数量等于子串 \(s_xs_{x+1}...s_y\).中的 \(\mathtt{1}\) 的数量。

输出所有可能的 \((l, r)\) modulo \(10^9+7\) 的计数之和。

思路:

  • 很明显这样的题目算区间贡献和前缀和有关
  • 我们定义子串左边位置为x 子串右边位置设为y
  • 我们发现一个01数量相同的子串 对于整个数组的贡献是 \(x * (n-y+1)\)
  • 算01相同数量我们可以用前缀和 + map实现
  • 每次将x + 1的贡献累加到map中
  • 计算答案 ans = mp[pre[i]] * (n-i+1)

代码:

void solve(){
    string s;
    cin >> s;
    int n = s.size();
    s = " " +s;
    map<int,int> h;
    int pre = 0;
    int ans = 0;
    h[0] = 1;
    for(int i = 1;i<=n;i++){
        if(s[i]=='0') pre--;
        else pre++;
        ans+=h[pre]*(n-i+1)%P;
        ans%=P;
        h[pre]+=i+1;
        h[pre]%P;
    }
    cout << ans <<endl;

}

F. Bomb

链接:

Problem - F - Codeforces

简要题意:

  • 给你两个数组 \(a[n]\) 和 $ b[n]$ \((1<= n <= 2e5)\)
  • 你可以执行最多k次下列操作 \((1<= k <= 1e9)\)
  • 选择 \(a[i]\) 将 \(a[i]\) 加入到你的分数中 然后使\(a[i] = max(0,a[i] - b[i])\)
  • 求最大分数

思路:

  • 首先一个O(k)的思路是 将所有a[i]放入优先队列中,然后不断取k个最大的优先队列top即可,每次取出来都更新优先队列的值 \(a[i] = max(0,a[i] - b[i])\) 再放入,因为可以取多次
  • k有1e9,显然该思路不行
  • 这么大的数据量 很自然的指向了二分,我们要二分什么?
  • 二分一个x,此x为我每次取数字,至少要取大于等于x的数 (核心)
  • 我们得到取大于等于x的数会有一个操作次数,我们要使操作次数f(x) <= k, 并且取max(f(x));
  • 然后我们就会得到一个x,我们把每个数取的的贡献不断加入答案中,这步可用等差数列求和
  • 我们还有一部分是 (k - f(x)) 即剩余操作次数,我们发现 如果x更小,那么操作次数f(x) 会 > k,那么我们只要取(k - f(x))个 x加入答案即可完成最后一部分的补充(这步很难理解,笔者建议多思考一下为什么这样补充)

代码:

const int N = 200005;
int a[N],b[N],n,k;
bool check(int mid){
    int y = 0;
    for(int i = 1;i<= n;i++){
        if(a[i]>=mid){
            y += (a[i] - mid + b[i] - 1)/b[i];
        }
    }
    return y <= k;
}
void solve(){
    cin >> n >> k;
    for(int i = 1;i<=n;i++){
        cin >>a[i];
    }
    for(int i = 1;i<=n;i++){
        cin >>b[i]; 
    }
    int l=0,r=1e12,x=-1;
    while(l <= r){
        int mid = (l + r) >> 1;
        if(check(mid)){
            r = mid - 1;
            x = mid;
        }else{
            l = mid + 1;
        }
    }
    int us = 0;
    int ans = 0;
    for(int i = 1;i<=n;i++){
        if(a[i] >= x){
            int t = (a[i] - x + b[i] - 1)/b[i];
            //(a[i] + a[i] - b[i]*(t - 1))*t/2;
            ans += (a[i] + a[i] - b[i]*(t - 1))*t/2;
            us+=t;
        } 
    } 
    ans += (k - us)*x;
    cout << ans << endl;

}

signed main(){
    std::ios::sync_with_stdio(0);
    std::cin.tie(0);
    std::cout.tie(0);
    int times = 1;
    cin >> times;
    while(times--){
        solve();
    }
    return 0;
}

标签:子串,int,mid,CF,leq,补题,ans,Div3
From: https://www.cnblogs.com/bananawolf/p/18329869

相关文章

  • Pinely Round 4 (Div. 1 + Div. 2) 补题记录(A~F)
    打成乐子A容易证明下标为奇数的地方可以取到,下标为偶数的地方不可以取到。直接模拟时间复杂度为\(O(n)\)。#include<bits/stdc++.h>#defineintlonglongusingnamespacestd;constintN=1000100;inta[N];signedmain(){intT;scanf("%lld",&T);......
  • CF685E 题解
    吐槽一下,为啥这道题\(\mathcal{O}(nm)\)可以过。联考的时候做到了这道题,还一直在想更快的算法。。。然后,这联考的数据是自己出的,有\(s=t\)的情况,我反正是完全没有想到,又因为是捆绑测试,直接痛失\(100pts\)。首先考虑转化一下题目。对于一个询问\(l,r,s,t\),就相当于是让你......
  • CF1906C Cursed Game 题解
    题目大意交互库有一个\(3\times3\)的01矩阵\(a\),每次询问一个\(n\timesn\)的01矩阵\(b\),交互库会返回一个\((n-2)\times(n-2)\)的01矩阵\(c\),满足:\[c_{x,y}=\bigoplus\limits_{1\lei,j\le3}\left(a_{i,j}\operatorname{AND}b_{x+i-1,y+j-1}\right......
  • CF526G Spiders Evil Plan 题解
    Description给定一棵\(n\)个节点的无根树,每条边有边权。有\(q\)次询问,每次询问给出\(x,y\),你需要选择\(y\)条树上的路径,使这些路径形成一个包含\(x\)的连通块,且连通块中包含的边权和最大。\(n,q\le10^5\),强制在线。Solution考虑只有一组询问怎么快速求答案。容......
  • Linux Kernel CFI机制简介及测试禁用
    PS:要转载请注明出处,本人版权所有。PS:这个只是基于《我自己》的理解,如果和你的原则及想法相冲突,请谅解,勿喷。环境说明  无前言  当我们为android移植linux的驱动程序的时候,总会遇到一些错误,这些错误有一部分就是android内核开启的安全的机制导致的。本文就会介绍一种......
  • CF292D 题解
    \(O(mk\alpha(n))\)暴力,考虑对于每个询问\(l,r\),枚举\(1\siml-1,r+1\simm\),并查集连边即可。1154ms。\(O(n(m+k\alpha(n)))\)我们发现枚举\(i\in[1,l),j\in(r,m]\)太慢了。考虑先预处理出并查集从\(1\)连边到编号为\(id\)的边的状态\(pre_{id}\),倒过来再处理出......
  • CF626G Raffles 题解
    Description有\(n\)个奖池,第\(i\)个奖池的奖金是\(p_i\),已经有\(l_i\)张彩票押在上面。现在你有\(t\)张彩票,你需要将你的彩票分配到这些奖池中,并且保证你在每个奖池中押的彩票数不能超过该奖池原有的彩票数。若你在第\(i\)个奖池中押了\(t_i\)张彩票,则你中奖的概......
  • vcf2gwas:简化全基因组关联分析
    vcf2gwas是一个Python构建的API,用于GEMMA、PLINK和bcftools,直接从VCF文件执行GWAS以及多个分析后操作。如何使用?vcf2gwas的使用非常简单。用户只需提供变异调用格式(VCF)文件和表型数据文件,即可通过一条命令行启动GWAS分析。例如:# 安装$ conda install vcf2gwas......
  • CF1060F Shrinking Tree
    考虑分别以每个点为根计算概率,可以计算所有边固定了收缩顺序的概率之和后除以\((n-1)!\)即为答案设\(f_{x,i}\)表示在\(x\)的子树内,删除最后\(i\)条边前后根的编号不发生变化的概率和,所求即为\(f_{rt,n-1}\)记当前点为\(v\),父节点为\(u\),因为收缩\((u,v)\)时,之前的......