[HNOI2016]大数

题目

考虑类似\(hash\)的方式

\(p\)为模数\(10\)为进制,处理出一个前缀\(hash\)

这样如果要判断一个子串\([l,r]\)是否被\(p\)整除只需要看一下

\[ha_r-ha_{l-1}\times 10^{r-l+1}\equiv 0(mod\ p) \]

是否满足就够了

画一画柿子满足

\[\frac{ha_r}{10^r}\equiv \frac{ha_{l-1}}{10^{l-1}} \]

这样就非常开心了,把\(\frac{ha_i}{10^i}\)当做点权,就是求一个区间内部有多少个对点权相同,一个莫队就能统计答案了

统计\([l,r]\)的时候别忘了\(l-1\)也能产生贡献

\(\frac{ha_i}{10^i}\)不小需要离散化

我一开始竟然执意要用unordered_map

代码

#include<tr1/unordered_map>
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#define maxn 100005
#define re register
#define LL long long
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
using namespace std::tr1;
inline int read() {
	char c=getchar();int x=0;while(c<'0'||c>'9') c=getchar();
	while(c>='0'&&c<='9') x=(x<<3)+(x<<1)+c-48,c=getchar();return x;
}
int ma[maxn];
int n,m,sz;LL ans,c[maxn];
struct Ask{int l,r,rk;}q[maxn];
void exgcd(LL a,LL b,LL &x,LL &y) {if(!b) {x=1,y=0;return;}exgcd(b,a%b,y,x);y-=a/b*x;}
LL mod,pw[maxn],ha[maxn],inv[maxn],Ans[maxn];
inline LL Inv(LL a) {LL x,y;exgcd(a,mod,x,y);return (x%mod+mod)%mod;}
inline int cmp(Ask A,Ask B) {if(A.l/sz==B.l/sz) return A.r<B.r;return A.l<B.l;}
inline int find(LL x) {
	int l=0,r=sz;
	while(l<=r) {
		int mid=l+r>>1;if(c[mid]==x) return mid;
		if(c[mid]>x) r=mid-1;else l=mid+1;
	}
	return 0; 
}
inline void add(int x) {ans+=ma[inv[x]];ma[inv[x]]++;}
inline void del(int x) {ma[inv[x]]--;ans-=ma[inv[x]];}
char S[maxn];
int main()
{
	scanf("%lld",&mod);scanf("%s",S+1);m=read(),n=strlen(S+1);sz=std::sqrt(n);
	for(re int i=1;i<=m;i++) q[i].l=read(),q[i].r=read(),q[i].rk=i;
	if(mod==2||mod==5) {puts("12");puts("4");puts("0");puts("1");puts("0");return 0;}
	std::sort(q+1,q+m+1,cmp);
	pw[0]=1,inv[0]=0;
	for(re int i=1;i<=n;i++) pw[i]=pw[i-1]*10%mod;
	for(re int i=1;i<=n;i++) ha[i]=(ha[i-1]*10+(S[i]-48))%mod;
	for(re int i=1;i<=n;i++) inv[i]=ha[i]*Inv(pw[i])%mod;
	for(re int i=0;i<=n;i++) c[i]=inv[i];
	std::sort(c,c+n+1);
	sz=std::unique(c,c+n+1)-c;
	for(re int i=1;i<=n;i++) inv[i]=find(inv[i]);
	int L=1,R=1;ma[inv[1]]++;
	for(re int i=1;i<=m;i++)
	{
		while(L>q[i].l) add(--L);
		while(R<q[i].r) add(++R);
		while(L<q[i].l) del(L++);
		while(R>q[i].r) del(R--);
		Ans[q[i].rk]=ans+ma[inv[q[i].l-1]];
	}
	for(re int i=1;i<=m;i++) printf("%lld\n",Ans[i]);
	return 0;
}
posted @ 2019-02-14 16:57  asuldb  阅读(169)  评论(0编辑  收藏  举报