题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=4542


给定一个由数字构成的字符串${S_{1,2,3,...,n}}$,一个正素数$P$,每次询问给定一对$l$,$r$求:

$${\sum_{l=1}^{n}\sum_{r=i}^{n}\left [ \sum _{i=l}^{r}S[i]*10^{r-i} \,\,\,\,MOD\,\,\,\,P=0 \right ]}$$


  即以位置$x$开头的后缀的数字$%P$之后的值为$val[x]$,如果一个数字对应区间${[l,r]}$它$%p$为$0$的充要条件为${val[l]=val[r-1]}$,然后套上莫队算法,离散化$val$数组,就变成了经典的查询一个区间相同数字的点对有多少个。

  注意:$p=2,5$的情况中并不满足以上性质,记一下前缀和然后特判即可。


 #include<iostream>
#include<cstdio>
#include<algorithm>
#include<vector>
#include<cstdlib>
#include<cmath>
#include<cstring>
using namespace std;
#define maxn 300100
#define llg long long
#define yyj(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout);
llg n,m,p,quan[maxn],tail,val[maxn],len,se[maxn],KUAI;
long long ans,Ans[maxn],jisuan1[maxn],jisuan2[maxn];
char s[maxn];
struct Q_
{
llg l,r,num;
}ask[maxn]; bool cmp(const Q_&a,const Q_&b) {if (a.l/KUAI==b.l/KUAI) return a.r<b.r; else return a.l/KUAI<b.l/KUAI; } llg find(llg x)
{
llg l=,r=len,wz=-,mid;
while (l<=r)
{
mid=(l+r)>>;
if (quan[mid]<=x) {wz=mid; l=mid+;}else {r=mid-;}
}
return wz;
} void solve()
{
cin>>m;
for(int i=;i<=n;i++)
{
jisuan1[i]=jisuan1[i-]; jisuan2[i]=jisuan2[i-];
if( (s[i]-'')%p== )
{
jisuan1[i]++;
jisuan2[i]+=i;
} }
for(int i=;i<=m;i++)
{
llg x,y;
scanf("%lld%lld",&x,&y);
printf("%lld\n",jisuan2[y]-jisuan2[x-] - (x-)*(jisuan1[y]-jisuan1[x-]));
}
} void init()
{
cin>>p;
scanf("%s",s+);
n=strlen(s+);
KUAI=sqrt(n)+;
if (p== || p==) return ;
cin>>m;
for (llg i=;i<=m;i++) scanf("%lld%lld",&ask[i].l,&ask[i].r),ask[i].r++,ask[i].num=i;
sort(ask+,ask+m+,cmp);
llg C=;
tail=;quan[]=;
for (llg i=n;i>=;i--)
{
val[i]=val[i+]+C*(s[i]-'');
val[i]%=p;
quan[++tail]=val[i];
C*=; C%=p;
}
sort(quan+,quan+tail+);
len=unique(quan+,quan+tail+)-(quan+);
for (llg i=;i<=n+;i++) val[i]=find(val[i]);
} int main()
{
yyj("number");
init();
if (p== || p==) {solve(); return ;} llg l=ask[].l,r=ask[].r;
for (llg i=l;i<=r;i++)
{
ans+=se[val[i]];
se[val[i]]++;
}
Ans[ask[].num]=ans;
for (llg k=;k<=m;k++)
{
llg nl=ask[k].l,nr=ask[k].r;
if (nr>r)
{
for (llg i=r+;i<=nr;i++)
{
ans+=se[val[i]];
se[val[i]]++;
}
}
if (nr<r)
{
for (llg i=r;i>nr;i--)
{
ans-=se[val[i]]-;
se[val[i]]--;
}
}
if (l<nl)
{
for (llg i=l;i<nl;i++)
{
ans-=se[val[i]]-;
se[val[i]]--;
}
}
if (l>nl)
{
for (llg i=l-;i>=nl;i--)
{
ans+=se[val[i]];
se[val[i]]++;
}
}
Ans[ask[k].num]=ans;
l=nl,r=nr;
}
for (llg i=;i<=m;i++) printf("%lld\n",Ans[i]);
return ;
}

