Luogu 题面

题目描述

给定一串长度为 \(n\) 的数字,数字为 \(0 \sim 9\) 之间的任意一个,下标从 \(1\) 记起。

然后进行 \(m\) 次区间查询,每次查找区间 \([l, r]\) 的区间和,并在查询结束后将区间里的每一个数都 $ + 1$。特殊地,如果 $ + 1$ 前的数字为 \(9\),那么 $ + 1$ 之后就变成了 \(0\)。

输出每次查询的区间和。

题目分析

使用支持区间加和区间求和的数据结构即可。在这里使用分块和线段树。

分块

跑的飞快。

\(9 + 1 \to 0\) 这个限制非常的难搞。然而我们发现一共只有十个数字,因此不管怎么暴力搞都是没问题的(大不了乘个大常数)。

在每个块内维护以下信息:当前块内 \(1 \sim 9\) 每个数出现的次数(\(cnt_i\)),当前块的懒标记(\(add\))。那么一个块内的区间和就是:

\[sum = \sum_{i \in [0, 9]} cnt_i \times [(i + add) \mod 10]
\]

代码示例如下:

#include <algorithm>
#include <iostream>
#include <cstring>
#include <cstdio>
#include <cmath> using namespace std; using LL = long long;
using PII = pair<int, int>;
using PLL = pair<LL, LL>; const int N = 250010, M = (int)sqrt(N) + 20;
const int INF = 0x3f3f3f3f; int n, m, len, w[N];
struct Block {
int l = INF, r = -INF, add;
int cnt[10];
}b[M]; int get(int x) {
return (int)x / len;
} void change(int k, int &x) {
b[k].cnt[x] -- ;
b[k].cnt[(x + 1) % 10] ++ ;
x = (x + 1) % 10;
} void modify(int l, int r) {
int lc = get(l), rc = get(r);
if (lc == rc) {
for (int i = l; i <= r; i ++ )
change(lc, w[i]);
return;
}
for (int i = l; i <= b[lc].r; i ++ ) change(lc, w[i]);
for (int i = r; i >= b[rc].l; i -- ) change(rc, w[i]);
for (int i = lc + 1; i <= rc - 1; i ++ )
b[i].add = (b[i].add + 1) % 10;
} int query_block(int k) {
int ans = 0;
for (int i = 0; i <= 9; i ++ )
ans += b[k].cnt[i] * ((i + b[k].add) % 10);
return ans;
} int query(int l, int r) {
int lc = get(l), rc = get(r);
int ans = 0;
if (lc == rc) {
for (int i = l; i <= r; i ++ )
ans += (w[i] + b[lc].add) % 10;
return ans;
}
for (int i = l; i <= b[lc].r; i ++ ) ans += (w[i] + b[lc].add) % 10;
for (int i = r; i >= b[rc].l; i -- ) ans += (w[i] + b[rc].add) % 10;
for (int i = lc + 1; i <= rc - 1; i ++ )
ans += query_block(i);
return ans;
} int main() {
scanf("%d%d", &n, &m); len = (int)sqrt(n);
for (int i = 1; i <= n; i ++ )
scanf("%1d", &w[i]);
for (int i = 1; i <= n; i ++ ) {
b[get(i)].l = min(b[get(i)].l, i);
b[get(i)].r = max(b[get(i)].r, i);
b[get(i)].cnt[w[i]] ++ ;
} while (m -- ) {
int l, r;
scanf("%d%d", &l, &r);
printf("%d\n", query(l, r));
modify(l, r);
} return 0;
}

单次询问复杂度为 \(O(\sqrt{n})\),空间复杂度 \(O(\sqrt{n})\)

线段树

思路与分块相同,即在每一个节点维护每个数出现的次数。注意 pushuppushdown 的处理。

单次询问复杂度 \(O(\log n)\) 明显快于分块。空间复杂度 \(O(n)\) 略劣。

代码示例如下:

#include <algorithm>
#include <iostream>
#include <cstring>
#include <cstdio> using namespace std; const int N = 250010; int w[N], n, m;
struct Tree {
int l, r;
int sum, add;
int cnt[10];
}tr[N << 2];
#define ls u << 1
#define rs u << 1 | 1 void pushup(int u) {
tr[u].sum = tr[ls].sum + tr[rs].sum;
for (int i = 0; i <= 9; i ++ )
tr[u].cnt[i] = tr[ls].cnt[i] + tr[rs].cnt[i];
} void push(int u, int k) { // 这里需要注意
tr[u].add += k;
while (k -- ) {
for (int i = 9; i; i -- )
swap(tr[u].cnt[i], tr[u].cnt[i - 1]);
}
tr[u].sum = 0;
for (int i = 1; i <= 9; i ++ )
tr[u].sum += tr[u].cnt[i] * i;
} void pushdown(int u) {
if (tr[u].add) {
tr[u].add %= 10;
push(ls, tr[u].add);
push(rs, tr[u].add);
tr[u].add = 0;
}
} void build(int u, int l, int r) {
tr[u] = {l, r};
if (l == r) {
tr[u].sum = w[r];
tr[u].cnt[w[r]] ++ ;
return;
}
int mid = l + r >> 1;
build(ls, l, mid), build(rs, mid + 1, r);
pushup(u);
} void modify(int u, int l, int r) {
if (tr[u].l >= l && tr[u].r <= r)
return (void)push(u, 1);
pushdown(u);
int mid = tr[u].l + tr[u].r >> 1;
if (l <= mid) modify(ls, l, r);
if (r > mid) modify(rs, l, r);
pushup(u);
} int query(int u, int l, int r) {
if (tr[u].l >= l && tr[u].r <= r)
return tr[u].sum;
pushdown(u);
int mid = tr[u].l + tr[u].r >> 1, sum = 0;
if (l <= mid) sum += query(ls, l, r);
if (r > mid) sum += query(rs, l, r);
return sum;
} int main() {
scanf("%d%d", &n, &m);
for (int i = 1; i <= n; i ++ )
scanf("%1d", &w[i]); build(1, 1, n); while (m -- ) {
int l, r;
scanf("%d%d", &l, &r);
printf("%d\n", query(1, l, r));
modify(1, l, r);
}
return 0;
}

