zoukankan      html  css  js  c++  java
  • 吉哥系列故事——恨7不成妻

    吉哥系列故事——恨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无关的数字的平方和。
     
    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;
    }
  • 相关阅读:
    20191318实验四 《Python程序设计》实验报告
    20191318实验三 Socket编程技术
    20191318实验二 Python程序设计入门
    实验一 Python程序设计入门
    《信息安全专业导论》第十二周学习总结
    《信息安全专业导论》第十一周学习总结
    markdown画思维导图
    markdown页面内跳转
    20191206 2019-2020-2 《Python程序设计》实验四报告
    20191206 实验三《Python程序设计》实验报告
  • 原文地址:https://www.cnblogs.com/dyzll/p/6395461.html
Copyright © 2011-2022 走看看