题目链接: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. Ubuntu镜像的MD5校验

    造冰箱的大熊猫@cnblogs 2018/9/7 1.在Ubuntu终端中,按照以下格式输入命令计算镜像文件ubuntu-xxx.iso的MD5校验和. md5sum ubuntu-xxx.iso 2 ...

  2. python3.6+Xadmin2.0系列(一) xadmin下载及安装

    环境配置:win7+python3.6+Django2.1+xadmin2+PyCharm 一.Xadmin下载及安装: 1.下载: 下载地址:https://github.com/sshwsfc/x ...

  3. BZOJ 1920 Luogu P4217 [CTSC2010]产品销售 (模拟费用流、线段树)

    题目链接 (bzoj) https://www.lydsy.com/JudgeOnline/problem.php?id=1920 (luogu) https://www.luogu.org/prob ...

  4. 用过消息队列?Kafka?能否手写一个消息队列?懵

    是否有同样的经历?面试官问你做过啥项目,我一顿胡侃,项目利用到了消息队列,kafka,rocketMQ等等. 好的,那请开始你的表演,面试官递过一支笔:给我手写一个消息队列!!WHAT? 为了大家遇到 ...

  5. 作业要求20191010-9 alpha week 1/2 Scrum立会报告+燃尽图 07

    此作业要求参见:https://edu.cnblogs.com/campus/nenu/2019fall/homework/8752 一.小组情况组长:贺敬文组员:彭思雨 王志文 位军营 杨萍队名:胜 ...

  6. java 中的多态

    多态是面向对象的重要特性,简单点说:“一个接口,多种实现”,就是同一种事物表现出的多种形态. 下面利用代码进行简单概述: 实现多态必须要满足三个条件 1必须有继承 2必须重写 3必须父类引用子类对象 ...

  7. Point-wise Mutual Information

    Point-wise Mutual Information (Yao, et al 2019) reclaimed a clear description of Point-wise Mutual I ...

  8. win下gvim或者linux下的vim安装vundle都适用的配置文件 - 在当前目录及其子目录下**, 的所有文件* 中, 搜索当前光标所在的单词

    gvim下的普通配置: if v:lang =~ "utf8$" || v:lang =~ "UTF-8$" set fileencodings=utf-8,g ...

  9. 监控部署nagios+snmp

    参看是否有安装:rpm -q gcc glibc glibc-common gd gd-devel xinetd openssl-devel 未安装基础支持套件的先安装: yum install -y ...

  10. centos7.7下docker与k8s安装(DevOps三)

    1.系统配置 centos7.7 docker 1.13.1 centos7下安装docker:https://www.cnblogs.com/pu20065226/p/10536744.html 2 ...