题目链接:https://vjudge.net/problem/HDU-4507

题意:定义如果一个整数符合下面3个条件之一,那么我们就说这个整数和7有关——
  1、整数中某一位是7;

  2、整数的每一位加起来的和是7的整数倍;

  3、这个整数是7的整数倍;

  给定l,r,求[l,r] 区间与7无关的数的平方和。

思路:这3条定义都是常规的数位dp,但题目求的并不是与7无关的数的个数,而是平方和,这也是该题的难点。这里需要数位dp维护3个值:

  1. 与7无关的数的个数num;

  2. 与7无关的数的和sum;

  3. 与7无关的数的平方和sum;

  (用结构体组织上述3个属性,假设当前求的结点是ans,当前点取i,递归得到的结点为tmp)

  第1条很简单,就是常规的数位dp:

    ans.num+=tmp.num;
    ans.num%=Mod;

  第2条需要用到第一条,求当前的所有数的和,即后面位的所有和+当前值×后面的数的个数:

    ans.sum+=(tmp.sum+(i*key[pos])%Mod*tmp.num%Mod)%Mod;
    ans.sum%=Mod;

  第3条需要用到前2条,求当前所有数的平方和。先考虑一个数,设该数后面的位数的值为b,当前要加的值为a(a=i*key[pos]),则该数平方和为(a+b)^2=a^2+2*a*b+b^2,然后考虑对所有数的平方和,即上述式子求和tmp.num次,上述式子有3项。第一项:即a^2*tmp.num。第二项:即2*a*tmp.sum。第三项:即tmp.sqsum。

        ans.sqsum+=tmp.num*i%Mod*i%Mod*key[pos]%Mod*key[pos]%Mod;
ans.sqsum%=Mod;
ans.sqsum+=*i*key[pos]%Mod*tmp.sum%Mod;
ans.sqsum%=Mod;
ans.sqsum+=tmp.sqsum;
ans.sqsum%=Mod;

还有要注意的是这道题的数据,因为num,sum,sqsum还有key[i]都可能超过Mod,所以每乘一次就要%Mod,不然会出现乘法溢出。

AC代码:

#include <cstdio>
using namespace std;
typedef long long LL;
const LL Mod=; struct node{
LL num,sum,sqsum;
}dp[][][]; int T,a[];
LL key[]; node dfs(int pos,int pre1,int pre2,bool limit){
if(pos==-){
node tmp;
tmp.num=(pre1!=&&pre2!=);
tmp.sum=tmp.sqsum=;
return tmp;
}
if(!limit&&dp[pos][pre1][pre2].num!=-)
return dp[pos][pre1][pre2];
int up=limit?a[pos]:;
node ans;
ans.num=ans.sum=ans.sqsum=;
for(int i=;i<=up;++i){
if(i==) continue;
node tmp=dfs(pos-,(pre1+i)%,(pre2*+i)%,limit&&i==a[pos]); ans.num+=tmp.num;
ans.num%=Mod; ans.sum+=(tmp.sum+(i*key[pos])%Mod*tmp.num%Mod)%Mod;
ans.sum%=Mod; ans.sqsum+=tmp.num*i%Mod*i%Mod*key[pos]%Mod*key[pos]%Mod;
ans.sqsum%=Mod;
ans.sqsum+=*i*key[pos]%Mod*tmp.sum%Mod;
ans.sqsum%=Mod;
ans.sqsum+=tmp.sqsum;
ans.sqsum%=Mod;
}
if(!limit) dp[pos][pre1][pre2]=ans;
return ans;
} LL solve(LL x){
int pos=;
while(x){
a[pos++]=x%;
x/=;
}
return dfs(pos-,,,true).sqsum;
} int main()
{
key[]=;
for(int i=;i<=;++i)
key[i]=(key[i-]*)%Mod;
for(int i=;i<;++i)
for(int j=;j<;++j)
for(int k=;k<;++k)
dp[i][j][k].num=-;
scanf("%d",&T);
while(T--){
LL l,r,ans;
scanf("%lld%lld",&l,&r);
ans=solve(r)-solve(l-);
ans=(ans%Mod+Mod)%Mod;
printf("%lld\n",ans);
}
return ;
}