P6357 题解的更多相关文章

  1. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

  2. noip2016十连测题解

    以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...

  3. BZOJ-2561-最小生成树 题解(最小割)

    2561: 最小生成树(题解) Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1628  Solved: 786 传送门:http://www.lyd ...

  4. Codeforces Round #353 (Div. 2) ABCDE 题解 python

    Problems     # Name     A Infinite Sequence standard input/output 1 s, 256 MB    x3509 B Restoring P ...

  5. 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解

    题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...

  6. 2016ACM青岛区域赛题解

    A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Jav ...

  7. poj1399 hoj1037 Direct Visibility 题解 (宽搜)

    http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...

  8. 网络流n题 题解

    学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...

  9. CF100965C题解..

    求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...

  10. JSOI2016R3 瞎BB题解

    题意请看absi大爷的blog http://absi2011.is-programmer.com/posts/200920.html http://absi2011.is-programmer.co ...

随机推荐

  1. 基于AvaSpe 2048测定物体的光谱曲线

      本文介绍基于AvaSpec-ULS2048x64光纤光谱仪测定植被.土壤等地物高光谱曲线的方法.   AvaSpec是由荷兰著名的光纤光谱仪器与系统开发公司Avantes制造的系列高性能光谱仪,广 ...

  2. 搞懂fflush(stdout)

    使用 printf 或 cout 打印内容时,输出永远不会直接写入"屏幕".而是,被发送到 stdout. (stdout 就像一个缓冲区) 默认情况下,发送到 stdout 的输 ...

  3. Python经典编程题40题(二)

    Python经典编程题40题(二)    题目 给你一个list L, 如 L=[2,8,3,50], 对L进行降序排序并输出, 如样例L的结果为[50,8,3,2] 输入示例 输入:L = [4, ...

  4. Arduino 麦克风声音传感器指南

    麦克风声音传感器 麦克风声音传感器,顾名思义,检测声音.它可以测量声音的响度. 这些传感器的种类繁多.  在下图中,您可以看到 Arduino 最常用的. 最左边是KY-038,右边是LM393麦克风 ...

  5. Dotnet工具箱:开源、免费的纯前端工具网站,带你探索10大工具分类和73个实时在线小工具

    1. 前言 大家好,我是沙漠尽头的狼. Dotnet工具箱是一个纯前端的.开源和免费的工具网站,周末我参考了开源项目it-tools,对网站界面文字进行了汉化,并重新部署了网站.该网站共有10大工具分 ...

  6. Swagger系列:SpringBoot3.x中使用Knife4j

    目录 一.简介 二.版本说明 三.使用 四.效果图 一.简介 官网:https://doc.xiaominfo.com/ Knife4j是一个集Swagger2 和 OpenAPI3 为一体的增强解决 ...

  7. 虹科分享 | 一起聊聊Redis企业版数据库与【微服务误解】那些事儿!

    如今,关于微服务依然存在许多误解,企业盲目追求这种炫酷技术并不可取.同时,这种盲目行为对于希望用微服务来有效解决问题的公司很不利.不是说任何特定的技术都是缺乏实际价值的,如微服务.Kubernetes ...

  8. Java服务总在半夜挂,背后的真相竟然是...

    写在前面 最近有用户反馈测试环境Java服务总在凌晨00:00左右挂掉,用户反馈Java服务没有定时任务,也没有流量突增的情况,Jvm配置也合理,莫名其妙就挂了 问题排查 问题复现 为了复现该问题,写 ...

  9. c# 使用打印机打印并设置打印位置及宽高

    1.在界面中使用自带的控件printDocument 2.将以下函数绑定到控件的PrintPage事件 private void printDocument1_PrintPage(object sen ...

  10. 广义 SAM 学习笔记

    开 CF 开到了一道广义 SAM,决定来学一学. 发现网上确实充斥着各种各样的伪广义 SAM,也看到了前人反复修改假板子的过程,所以试着来整理一下这堆奇奇怪怪的问题. 当然本文的代码也不保证百分百正确 ...