吉哥系列故事——恨7不成妻
Time Limit: 1000/500 MS (Java/Others) Memory Limit: 65535/32768 K (Java/Others)
Problem Description
单身!
依然单身!
吉哥依然单身!
DS级码农吉哥依然单身!
所以,他生平最恨情人节,不管是214还是77,他都讨厌!
吉哥观察了214和77这两个数,发现:
2+1+4=7
7+7=7*2
77=7*11
最终,他发现原来这一切归根到底都是因为和7有关!所以,他现在甚至讨厌一切和7有关的数!
什么样的数和7有关呢?
如果一个整数符合下面3个条件之一,那么我们就说这个整数和7有关——
1、整数中某一位是7;
2、整数的每一位加起来的和是7的整数倍;
3、这个整数是7的整数倍;
现在问题来了:吉哥想知道在一定区间内和7无关的数字的平方和。
依然单身!
吉哥依然单身!
DS级码农吉哥依然单身!
所以,他生平最恨情人节,不管是214还是77,他都讨厌!
吉哥观察了214和77这两个数,发现:
2+1+4=7
7+7=7*2
77=7*11
最终,他发现原来这一切归根到底都是因为和7有关!所以,他现在甚至讨厌一切和7有关的数!
什么样的数和7有关呢?
如果一个整数符合下面3个条件之一,那么我们就说这个整数和7有关——
1、整数中某一位是7;
2、整数的每一位加起来的和是7的整数倍;
3、这个整数是7的整数倍;
现在问题来了:吉哥想知道在一定区间内和7无关的数字的平方和。
Input
输入数据的第一行是case数T(1 <= T <= 50),然后接下来的T行表示T个case;每个case在一行内包含两个正整数L, R(1 <= L <= R <= 10^18)。
Output
请计算[L,R]中和7无关的数字的平方和,并将结果对10^9 + 7 求模后输出。
Sample Input
3
1 9
10 11
17 17
Sample Output
236
221
0
分析:这题和普通数位dp相比,要算平方和,而不是个数;
思考怎么算平方和,就是每次数在前面加了一位;
(a1+p)2+(a2+p)2+...+(an+p)2=(a12+a22+...an2)+n*p2+2*(a1+a2+...+an)*p;
所以维护三个东西,个数n,数的和,数的平方和;
注意预处理数取模和减法取模防止溢出;
代码:
#include <iostream> #include <cstdio> #include <cstdlib> #include <cmath> #include <algorithm> #include <climits> #include <cstring> #include <string> #include <set> #include <bitset> #include <map> #include <queue> #include <stack> #include <vector> #define rep(i,m,n) for(i=m;i<=n;i++) #define mod 1000000007 #define inf 0x3f3f3f3f #define vi vector<int> #define pb push_back #define mp make_pair #define fi first #define se second #define ll long long #define pi acos(-1.0) #define pii pair<int,int> #define piii pair<int,pair<ll,ll> > #define sys system("pause") const int maxn=1e5+10; const int N=5e4+10; const int M=N*10*10; using namespace std; inline ll gcd(ll p,ll q){return q==0?p:gcd(q,p%q);} inline ll qpow(ll p,ll q){ll f=1;while(q){if(q&1)f=f*p;p=p*p;q>>=1;}return f;} inline void umax(ll &p,ll q){if(p<q)p=q;} inline void umin(ll &p,ll q){if(p>q)p=q;} inline ll read() { ll x=0;int f=1;char ch=getchar(); while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();} return x*f; } int n,m,k,t,num[20],pos; piii dp[20][7][7]; bool vis[20][7][7]; ll l,r,p[20]; piii dfs(int pos,int x,int y,int z) { if(pos<0)return mp(x!=0&&y!=0,mp(0,0)); if(z&&vis[pos][x][y])return dp[pos][x][y]; int now=z?9:num[pos],i; piii ret=mp(0,mp(0,0)); rep(i,0,now) { if(i==7)continue; piii k=dfs(pos-1,(x+i)%7,(y*10+i)%7,z||i<num[pos]); ll u=p[pos]*i%mod; ret.fi+=k.fi; ret.fi%=mod; ret.se.fi+=k.se.fi+k.fi*u%mod; ret.se.fi%=mod; ret.se.se+=k.se.se+k.fi*u%mod*u%mod+2*k.se.fi*u%mod; ret.se.se%=mod; } if(z) { vis[pos][x][y]=true; dp[pos][x][y]=ret; } return ret; } ll gao(ll x) { pos=0; while(x)num[pos++]=x%10,x/=10; return dfs(pos-1,0,0,0).se.se; } int main() { int i,j; p[0]=1; rep(i,1,19)p[i]=p[i-1]*10%mod; scanf("%d",&t); while(t--) { scanf("%lld%lld",&l,&r); printf("%lld ",(gao(r)-gao(l-1)+mod)%mod); } return 0; }