思路:

莫队算法+树状数组。

莫队算法的基本思想是对大量要查询的区间进行离线处理,按照一定的顺序计算,来降低复杂度。概括来说,我们在知道了[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. vector draw 试用期结束的 激活方法

     [HKEY_LOCAL_MACHINE\SOFTWARE\Classes\Licenses\DBEA4D42-0745-428e-B17A-A5B6CA3AB34B] 把这个注冊表给删 了

  2. [Sciter] 资源引用

    http://www.cnblogs.com/yinxufeng/p/fb343eecda564aa63bce0bdf15709ddf.html 方式一. 加载外部文件方式二. 加载内存方式三. 加载 ...

  3. VC2010 利用 def 文件生成 dll 文件的方法

    近期有个需求,要生成一个dll 文件.文件里的函数都是採用 stdcall 函数调用约定,可是不希望函数名被修饰(add 被修饰成 add@8). 这时就要用def 文件了. 比方我有以下两个函数: ...

  4. Effective C++学习笔记(Part Four:Item 18-25)

     近期最终把effectvie C++细致的阅读了一边.非常惊叹C++的威力与魅力.近期会把近期的读书心得与读书笔记记于此,必备查找使用,假设总结有什么不 当之处,欢迎批评指正: 如今仅仅列出框架 ...

  5. 珠海鼎芯(D-Chip)IMX6读取CPU的UID的方法【转】

    本文转载自:http://blog.csdn.net/williamdedong/article/details/52712084 在使用IMX6板子的时候,有时会想着是否可以把板子搞一个唯一标识呢, ...

  6. 基于TCP的字符串传输程序

    ---恢复内容开始--- LINUX中的网络编程是通过SOCKET接口来进行的. Socket(套接字) Socket相当于进行网络通信两端的插座,只要对方的Socket和自己的Socket有通信联接 ...

  7. HAProxy+Keepalived构建高可用负载均衡

    http://www.linuxidc.com/Linux/2012-03/55880.htm web1 IP 192.168.0.47 web2 IP 192.168.0.48 haproxy_ma ...

  8. mac idea 内存

    vim /Applications/IntelliJ\ IDEA.app/Contents/bin/idea.vmoptions -Xms512m -Xmx2048m -XX:ReservedCode ...

  9. CSYZDay2模拟题解

    T1.rotate [问题描述] ZYL有N张牌编号分别为1, 2,……,N.他把这N张牌打乱排成一排,然后他要做一次旋转使得旋转后固定点尽可能多.如果第i个位置的牌的编号为i,我们就称之为固定点.旋 ...

  10. AutoIT: 捕捉路由器登陆的小例子

    $url2 = "AX.XXX.XXX.COM" $oIE = _IECreate ($url2, , , , ) Sleep() ControlSetText('[REGEXPT ...