题意是,有$n$个石头,每个石头有初始能量$E_i$,每秒能量增长$L_i$,以及能量上限$C_i$,有$m$个收能量的时间点,每次把区间$\left[S_i, T_i\right]$石头的能量都给收掉,石头的能量都置零重新开始增长。问最后收了多少能量。

看完题解觉得好有道理...我好菜...
考虑每个石头在多少个时间点收能量,然后每次收的能量就和这些时间点的时间间隔有关。
若时间间隔大于等于$\dfrac {C_i}{L_i}$,那么这一段对答案的贡献就是$C_i$了,统计有多少这样的段即可。
若时间间隔小于$\dfrac {C_i}{L_i}$那么对答案的贡献就是时间长度$t \times L_i$。
用两个权值树状数组可以维护对应时间长度的和及个数。
时间点可以用set维护。从前到后遍历,遇到一个$S_i$就把对应是时间加入,遇到一个$T_i + 1$就把时间删去,同时维护树状数组即可。感觉看代码就很好懂?

#include <bits/stdc++.h>
#define ll long long
using namespace std; const int N = 2e5 + ;
int n, m;
ll E[N], C[N], L[N];
set<int> st;
vector<int> G[N]; struct BIT {
ll tree1[N], tree2[N];
inline void clear() {
memset(tree1, , sizeof tree1);
memset(tree2, , sizeof tree2);
}
inline int lowbit(int x) {
return x & -x;
}
inline void add(int x, int val) {
if (!x) return;
for (int i = x; i < N; i += lowbit(i)) {
if (val > ) tree1[i]++;
else tree1[i]--;
tree2[i] += val;
}
}
inline int query1(int x) {
int ans = ;
for (int i = x; i; i -= lowbit(i))
ans += tree1[i];
return ans;
}
inline int query2(int x) {
int ans = ;
for (int i = x; i; i -= lowbit(i))
ans += tree2[i];
return ans;
}
} bit; inline void init() {
st.clear();
bit.clear();
for (int i = ; i <= n; i++) G[i].clear();
} void add(int x) {
if (st.empty()) {
st.insert(x);
return;
}
auto p = st.lower_bound(x);
if (p == st.begin()) {
bit.add((*p - x), (*p - x));
st.insert(x);
return;
}
if (p == st.end()) {
bit.add(x - (*prev(p)), x - (*prev(p)));
st.insert(x);
return;
}
int x1 = (*p) - x, x2 = x - (*prev(p));
bit.add(x1, x1);
bit.add(x2, x2);
bit.add(x1 + x2, -x1 - x2);
st.insert(x);
} void del(int x) {
auto p = st.find(x);
if (st.size() == ) {
st.erase(p);
return;
}
if (p == st.begin()) {
bit.add((*next(p)) - x, x -(*next(p)));
st.erase(p);
return;
}
if (p == prev(st.end())) {
bit.add(x - (*prev(p)), (*prev(p)) - x);
st.erase(p);
return;
}
int x1 = (*next(p)) - x, x2 = x - (*prev(p));
bit.add(x1, -x1);
bit.add(x2, -x2);
bit.add(x1 + x2, x1 + x2);
st.erase(p);
} int main() {
int T, kase = ;
scanf("%d", &T);
while (T--) {
scanf("%d", &n);
init();
for (int i = ; i <= n; i++)
scanf("%lld%lld%lld", &E[i], &L[i], &C[i]);
scanf("%d", &m);
for (int i = ; i <= m; i++) {
int l, r, t;
scanf("%d%d%d", &t, &l, &r);
G[l].push_back(t); G[r + ].push_back(-t);
}
ll ans = ;
for (int i = ; i <= n; i++) {
for (auto x: G[i]) {
if (x > ) add(x);
else del(-x);
}
if (st.empty()) continue;
ans += min(C[i], 1LL * (*st.begin()) * L[i] + E[i]);
if (!L[i]) continue;
ans += (st.size() - - bit.query1(C[i] / L[i])) * C[i] + bit.query2(C[i] / L[i]) * L[i];
}
printf("Case #%d: %lld\n", ++kase, ans);
}
return ;
}

