[voj 1551]E - Pairs 2014年武汉大学邀请赛E题 莫队算法

题目大意

有n个数,m个查询,对于每个查询,询问指定区间,有多少个数对的绝对值小于等于2。

解题思路

莫队O^1.5

首先将询问离线处理左端点进行编号,每sqrt(n)个为一组

sort结构体 当左端点编号相同时,比较右端点大小。小的放在前面。

对于每组询问暴力处理,只需处理当前新加入(删除的数字在当前区间内有多少点和它的绝对值只差小于2即可)

唯一要注意的是加点是先更新答案再计数,删点是先计数器-1再更新答案

因为对于每个询问,左端点的修改量不超过sqrt(n)

右端点每一组最坏的复杂度是修改n次 共sqrt(n)组

m,n相当的情况下,莫队算法整体的时间复杂度是n^1.5

code:

#include <cstdio>
#include <iostream>
#include <algorithm>
#include <ctime>
#include <cctype>
#include <cmath>
#include <string>
#include <cstring>
#include <stack>
#include <queue>
#include <list>
#include <vector>
#include <map>
#include <set>

#define sqr(x) ((x)*(x))
#define LL long long
#define INF 0x3f3f3f3f
#define PI acos(-1.0)
#define eps 1e-10
using namespace std;
int n,m,bl;
int cnt[100010];
int a[100010];
struct node{
    int l,r,x;
}s[100005];
LL anss[100005];
int cmp(node a,node b)
{
    if (a.l/bl==b.l/bl) return a.r<b.r;
    return (a.l/bl)<(b.l/bl);
}
inline LL count(int x)
{
    return cnt[x-2]+cnt[x-1]+cnt[x]+cnt[x+1]+cnt[x+2];
}
int main(int argc, char const *argv[])
{
    int ca=0;
    while (~scanf("%d%d",&n,&m))
    {
        bl=sqrt((double)n);
        for (int i=1;i<=n;i++){
            scanf("%d",&a[i]);
            a[i]+=5;//为了防止处理边界带来的多余的复杂度,直接全部加上5
        }
        for (int i=1;i<=m;i++)
            {
                scanf("%d%d",&s[i].l,&s[i].r);
                s[i].x=i;
            }
        sort(s+1,s+1+m,cmp);
        memset(cnt,0,sizeof cnt);
        int l=1,r=1;
        cnt[a[1]]=1;
        LL ans=0;
        for (int i=1;i<=m;i++)
        {
            if (r<s[i].r){
                for (;r-s[i].r;)
                {
                    r++;
                    ans+=count(a[r]);
                    cnt[a[r]]++;
                }
            }
            if (r>s[i].r){
                for (;r-s[i].r;)
                {
                    cnt[a[r]]--;
                    ans-=count(a[r]);
                    r--;
                }
            }
            if (l<s[i].l){
                for (;l-s[i].l;)
                {
                    cnt[a[l]]--;
                    ans-=count(a[l]);
                    l++;
                }
            }
            if (l>s[i].l){
                for (;l-s[i].l;)
                {
                    l--;
                    ans+=count(a[l]);
                    cnt[a[l]]++;
                }
            }
            anss[s[i].x]=ans;
        }
        printf("Case %d:\n", ++ca);
        for (int i = 1; i <= m; ++i)
        {
            printf("%lld\n",anss[i]);
        }
    }
    return 0;
}
时间: 2024-10-22 03:09:44

[voj 1551]E - Pairs 2014年武汉大学邀请赛E题 莫队算法的相关文章

2014湘潭全国邀请赛I题 Intervals /POJ 3680 / 在限制次数下取有权区间使权最大/小问题(费用流)

