当前位置: 首页 > news >正文

2019牛客暑期多校训练营(第二场) - J - Go on Strike! - 前缀和预处理

题目链接:https://ac.nowcoder.com/acm/contest/882/C
来自:山东大学FST_stay_night的的题解,加入一些注释帮助理解神仙代码。

好像题解被套了一次又一次

要学习的地方我觉得是2点:

1.使用dp(贪心)的思想求出每段所在的连续段
2.因为前缀和是连续变化的,可以用lazy标记来代替树状数组来维护。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;

#define ERR(args...) { string _s = #args; replace(_s.begin(), _s.end(), ',', ' '); stringstream _ss(_s); istream_iterator<string> _it(_ss); err(_it, args); }

void err(istream_iterator<string> it) {
    cerr << "\n";
}
template<typename T, typename... Args>
void err(istream_iterator<string> it, T a, Args... args) {
    cerr << *it << "=" << a << ", ";
    err(++it, args...);
}

#define ERR1(arg,n) { cerr<<""<<#arg<<"=\n  "; for(int i=1;i<=n;i++) cerr<<arg[i]<<" "; cerr<<"\n"; }
#define ERR2(arg,n,m) { cerr<<""<<#arg<<"=\n"; for(int i=1;i<=n;i++) { cerr<<"  "; for(int j=1;j<=m;j++)cerr<<arg[i][j]<<" "; cerr<<"\n"; } }

const int INF = 0x3f3f3f3f;
const int MAXN = 10000000, MAXM = 1000000;

int l[MAXM + 5], r[MAXM + 5], f[MAXM + 5], g[MAXM + 5];
int sum[MAXN * 3 + 5], b[MAXN * 3 + 5], c[MAXN * 3 + 5];

int main() {
#ifdef Yinku
    freopen("Yinku.in", "r", stdin);
#endif // Yinku
    int n;
    scanf("%d", &n);
    for(int i = 1; i <= n; i++)
        scanf("%d%d", &l[i], &r[i]);
    f[1] = r[1] - l[1] + 1;
    //f[i]以i段右端点为结尾的能构造出的最大的前缀和
    for(int i = 2; i <= n; i++)
        f[i] = max(0, f[i - 1] - (l[i] - r[i - 1] - 1)) + r[i] - l[i] + 1;
    //0:以i-1段右端点结尾的能构造出的最大的前缀和都不足够跨过[i-1,i]之间的-1
    //f[i - 1] - (l[i] - r[i - 1] - 1):跨过之后还剩下多少贡献给这段
    g[n] = r[n] - l[n] + 1;
    //g[i]以i段左端点为开头的能构造出的最大的前缀和
    for(int i = n - 1; i >= 1; i--)
        g[i] = max(0, g[i + 1] - (l[i + 1] - r[i] - 1)) + r[i] - l[i] + 1;
    //ERR1(f, n);
    //ERR1(g, n);
    int i = 1, base = 10000000;
    ll ans = 0;
    while(i <= n) {
        int j = i + 1;
        while(j <= n && g[j] + f[j - 1] >= l[j] - r[j - 1] - 1) {
            //说明这个[j-1,j]之间的-1段可以因为两侧的f[j-1]和g[j]足够大而连接起来
            j++;
        }
        j--;
        //此时j是从i开始最远能够连接到的区间
        int left = max(0, l[i] - g[i]), right = min(1000000000 - 1, r[j] + f[j]);
        //left,right是至少会产生一个贡献的范围
        //ERR(left, right);
        int t = i, mi = INF, mx = 0;
        sum[0] = 0;
        for(int k = left; k <= right; k++) {
            //统计这一整段可连接区间的前缀和
            if(k >= l[t] && k <= r[t])
                sum[k - left + 1] = sum[k - left] + 1;
            else
                sum[k - left + 1] = sum[k - left] - 1;
            if(k == r[t])
                t++;
            mi = min(mi, sum[k - left + 1] + base);
            mx = max(mx, sum[k - left + 1] + base);
            //b记录前缀和出现过的次数
            b[sum[k - left + 1] + base] ++;
        }
        //ERR1(sum, right);
        //b记录前缀和出现过的次数的后缀和
        for(int k = mx - 1; k >= mi; k--)
            b[k] += b[k + 1];
        //包含最左侧点的贡献
        ans += b[base + 1];
        for(int k = left; k <= right; k++) {
            t = sum[k - left + 1] + base;
            //t表示k位置sum的值
            //b[t+1]比t大的值的个数
            //c[t+1]比在k位置左侧的比t大的值的个数的lazy
            b[t + 1] -= c[t + 1]; //把lazy加上去
            c[t] += c[t + 1] + 1; //lazy标记下移
            c[t + 1] = 0; //清空lazy
            ans += b[t + 1];
        }
        for(int k = mi; k <= mx; k++)
            b[k] = 0, c[k] = 0;
        i = j + 1;
    }
    printf("%lld", ans);
    return 0;
}