F Energy stones的更多相关文章

  1. 2019牛客多校第七场 F Energy stones 树状数组+算贡献转化模拟

    Energy stones 题意 有n块石头,每块有初始能量E[i],每秒石头会增长能量L[i],石头的能量上限是C[i],现有m次时刻,每次会把[s[i],t[i]]的石头的能量吸干,问最后得到了多 ...

  2. Kick Start 2019 Round B Energy Stones

    对我很有启发的一道题. 这道题的解法中最有思维难度的 observation 是 For simplicity, we will assume that we never eat a stone wi ...

  3. 2019牛客暑期多校训练营(第七场)E F H I

    E Find the median 题意:每次往序列中增加连续的[l,r]的数,每加入一次就询问当前序列的中位数. 解法:此题没有要求在线,那么直接离线+线段树+二分就可以了.求出每个端点之后排序得到 ...

  4. 2019nc#7

    题号 标题 已通过代码 题解/讨论 通过率 团队的状态 A String 点击查看 进入讨论 566/3539  通过 B Irreducible Polynomial 点击查看 规律 730/229 ...

  5. 2019牛客多校 Round7

    Solved:5 Rank:296 E Find the median (线段树) 题意:最开始一个空的数组 4e5次操作 每次把Li,Ri中的每个数插入进来 问当前的中位数 题解:把这n个区间离散化 ...

  6. ZJUT11 多校赛补题记录

    牛客第一场 (通过)Integration (https://ac.nowcoder.com/acm/contest/881/B) (未补)Euclidean Distance (https://ac ...

  7. hdu 3681 Prison Break(状态压缩+bfs)

    Problem Description Rompire . Now it’s time to escape, but Micheal# needs an optimal plan and he con ...

  8. 过河(DP)

    问题描述] 在河上有一座独木桥,一只青蛙想沿着独木桥从河的一侧跳到另一侧.在桥上有一些石子,青蛙很讨厌踩在这些石子上.由于桥的长度和青蛙一次跳过的距离都是正整数,我们可以把独木桥上青蛙可能到达的点看成 ...

  9. 【模拟】bzoj1686: [Usaco2005 Open]Waves 波纹

    打完模拟题来庆祝一波:):感觉最近陷入一种“口胡五分钟打题两小时”的巨坑之中…… Description Input     第1行:四个用空格隔开的整数Pj Bi,B2,R. P(1≤P≤5)表示石 ...

随机推荐

  1. spring cloud 客户端负载均衡 - Ribbon

    Spring Cloud Ribbon是一个基于HTTP和TCP的客户端负载均衡工具,基于Netflix Ribbon实现的,Ribbon不像注册中心.网关那样需要单独部署,它是作为一个工具直接集成到 ...

  2. char[],char *,string之间转换

    char []与char *之间转换 char []转char *:直接进行赋值即可 // char[] 转char *char str[] = "lala";char *str1 ...

  3. 9. Scala隐式转换和隐式值

    9.1 隐式转换 9.1.1 提出问题 先看一个案例演示,引出隐式转换的实际需要=>指定某些数据类型的相互转化 object boke_demo01 { def main(args: Array ...

  4. 【题解】选数字 [51nod1354]

    [题解]选数字 [51nod1354] 传送门:选数字 \([51nod1354]\) [题目描述] 共 \(T\) 组测试点,每一组给定一个长度为 \(n\) 的序列和一个整数 \(K\),找出有多 ...

  5. Queue介绍

    美人如斯! 前言 队列是一种先进先出(FIFO)的数据结构,与生活中的排队类似,即先来先被服务,这样的特点决定了其具有一定的优先级含义,可以被用于任务调度等场景.队列模型如图: 图1.队列模型 jav ...

  6. golang学习笔记 ---interface

    1. 什么是interface接口 interface 是GO语言的基础特性之一.可以理解为一种类型的规范或者约定.它跟java,C# 不太一样,不需要显示说明实现了某个接口,它没有继承或子类或“im ...

  7. linux部署安装SRS流媒体服务器教程

    这段时间一直在搞RTMP流媒体直播项目,期间踩过很多坑,刚开始是用的nginx-rtmp作为流媒体转发服务器,但是效果并不尽人意,推拉流不稳定,特别是拉流,速度特别慢,平均要十多秒才能拉到流,并且交互 ...

  8. java基础 super和this

    /** * super关键字的用法有三种: * 1.在子类的成员方法中,访问父类的成员变量 * 2.在子类的成员方法中,访问父类的成员方法 * 3.在子类的构造方法中,访问父类的构造方法 * * th ...

  9. 完全图解 HTTPS

    安全基础 我们先来看下数据在互联网上数据传递可能会出现的三个比较有代表性的问题,其实后面提到的所有方法,都是围绕解决这三个问题而提出来的. 窃听 伪造 否认 对称密钥加密 假设 A 正在通过互联网向  ...

  10. Golang 是否有必要内存对齐?

    原文:https://ms2008.github.io/2019/08/01/golang-memory-alignment/ 内存模型 Posted by ms2008 on August 1, 2 ...