[woj 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;
}



  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
单片微型计算机(MCU)经过多的发展,在性能上有很大的进步,在型号上发展到上千种类,已经广泛应用于人类社会生活的各个领域。单片机课程已经成为高校计算机、自动化、测控以及电子信息工程等专业的重要课程。该课程是一门理论性和实践性都很强的课程,在实际教学中,应将理论教学和实验教学紧密结合。学生在掌握理论知识之余,必须通过编写程序、设计硬件电路、仿真、调试这一系列的实验过程,才能更好地掌握单片机的结构原理和应用技能。随着单片机及其接口技术的飞速发展,目前市场上供应的编程仿真实验资源并不能完全满足高校单片机课程教与学的需求,构建低成本、技术先进、源码公开的单片机编程仿真实验系统,对我国单片机课程的教学和单片机领域人才的培养具有重要的现实意义。 本论文结合目前教学中对单片机编程仿真实验系统的实际需求,采用模块化结构设计思想,精心设计和开发了单片机编程仿真实验系统。该单片机编程仿真实验系统由PC机端单片机编程控制软件和单片机编程仿真实验板两部分组成。PC机端的单片机编程控制软件可以自动检测到连接到单片机编程仿真实验板上的单片机,控制单片机编程器擦除、写入、读出、校验目标单片机ROM中的程序,以十六进制文件(.HEX文件)格式显示在控制界面内;单片机仿真实验系统能够把写入单片机的程序实时地运行,并呈现实际运行效果。单片机编程控制软件和单片机仿真实验板组成一个完整的单片机编程仿真实验系统。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值