zoukankan      html  css  js  c++  java
  • 【算法】分块——教主的魔法&不勤劳的图书管理员

    由不勤劳的图书管理员带入了分块的坑,深深地被其暴力与优雅所征服。分块的实质就是将暴力块状封装起来,一整块的部分就一整块处理,零碎的部分就怎么暴力怎么来。因为分块大小的原因,限制了零碎部分数据的数量级,所以复杂度得以保证。

    1.教主的魔法:可以算得上是一个分块的板子题。对于每一个块内sort排序,保存id值。对于修改,块内的找到点暴力修改之后重新排序,一整个块的不改变相对大小关系,所以直接外部记录累加的值。查询也一样,块外暴力,块内二分查找。

    #include <bits/stdc++.h>
    using namespace std;
    #define int long long
    #define maxn 2000000
    #define maxb 1050
    int n, q, B, sum[maxb], a[maxn], cnt[maxb];
    struct node
    {
        int v, id;
    }c[maxb][maxb];
    
    int read()
    {
        int x = 0;
        char c;
        c = getchar();
        while(c < '0' || c > '9') c = getchar();
        while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
        return x;
    }
    
    bool cmp(node a, node b)
    {
        return a. v < b.v;
    }
    
    void Vio_C(int b, int x, int w)
    {
        c[b][x].v += w;
    }
    
    void Change(int L, int R, int W)
    {
        int a = L / B, b = R / B;
        if(a == b)
        {
            for(int i = 1; i <= cnt[a]; i ++) 
                if(c[a][i].id >= L && c[a][i].id <= R) Vio_C(a, i, W);
                sort(c[a] + 1, c[a] + cnt[a] + 1, cmp);
         return 0; }
    for(int i = 1; i <= cnt[a]; i ++) if(c[a][i].id >= L) Vio_C(a, i, W); sort(c[a] + 1, c[a] + cnt[a] + 1, cmp); for(int i = 1; i <= cnt[b]; i ++) if(c[b][i].id <= R) Vio_C(b, i, W); sort(c[b] + 1, c[b] + cnt[b] + 1, cmp); for(int i = a + 1; i < b; i ++) sum[i] += W; } int check(int b, int C) { C -= sum[b]; int ans = cnt[b] + 1, l = 1, r = cnt[b]; while(l <= r) { int mid = (l + r) >> 1; if(c[b][mid].v >= C) ans = mid, r = mid - 1; else l = mid + 1; } return cnt[b] - ans + 1; } void Query(int L, int R, int C) { int a = L / B, b = R / B; int ans = 0; if(a == b) { C -= sum[a]; for(int i = 1; i <= cnt[a]; i ++) if(c[a][i].id >= L && c[a][i].id <= R && c[a][i].v >= C) ans ++; printf("%lld ", ans); return; } for(int i = 1; i <= cnt[a]; i ++) if(c[a][i].id >= L && c[a][i].v + sum[a] >= C) ans ++; for(int i = 1; i <= cnt[b]; i ++) if(c[b][i].id <= R && c[b][i].v + sum[b] >= C) ans ++; for(int i = a + 1; i < b; i ++) ans += check(i, C); printf("%lld ", ans); return; } signed main() { n = read(), q = read(); B = sqrt(n); for(int i = 1; i <= n; i ++) { a[i] = read(); int block = i / B; c[block][++ cnt[block]].v = a[i]; c[block][cnt[block]].id = i; if(((i / B) != (i + 1) / B) || i == n) sort(c[block] + 1, c[block] + cnt[block] + 1, cmp); } for(int i = 1; i <= q; i ++) { char c; cin >> c; int L = read(), R = read(), W = read(); if(c == 'M') Change(L, R, W); else Query(L, R, W); } return 0; }

    2.不勤劳的图书管理员

    这题首先注意到交换a,b的位置,只会影响到a,b之间的数。在一段区间里,a对杂乱值的贡献是多少?不难发现=在a之前且应当在a之后的书本杂乱值之和+v[a]*前面书本的个数。所以我们对每一个块使用两个树状数组,一个纪录个数,一个记录页数的前缀和,块内的利用树状数组快速查询,块外的暴力枚举计算即可。

    因为此题代码是Kuai的(那个时候还不会写分块),所以就不贴代码啦。

  • 相关阅读:
    Lua之math
    Lua之table
    Unity shader学习之轮廓效果
    c#之根据出生日期获得星座信息
    unity之复制文本到剪贴板
    Unity shader学习之屏幕后期处理效果之运动模糊
    Unity之如何从fbx提取Animation clip文件
    c#如何判断字符串是否含中文
    c#之如何转换文本文件编码格式为utf-8
    《Photon》
  • 原文地址:https://www.cnblogs.com/twilight-sx/p/8487856.html
Copyright © 2011-2022 走看看