首页 > 其他分享 >P6357 题解

P6357 题解

时间:2022-12-27 13:34:37浏览次数:65  
标签:10 cnt int 题解 sum P6357 rc include

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

标签:10,cnt,int,题解,sum,P6357,rc,include
From: https://www.cnblogs.com/LcyRegister/p/17007886.html

相关文章

  • answerOpenCV轮廓类问题解析
    contour在opencv中是一个基础的数据结构,灵活运用的话,作用很大。以contour为关键字,在answerOpenCV中能够发现很多有趣的东西。 1、无法解决的问题​​......
  • 【题解】ABC283
    \(\text{AtCoderBeginnerContest283}\)APower无意义题,直接输出。BFirstQueryProblem无意义题,维护一个支持单点修改、单点查询的数据结构。(雾)CCashRegister......
  • Atcoder Beginner Contest ABC 283 Ex Popcount Sum 题解 (类欧几里得算法)
    题目链接令\(p=\lfloor\frac{n-r}m\rfloor\),则我们其实是要对所有\(0\lei\le29\)求\(\sum_{j=0}^p(\lfloor\frac{mj+r}{2^i}\rfloormod\2)\)。右边那个东西如果没......
  • 2022 International Collegiate Programming Contest, Jinan Site 部分题目简要题解
    从这里开始比赛目录傻逼学院负责人ProblemBTorch注意到$a_1,b_1,a_2,b_2$的和不会超过$10^6$考虑胖先生的周期开始的时候,瘦先生的周期在时......
  • USACO22DEC青铜组题解
    T1:CowCollege总学费\(=\)设置的单人学费\(\times\)接受的奶牛数一旦固定单人学费,就能确定接受的奶牛数单人学费可以是哪些值?\(\{c_1,c_2,\cdots,c_n\}\)其中......
  • P3537 [POI2012]SZA-Cloakroom 题解
    题目大意有\(n\)件物品,每件物品有三个属性\(a_i,b_i,c_i(a_i<b_i)\)。再给出\(q\)个询问,每个询问由非负整数\(m,k,s\)组成,问是否能够选出某些物品使得:对......
  • P4928 [MtOI2018]衣服?身外之物! 题解
    题意gcd共有\(n\)件衣服,编号为\(A_1,A_2,\cdotsA_n\)。每一件衣服分别拥有颜色值和清洗时间,他在每一件衣服穿完以后都会将其送去清洗,而这件衣服当天所拥有的舒适感......
  • NSIS编辑时的乱码问题解决方法
    在Windows中文系统中,HMNISEdit下使用非中文和英文,比如韩文、日语或者阿拉伯语等。会发现编辑的文字变成乱码或者问号。     1、在安装的过程中显示乱码。2、......
  • AcWing291.蒙德里安的梦想题解
    题解:蒙德里安的梦想注:本题解内容简陋,多有不周,敬请谅解。如果有问题请在评论区留言。谢谢。由于作者能力有限,这篇题解不会给出太严谨的证明,只是旨在帮助大家更好地理解此......
  • AcWing83场周赛题解
    第一题、奇偶题目链接:https://www.acwing.com/activity/content/problem/content/7862/比较麻烦(本人做法)找出不同字符个数,再判断。#include<iostream>usingnamespac......