hdoj4507(数位dp)的更多相关文章

  1. 专题训练之数位DP

    推荐以下一篇博客:https://blog.csdn.net/wust_zzwh/article/details/52100392 1.(HDOJ2089)http://acm.hdu.edu.cn/ ...

  2. 【BZOJ1662】[Usaco2006 Nov]Round Numbers 圆环数 数位DP

    [BZOJ1662][Usaco2006 Nov]Round Numbers 圆环数 Description 正如你所知,奶牛们没有手指以至于不能玩"石头剪刀布"来任意地决定例如谁 ...

  3. bzoj1026数位dp

    基础的数位dp 但是ce了一发,(abs难道不是cmath里的吗?改成bits/stdc++.h就过了) #include <bits/stdc++.h> using namespace ...

  4. uva12063数位dp

    辣鸡军训毁我青春!!! 因为在军训,导致很长时间都只能看书yy题目,而不能溜到机房鏼题 于是在猫大的帮助下我发现这道习题是数位dp 然后想起之前讲dp的时候一直在补作业所以没怎么写,然后就试了试 果然 ...

  5. HDU2089 不要62[数位DP]

    不要62 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

  6. 数位DP GYM 100827 E Hill Number

    题目链接 题意:判断小于n的数字中,数位从高到低成上升再下降的趋势的数字的个数 分析:简单的数位DP,保存前一位的数字,注意临界点的处理,都是套路. #include <bits/stdc++. ...

  7. 数位dp总结

    由简单到稍微难点. 从网上搜了10到数位dp的题目,有几道还是很难想到的,前几道基本都是模板题,供入门用. 点开即可看题解. hdu3555 Bomb hdu3652 B-number hdu2089 ...

  8. 数位DP入门

    HDU 2089 不要62 DESC: 问l, r范围内的没有4和相邻62的数有多少个. #include <stdio.h> #include <string.h> #inc ...

  9. 数位DP之奥义

    恩是的没错数位DP的奥义就是一个简练的dfs模板 int dfs(int position, int condition, bool boundary) { ) return (condition ? ...

随机推荐

  1. JVM启动参数大全及默认值

    Java启动参数共分为三类: 其一是标准参数(-),所有的JVM实现都必须实现这些参数的功能,而且向后兼容: 其二是非标准参数(-X),默认jvm实现这些参数的功能,但是并不保证所有jvm实现都满足, ...

  2. linux运维、架构之路-K8s滚动更新及回滚

    一.滚动更新        应用程序一次只更新一小部分副本,更新成功后,再更新更多的副本,最终完成所有副本的更新. 滚动更新的优点:零停机,整个更新过程始终有副本在运行,从而保证了业务的连续性. 1. ...

  3. cdh-完整

    安装包 CLOUDERA管理安装包 http://archive.cloudera.com/cm5/cm/5/ http://archive.cloudera.com/cm5/cm/5/clouder ...

  4. javascript中创建对象的方式及优缺点(二)

    一.工厂模式 流程: 定义一个函数,函数返回对象. 适用场景: 需要创建多个对象,都是Object类型. 优点:完成了返回一个对象的要求. 缺点: 对象没有一个具体的类型,无法通过constructo ...

  5. 16位masm汇编实现筛法,状压求十万以内素数

    .model small .data table byte 3,12500 dup (0);;0和1不是质数 i word 0 j word 0 .stack 4096 .code main proc ...

  6. 使用svn在github上下载文件夹

    今天想在github上下载mybatis-generator的eclipse插件,可是如何在github上下载一个文件夹而不用把这个项目clone呢,搜了一下,发现可以直接用svn来下载 只需将将路径 ...

  7. js获取iframe里面的dom

    最近在写页面遇到了问题,一个dom好多地方用到,然后我就单独写了个html页面,然后用iframe引入,但是,想获取iframe里面input的value,获取不到input,后面才知道原来js不能直 ...

  8. javaScrpit插件学习制作

    最近一直在学习javaScrpit插件制作,前几天学习制作了一个插件但存在严重缺陷. javaScrpit插件写法多种多样通过这几天的学习终于找到了适合自己的方法.前几天的缺陷也得到了解决.下面我们用 ...

  9. Mac开发如何处理键盘事件

    Mac上输入与手机输入的不同是,Mac需要处理更多的键盘交互,因为Mac上的键盘输入会有多种快捷键组合. 代理方法处理 NSTextField #pragma mark - NSTextFieldDe ...

  10. html5 canvas简易时钟

    <canvas id='clock' width=500 height=500> 您的浏览器需要升级 </canvas> <script type="text/ ...