思路:

莫队算法+树状数组。

莫队算法的基本思想是对大量要查询的区间进行离线处理,按照一定的顺序计算,来降低复杂度。概括来说,我们在知道了[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 排队接水的更多相关文章

  1. hihocoder offer收割编程练习赛11 C 岛屿3

    思路: 并查集的应用. 实现: #include <iostream> #include <cstdio> using namespace std; ][]; int n, x ...

  2. hihocoder offer收割编程练习赛11 B 物品价值

    思路: 状态压缩 + dp. 实现: #include <iostream> #include <cstdio> #include <cstring> #inclu ...

  3. hihocoder offer收割编程练习赛11 A hiho字符串

    思路: 我用的尺取. 注意题目描述为恰好2个'h',1个'i',1个'o'. 实现: #include <iostream> #include <cstdio> #includ ...

  4. hihocoder [Offer收割]编程练习赛4

    描述 最近天气炎热,小Ho天天宅在家里叫外卖.他常吃的一家餐馆一共有N道菜品,价格分别是A1, A2, ... AN元.并且如果消费总计满X元,还能享受优惠.小Ho是一个不薅羊毛不舒服斯基的人,他希望 ...

  5. hihocoder [Offer收割]编程练习赛61

    [Offer收割]编程练习赛61 A:最小排列 给定一个长度为m的序列b[1..m],再给定一个n,求一个字典序最小的1~n的排列A,使得b是A的子序列. 贪心即可,b是A的子序列,把不在b中的元素, ...

  6. 【[Offer收割]编程练习赛11 D】排队接水

    [题目链接]:http://hihocoder.com/problemset/problem/1488 [题意] 中文题 [题解] 莫队算法+树状数组; 首先贪心地知道,应该按照时间从小到大的顺序打水 ...

  7. ACM学习历程—Hihocoder [Offer收割]编程练习赛1

    比赛链接:http://hihocoder.com/contest/hihointerview3/problem/1 大概有一个月没怎么打算法了.这一场的前一场BC,也打的不是很好.本来Div1的A和 ...

  8. hihocoder offer收割编程练习赛8 C 数组分拆

    思路:(引自bfsoyc的回答:http://hihocoder.com/discuss/question/4160) 动态规划.状态dp[i]表示 前i个数的合法的方案数,转移是 dp[i] = s ...

  9. 【[Offer收割]编程练习赛11 B】物品价值

    [题目链接]:http://hihocoder.com/problemset/problem/1486 [题意] [题解] 设f[i][j]表示前i个物品,每种属性的状态奇偶状态为j的最大价值; 这里 ...

随机推荐

  1. leetcode:238. Product of Array Except Self(Java)解答

    转载请注明出处:z_zhaojun的博客 原文地址 题目地址 Product of Array Except Self Given an array of n integers where n > ...

  2. Qt学习--初学注意事项

    过程.心得: 1)Qt Creator与相关的安装包的安装 我在选择去学习Qt之后,第一件事就是Qt SDK下载安装与配置.最初,在网上发现Qt使用的IDE环境        在Windows上可以选 ...

  3. Linux bash介绍与使用

    Linux————bash的简单使用 对于一个操作系统来说,shell相当于内核kernel外的一层外壳,作为用户接口.一般来说,操作系统的接口分为两类:CLI:command line interf ...

  4. VTMagic的使用

    // VTMagic的使用 //  CFOrderViewController.m //  qifuyuniOS //// /** *  @author 李洪强, 16-08-30 10:08:50 ...

  5. malloc内存分配

    网上总结到的信息: (1) 静态分派:是在栈上分配,是由用户自己申请,是由操作系统自己释放的 动态分配:是由编译器分配,操作系统没有提供这样的机制,所以自己申请,必须自己删除! (2)你也要明确.栈的 ...

  6. 行转列--Excel和数据库的完美结合

    入职到如今已经有一段时间了,除了不断完好新功能外,线上运维也是一项非常重要的任务,每天都须要占用一 定量的时间来处理,这时候如何高效准确的来解决这些问题是非常值得考虑的.       今天客服人员给我 ...

  7. HDU 2457/POJ 3691 DNA repair AC自动机+DP

    DNA repair Problem Description   Biologists finally invent techniques of repairing DNA that contains ...

  8. java8--NIO(java疯狂讲义3复习笔记)

    NIO采用内存映射文件的方式处理输入输出,NIO将文件或文件的一段区域映射到内存中,这样就可以像访问内存一样来访问文件了(这种方式模拟了操作系统上的虚拟内存的概念),通过这种方式来进行输入输出比传统的 ...

  9. 运行tomcat6w.exe ,提示 指定的服务未安装 unable to open the service 'tomcat6'

    错误:运行tomcat6w.exe ,提示 指定的服务未安装 unable to open the service 'tomcat6'(我用的是官网下载的解压版) 解决方法: 打开命令行提示符窗口=& ...

  10. HBase使用教程

    1     基本介绍 1.1 前言 HBase – Hadoop Database.是一个分布式的.面向列的开源数据库,该技术来源于 Fay Chang 所撰写的Google论文"Bigta ...