【BZOJ】4542: [Hnoi2016]大数的更多相关文章

  1. BZOJ.4542.[HNOI2016]大数(莫队)

    题目链接 大数除法是很麻烦的,考虑能不能将其条件化简 一段区间[l,r]|p,即num[l,r]|p,类似前缀,记后缀suf[i]表示[i,n]的这段区间代表的数字 于是有 suf[l]-suf[r+ ...

  2. bzoj 4542 [Hnoi2016]大数 (坑)

    题面 https://www.lydsy.com/JudgeOnline/problem.php?id=4542 题解 Code #include<bits/stdc++.h> using ...

  3. bzoj 4542: [Hnoi2016]大数

    Description 小 B 有一个很大的数 S,长度达到了 N 位:这个数可以看成是一个串,它可能有前导 0,例如00009312345 小B还有一个素数P.现在,小 B 提出了 M 个询问,每个 ...

  4. bzoj 4542: [Hnoi2016]大数 (莫队)

    Description 小 B 有一个很大的数 S,长度达到了 N 位:这个数可以看成是一个串,它可能有前导 0,例如00009312345.小B还有一个素数P.现在,小 B 提出了 M 个询问,每个 ...

  5. 4542: [Hnoi2016]大数

    4542: [Hnoi2016]大数 链接 分析: 如果p等于2或者5,可以根据最后一位直接知道是不是p的倍数,所以直接记录一个前缀和即可. 如果p不是2或者5,那么一个区间是p的倍数,当且仅当$\f ...

  6. 4542: [Hnoi2016]大数

    Description 小 B 有一个很大的数 S,长度达到了 N 位:这个数可以看成是一个串,它可能有前导 0,例如00009312345.小B还有一个素数P.现在,小 B 提出了 M 个询问,每个 ...

  7. 【LG3245】[HNOI2016]大数

    [LG3245][HNOI2016]大数 题面 洛谷 题解 60pts 拿vector记一下对于以每个位置为右端点符合要求子串的左端点, 则每次对于一个询问,扫一遍右端点在vector里面二分即可, ...

  8. BZOJ 3110 K大数查询 | 整体二分

    BZOJ 3110 K大数查询 题面 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c 如果是2 a b c形式,表示询问从第a个 ...

  9. 【BZOJ4542】[Hnoi2016]大数 莫队

    [BZOJ4542][Hnoi2016]大数 Description 小 B 有一个很大的数 S,长度达到了 N 位:这个数可以看成是一个串,它可能有前导 0,例如00009312345.小B还有一个 ...

随机推荐

  1. JustOj 2009: P1016 (dp)

    题目描述 有一个箱子容量为v(正整数,o≤v≤20000),同时有n个物品(o≤n≤30),每个物品有一个体积  (正整数).要求从  n  个物品中,任取若干个装入箱内,使箱子的剩余空间为最小.  ...

  2. 发布webservice服务

    1,定义一个接口 public interface HaiService { //定义一个方法 String speak(String str); } 2,编写一个实现类 import javax.j ...

  3. CSM与UEFI

    最近公司产品部购置一批新电脑,但是预装的win10不能保证兼容老平台软件,于是安装win7系统的任务就落到了我的手中. 观察参数,是8代的U,产品说运维说无能为力,装不了win7.我在网上搜了一下,是 ...

  4. linux 函数库使用

    程序函数库可分为3种类型:静态函 数库(static libraries).共享函数库(shared libraries)和动态加载函数库(dynamically loaded libraries) ...

  5. sql xml 入门 (二)

    DECLARE @myDoc xml --http://www.paymob.cn --话费充值api,充值api,话费充值接口,手机话费充值,车贝手机,贝萌手机,移动话费充值,联通话费充值,电信话费 ...

  6. CEF 添加F5刷新快捷键

    Keyboardcodes:https://www.androidos.net.cn/android/4.3_r1/xref/external/webkit/Source/WebCore/platfo ...

  7. 0基础学安卓--初识安卓Activity

    知识储备:windows+ Android Studio 等环境安装. 安卓中Activity代表页的意思,也就是☞我们手机上当前的整个界面显示,点击按钮等操作可以跳转到另外一个Activity中. ...

  8. bzoj 2527: [Poi2011]Meteors

    昨天写了一晚,越写复杂度越感觉不对,早上一想果然是假的. (这里n,m,k我就不区分了) 首先一个城市的询问可以很容易的二分 check用树状数组维护区间(区间修改,单点查询的那种) 一次是\(O(n ...

  9. 前缀判断|2013年蓝桥杯B组题解析第五题-fishers

    前缀判断 如下的代码判断 needle_start指向的串是否为haystack_start指向的串的前缀,如不是,则返回NULL. 比如:"abcd1234" 就包含了 &quo ...

  10. Java中 System.arraycopy() 和 Arrays.copyOf()方法

    System.arraycopy() 和 Arrays.copyOf()方法 阅读源码的话,我们就会发现 ArrayList 中大量调用了这两个方法.比如:我们上面讲的扩容操作以及add(int in ...