D-【乐】k进制数(同余)

题目

https://ac.nowcoder.com/acm/contest/907/D

做法

\((x)_k\)定义编号,如果\(a+b\)加到一起能进一位,\(a+b\rightarrow 1+(a+b-k)=a+b-(k-1)\),故\(d(a_{l,r})=\sum\limits_{i=l}^r a_i\% k-1\)

但我们发现\(k-1\)这一块缺失了,显然为\(0\)当且仅当区间均为\(0\),其他情况得出\(0\)的时候实际结果为\(k-1\)

  • \(b=0\):全\(0\)区间个数

  • \(b=k-1\):满足\(/%(k-1)=0\)的个数-全\(0\)区间个数

  • 其他情况:\(a_{l,r}=sum_r-sum_{l-1}\%(k-1),sum_r-sum_{l-1}\equiv b (\%k-1),sum_r-b\equiv sum_{l-1}(\%k-1)\)

    Code

#include<bits/stdc++.h>
typedef long long LL;
const LL maxn=1e6+9;
inline LL Read(){
    LL x(0),f(1); char c=getchar();
    while(c<'0' || c>'9'){
        if(c=='-') f=-1; c=getchar();
    }
    while(c>='0' && c<='9'){
        x=(x<<3)+(x<<1)+c-'0'; c=getchar();
    }return x*f;
}
LL k,b,n,ret,num,ze;
LL a[maxn],sum[maxn];
std::map<LL,LL> cnt;
int main(){
    k=Read(); b=Read(); n=Read();
    for(LL i=1;i<=n;++i) a[i]=Read();
    for(LL i=1;i<=n;++i){
        sum[i]=(sum[i-1]+a[i])%(k-1);
        if(!a[i]){
            ++num;
            ze+=num;
        }else
            num=0;
    }
    if(!b){
        printf("%lld\n",ze);
        return 0;
    }
    cnt[0]++;
    for(LL i=1;i<=n;++i){
        LL val((sum[i]-b+k-1)%(k-1));
        ret+=cnt[val];
        ++cnt[sum[i]];
    }
    if(b==k-1) ret-=ze;
    printf("%lld\n",ret);
    return 0;
}

转载于:https://www.cnblogs.com/y2823774827y/p/10957899.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值