转载于:https://www.cnblogs.com/Yinku/p/11221494.html

相关文章:

  • OS的发展和分类
  • VBScript 内置函数
  • P1020 导弹拦截(nlogn求最长不下降子序列)
  • P1090 合并果子(哈弗曼树)
  • 推荐阅读链接
  • MySQL 5.7 zip 安装
  • P1004 方格取数(四维动态规划)
  • SCRUM Day 8
  • 2.3_Database Interface ODBC组成原理
  • 石子合并(区间dp典型例题)
  • 石子合并2(环形求最优解 区间dp)
  • 恢复系统管理员密码的五大奇招
  • P1082 同余方程(拓展欧几里德)
  • Mac下eclipse安装SVN插件
  • 程序员真的很懒
  • .pyc 想到的一些问题
  • 【MySQL经典案例分析】 Waiting for table metadata lock
  • 11111111
  • GraphQL学习过程应该是这样的
  • Java超时控制的实现
  • Netty源码解析1-Buffer
  • Python socket服务器端、客户端传送信息
  • SpiderData 2019年2月13日 DApp数据排行榜
  • SQLServer之索引简介
  • Sublime text 3 3103 注册码
  • 初探 Vue 生命周期和钩子函数
  • 从setTimeout-setInterval看JS线程
  • 精益 React 学习指南 (Lean React)- 1.5 React 与 DOM
  • 理解IaaS, PaaS, SaaS等云模型 (Cloud Models)
  • 利用阿里云 OSS 搭建私有 Docker 仓库
  • 目录与文件属性:编写ls
  • 前端存储 - localStorage
  • 如何解决微信端直接跳WAP端
  • 手写双向链表LinkedList的几个常用功能
  • 译自由幺半群
  • CMake 入门1/5:基于阿里云 ECS搭建体验环境
  • (2)关于RabbitMq 的 Topic Exchange 主题交换机
  • (ZT)北大教授朱青生给学生的一封信:大学,更是一个科学的保证
  • (二)Linux——Linux常用指令
  • (附源码)springboot社区居家养老互助服务管理平台 毕业设计 062027
  • (附源码)小程序 交通违法举报系统 毕业设计 242045
  • (四)Tiki-taka算法(TTA)求解无人机三维路径规划研究(MATLAB)
  • (原创)攻击方式学习之(4) - 拒绝服务(DOS/DDOS/DRDOS)
  • (转)C#开发微信门户及应用(1)--开始使用微信接口
  • (转)VC++中ondraw在什么时候调用的
  • (转载)PyTorch代码规范最佳实践和样式指南
  • *上位机的定义
  • .NET 5.0正式发布,有什么功能特性(翻译)
  • .net core Swagger 过滤部分Api
  • .NET DataGridView数据绑定说明
  • .net mvc actionresult 返回字符串_.NET架构师知识普及
  • .NET Remoting学习笔记(三)信道
  • .net 生成二级域名
  • .NET/C# 编译期间能确定的相同字符串,在运行期间是相同的实例
  • .NET导入Excel数据