先说POJ3680:给n个有权(权<10w)开区间(n<200),(区间最多数到10w)保证数轴上所有数最多被覆盖k次的情况下要求总权最大,输出最大权. 思路:       限制的处理:s-->开始流量为k,要求总权最大,即费用最大,所以费用取负,最小费用最大流即可.对于输入区间[a,b]:w,添加边:a-->b,流量为1,费用为-w. 对于点i,i+1,添加边,费用为0,流量无穷.显然这种处理,限制了区间最多取k次,(流量控制),跑最大流能走添加的边尽量走,且越大越好(负数刚刚是

whu oj 1551 Pairs (莫队算法)

problem_id=1551">题目链接 题目大意: 给出的询问,求出这个区间的里 差小于等于 2 的数字的对数. 思路分析: 莫队算法. 然后分析一下. 假设添加了一个数字.那么就要加它旁边相差为2 的数字的和. 反之降低一个.就要降低相差为2 的数字的和.再减去自己这个1.. #include <cstdio> #include <iostream> #include <cstring> #include <algorithm> #in

BNUOJ 34985 Elegant String 2014北京邀请赛E题 动态规划 矩阵快速幂

Elegant String Time Limit: 1000msMemory Limit: 65536KB 64-bit integer IO format: %lld      Java class name: Main We define a kind of strings as elegant string: among all the substrings of an elegant string, none of them is a permutation of "0, 1,-, k

2014 BNU 邀请赛E题(递推+矩阵快速幂)

Elegant String 题意:给定一个字符串,由0-k数字组成,要求该串中,子串不包含0-k全排列的方案数 思路:dp[i][j]表示放到i个数字,后面有j个不相同,然后想递推式,大概就是对应每种情况k分别能由那几种状态转移过来,在纸上画画就能构造出矩阵了,由于n很大,所以用快速幂解决 代码: #include <stdio.h> #include <string.h> const long long MOD = 20140518; int t; long long n; i

BNUOJ 34985 Elegant String 2014北京邀请赛E题 矩阵快速幂

题目链接:http://acm.bnu.edu.cn/bnuoj/problem_show.php?pid=34985 题目大意:问n长度的串用0~k的数字去填,有多少个串保证任意子串中不包含0~k的某一个全排列 邀请赛上A的较多的一道题,比赛的时候死活想不出,回来之后突然就想通了,简直..... = =! 解题思路: 对于所有串我们都只考虑末尾最多有多少位能构成全排列的一部分(用l来表示),即最多有多少位不重复的数字出现,将问题转化为求末尾最多有k位能构成全排列的串的总数量 假设k为5,有一个

2014北京邀请赛E题-矩阵快速幂

题意:长度为n(1<=n<=10^18)的并且任意连续子串都不是0-k(1<=k<=9)的一个排列的字符串有多少种. 解法:矩阵快速幂.dp[i][j]表示i长度最后连续j个不同(即最后j个无重复,最后j+1个有重复)的字符串的个数.状态选好很重要.设计状态时最重要考虑是唯一性和可传递性,比赛时明明知道肯定是矩阵快速幂,但是一直没想到这个状态表示,自己设计的自己都不会转移. dp[i][j]有了后,后边加一个字符,这个字符可以是j之内的任意一个,也可以是j以外的,这样枚举每种情况,

HDOJ Page Rank 5097【2014上海邀请赛H题-简单矩阵】

Page Rank Time Limit: 3000/1500 MS (Java/Others)    Memory Limit: 100000/100000 K (Java/Others) Total Submission(s): 282    Accepted Submission(s): 77 Problem Description Evaluation and rank of web pages is a hot topic for many internet companies and

2014 BNU 邀请赛A题(构造问题)

A Matrix 题意:按照题目中给定的方法,给你一个矩阵,求出变换出该矩阵的字符串 思路:构造问题,在纸上多画几组就能发现,每次必须从上往下找到一条路径,最后输出这些路径,按照开头最大的最晚输出,找的过程中只要不断往下一层找一个大的即可,并且如果一开使有一行是非递增就是错误 代码: #include <stdio.h> #include <string.h> #include <vector> #include <map> using namespace

2014 BNU 邀请赛B题(枚举)

Beautiful Garden 题意:x轴上放了一些树,现在要移动一些树使得所有树都等间距,问最少要移动多少棵 思路:枚举,枚举第一棵树,和另一棵树,以及中间有多少树,这样就能知道等差数列的首项和公差,然后再循环一边计算出答案,保存最小值 代码: #include <stdio.h> #include <string.h> #include <algorithm> #include <math.h> using namespace std; #define