hihocoder offer收割编程练习赛11 D 排队接水
思路:
莫队算法+树状数组。
莫队算法的基本思想是对大量要查询的区间进行离线处理,按照一定的顺序计算,来降低复杂度。概括来说,我们在知道了[l, r]的解,并且可以通过一个较低的复杂度推出[l - 1, r], [l, r - 1], [l + 1, r], [l, r + 1]的解的情况下,则可使用该算法。
对该算法比较好的介绍:
1.https://blog.sengxian.com/algorithms/mo-s-algorithm
2.http://blog.csdn.net/bossup/article/details/39236275
在本题中,对于一个区间[l, r],实际上是求a[l] * (r - l + 1) + a[l + 1] * (r - l) +... + a[r] * 1。
那么在知道了[l, r]的解的前提下,如何转移到[l', r']呢?
我们可以观察一个简单的情况:[l, r] -> [l - 1, r]。比如[1, 2, 3, 4] -> [3, 1, 2, 3, 4]。
对于这个例子,假设原来的解为res。那么在左边加上一个‘3’之后,新的解res_new为
1 * 5 + 2 * 4 + 3 * 3 + 3 * 2 + 4 * 1 =
1 * 4 + 2 * 3 + 3 * 2 + 4 * 1 + 1 + 2 + 3 * 3 =
res + 1 + 2 + 3 * 3。
实际上就是原来的解res + 原来的区间内比3小的数的和 + 3 * (原来的区间内大于等于3的数的个数 + 1);
反过来,[3, 1, 2, 3, 4] -> [1, 2, 3, 4]的过程类似。
我们可以使用两个树状数组来维护变化的部分。具体来说其中一个维护大于等于3的数的个数,另一个维护小于3的数的和即可。
实现:
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std;
typedef long long ll; const int N = ; int t, n, m, L, R, a[N + ], block;
ll tot = , ans[N + ];
ll bit1[N + ]; //统计个数
ll bit2[N + ]; //统计和 struct query
{
int l, r, id;
};
query Q[N + ]; bool cmp(const query & x, const query & y)
{
if (x.l / block != y.l / block)
return x.l / block < y.l / block;
return x.r < y.r;
} void bit_add(ll * bit, int i, int x)
{
if (i == )
return;
while (i <= N)
{
bit[i] += x;
i += i & -i;
}
} ll bit_sum(ll * bit, int i)
{
ll s = ;
while (i)
{
s += bit[i];
i -= i & -i;
}
return s;
} ll bit_query(ll * bit, int l ,int r)
{
return bit_sum(bit, r) - bit_sum(bit, l - );
} void Add(int pos)
{
tot += (ll)a[pos] * (bit_query(bit1, a[pos], N + ) + );
tot += bit_query(bit2, , a[pos] - );
bit_add(bit1, a[pos], );
bit_add(bit2, a[pos], a[pos]);
} void Del(int pos)
{
tot -= (ll)a[pos] * bit_query(bit1, a[pos], N + );
tot -= bit_query(bit2, , a[pos] - );
bit_add(bit1, a[pos], -);
bit_add(bit2, a[pos], -a[pos]);
} void init()
{
block = sqrt(n);
L = R = ;
tot = a[];
memset(bit1, , sizeof(bit1));
memset(bit2, , sizeof(bit2));
bit_add(bit1, a[], );
bit_add(bit2, a[], a[]);
} int main()
{
cin >> t;
while (t--)
{
cin >> n >> m;
for (int i = ; i <= n; i++)
{
scanf("%d", &a[i]);
}
init();
for (int i = ; i < m; i++)
{
scanf("%d %d", &Q[i].l, &Q[i].r);
Q[i].id = i;
}
sort(Q, Q + m, cmp);
for (int i = ; i < m; i++)
{
while (L < Q[i].l)
Del(L++);
while (L > Q[i].l)
Add(--L);
while (R < Q[i].r)
Add(++R);
while (R > Q[i].r)
Del(R--);
ans[Q[i].id] = tot;
}
for (int i = ; i < m; i++)
{
printf("%lld\n", ans[i]);
}
}
return ;
}
hihocoder offer收割编程练习赛11 D 排队接水的更多相关文章
- hihocoder offer收割编程练习赛11 C 岛屿3
思路: 并查集的应用. 实现: #include <iostream> #include <cstdio> using namespace std; ][]; int n, x ...
- hihocoder offer收割编程练习赛11 B 物品价值
思路: 状态压缩 + dp. 实现: #include <iostream> #include <cstdio> #include <cstring> #inclu ...
- hihocoder offer收割编程练习赛11 A hiho字符串
思路: 我用的尺取. 注意题目描述为恰好2个'h',1个'i',1个'o'. 实现: #include <iostream> #include <cstdio> #includ ...
- hihocoder [Offer收割]编程练习赛4
描述 最近天气炎热,小Ho天天宅在家里叫外卖.他常吃的一家餐馆一共有N道菜品,价格分别是A1, A2, ... AN元.并且如果消费总计满X元,还能享受优惠.小Ho是一个不薅羊毛不舒服斯基的人,他希望 ...
- hihocoder [Offer收割]编程练习赛61
[Offer收割]编程练习赛61 A:最小排列 给定一个长度为m的序列b[1..m],再给定一个n,求一个字典序最小的1~n的排列A,使得b是A的子序列. 贪心即可,b是A的子序列,把不在b中的元素, ...
- 【[Offer收割]编程练习赛11 D】排队接水
[题目链接]:http://hihocoder.com/problemset/problem/1488 [题意] 中文题 [题解] 莫队算法+树状数组; 首先贪心地知道,应该按照时间从小到大的顺序打水 ...
- ACM学习历程—Hihocoder [Offer收割]编程练习赛1
比赛链接:http://hihocoder.com/contest/hihointerview3/problem/1 大概有一个月没怎么打算法了.这一场的前一场BC,也打的不是很好.本来Div1的A和 ...
- hihocoder offer收割编程练习赛8 C 数组分拆
思路:(引自bfsoyc的回答:http://hihocoder.com/discuss/question/4160) 动态规划.状态dp[i]表示 前i个数的合法的方案数,转移是 dp[i] = s ...
- 【[Offer收割]编程练习赛11 B】物品价值
[题目链接]:http://hihocoder.com/problemset/problem/1486 [题意] [题解] 设f[i][j]表示前i个物品,每种属性的状态奇偶状态为j的最大价值; 这里 ...
随机推荐
- hdu2222--Keywords Search+AC自己主动机模板
题目链接:pid=2222">点击进入 KMP对模式串进行处理.然后就能够方便的推断模式串是否在目标串中出现了:这显示适合一个模式串多个目标串的情况.可是假设模式串有多个,这时假设还用 ...
- Redis入门教程(三)— Java中操作Redis
在Redis的官网上,我们可以看到Redis的Java客户端众多 其中,Jedis是Redis官方推荐,也是使用用户最多的Java客户端. 开始前的准备 使用jedis使用到的jedis-2.1.0. ...
- IO流(SequenceInputStream序列流--文件拆分与合并)
一.文件拆分 1.将一个文件按照规定的大小进行分割成多个文件并将被分割详细信息保存至配置信息中 Properties 类表示了一个持久的属性集.Properties 可保存在流中或从流中加载,属性列表 ...
- Xcode中使用git
项目中添加git 也可在开始新建项目时勾选git,这是针对开始没有勾选git的情况 打开终端 cd 项目文件目录 //初始化一个代码仓库, git init //将当前目录及子目录中的文件标记为要添加 ...
- (ros/moveit)cob_simulation報錯
cob_simulation報錯 依照官網說明 http://wiki.ros.org/cob_bringup_sim 1. git clone https://github.com/ipa320/c ...
- java生成随机汉字
方法一: public static char getRandomChar() { return (char) (0x4e00 + (int) (Math.random() * (0x9fa5 - 0 ...
- YTU 2955: A改错题--销售部的打印机
2955: A改错题--销售部的打印机 时间限制: 1 Sec 内存限制: 128 MB 提交: 61 解决: 47 题目描述 销售部新进了一台快速打印机,使用频率很高.为了能够对打印情况进行统计 ...
- 如何完成DEDE CMS外部数据库调用|不同数据库调用数据
dedecms如何完成2个数据库内容彼此调用?这是笔者今日要和我们共享的内容.百度了一大堆,大多语焉不详.常识有限,所以就说下笔者的做法, 能够还有其他有用的办法,欢送共享.笔者站点是dedecms5 ...
- Linux 杀死所有进程
方法一: sudo killall -9 netease-cloud-music 这种方法,必须要写全称. sudo netease-cloud-music QStandardPaths: XDG_R ...
- SPOJ:Triple Sums(母函数+FFT)
You're given a sequence s of N distinct integers.Consider all the possible sums of three integers fr ...