zoukankan      html  css  js  c++  java
  • P2709 小B的询问 莫队

    这个题,莫队很容易想到(格式很明显),然后直接用数学公式维护一下和就行了。

    题干:

    题目描述
    小B有一个序列,包含N个1~K之间的整数。他一共有M个询问,每个询问给定一个区间[L..R],求Sigma(c(i)^2)的值,其中i的值从1到K,其中c(i)表示数字i在[L..R]中的重复次数。小B请你帮助他回答询问。
    输入输出格式
    输入格式:
    第一行,三个整数N、M、K。
    第二行,N个整数,表示小B的序列。
    接下来的M行,每行两个整数L、R。
    输出格式:
    M行,每行一个整数,其中第i行的整数表示第i个询问的答案

    代码:

    #include<iostream>
    #include<cstdio>
    #include<cmath>
    #include<ctime>
    #include<queue>
    #include<algorithm>
    #include<cstring>
    using namespace std;
    #define duke(i,a,n) for(register int i = a;i <= n;i++)
    #define lv(i,a,n) for(register int i = a;i >= n;i--)
    #define clean(a) memset(a,0,sizeof(a))
    const int INF = 1 << 30;
    typedef long long ll;
    typedef double db;
    template <class T>
    void read(T &x)
    {
        char c;
        bool op = 0;
        while(c = getchar(), c < '0' || c > '9')
            if(c == '-') op = 1;
        x = c - '0';
        while(c = getchar(), c >= '0' && c <= '9')
            x = x * 10 + c - '0';
        if(op) x = -x;
    }
    template <class T>
    void write(T x)
    {
        if(x < 0) putchar('-'), x = -x;
        if(x >= 10) write(x / 10);
        putchar('0' + x % 10);
    }
    struct node
    {
        int l,r,id,pos;
    }a[50005];
    int blo,n,m,k;
    ll b[50005],num[56000];
    ll cnt[50005];
    bool cmp(node a,node b)
    {
        if(a.pos != b.pos) return a.pos < b.pos;
        else
        return a.r < b.r;
    }
    int main()
    {
        read(n);read(m);read(k);
        blo = sqrt(n);
        duke(i,1,n)
        {
            read(b[i]);
        }
        duke(i,1,m)
        {
            read(a[i].l);
            read(a[i].r);
            a[i].id = i;
            a[i].pos = (a[i].l - 1) / blo + 1;
        }
        sort(a + 1,a + m + 1,cmp);
        int l  = 1,r = 0;
        ll ans = 0;
        duke(i,1,m)
        {
            while(l > a[i].l) l--,cnt[b[l]]++,ans += 2 * cnt[b[l]] - 1;
            while(r < a[i].r) r++,cnt[b[r]]++,ans += 2 * cnt[b[r]] - 1;
            while(l < a[i].l) cnt[b[l]]--,ans -= 2 * cnt[b[l]] + 1,l++;
            while(r > a[i].r) cnt[b[r]]--,ans -= 2 * cnt[b[r]] + 1,r--;
            num[a[i].id] = ans;
        }
        duke(i,1,m)
        {
            printf("%lld
    ",num[i]);
        }
        return 0;
    }
  • 相关阅读:
    fiddler的详细分析
    在Linux中mysql的一些基本操作
    MySQL运行状态show status详解
    第1节-软件测试基本概念及分类
    HTML
    python 目录
    Web框架 Bottle 、Flask 、Tornado
    Query
    JavaScript基础篇
    DOM、BOM 操作
  • 原文地址:https://www.cnblogs.com/DukeLv/p/9868587.html
Copyright © 2011-2022 走看看