【Loj #10168. 「一本通 5.3 练习 3」恨 7 不成妻】题解
题目链接
题目
单身!
依然单身!
吉哥依然单身!
DS 级码农吉哥依然单身!
所以,他平生最恨情人节,不管是 还是 ,他都讨厌!
吉哥观察了 和 这两个数,发现:
最终,他发现原来这一切归根到底都是因为和 有关!所以,他现在甚至讨厌一切和 有关的数!
什么样的数和 有关呢?如果一个整数符合下面三个条件之一,那么我们就说这个整数和 有关:
整数中某一位是 ;
整数的每一位加起来的和是 的整数倍;
这个整数是 的整数倍。
现在问题来了:吉哥想知道在一定区间内和 无关的数字的平方和。
思路
抛开平方和来说,这就是一道普通的数位dp。
但这题有平方和啊!
首先,假设当前这个数为 , 为我们数位dp枚举的首位, 为后面的数,假设有 位,然后让我们求在 确定的情况下,所有 的可能值下 的平方和,记为
我们先用数位dp求一个 ,表示 的方案数,于是上面的问题可以转化为:
然后可以愉快地开始化简。
首先先把 变成一个数学式子:
后面有平方和公式拆开:
把 套进去:
弄好看点:
显然 ,因为这就是 的定义:
这个式子一看,还有 未知,怎么办?那我们就设 。
于是式子就变成了这样:
然后我们来推一推 吧。
拆一下 :
把 乘进去:
然后我们发现, 不就是 吗?然后套进去:
然后这道题就做成来了。
总结
这道题真得是一道不错的题。
很好的数位dp+推式子。
这道题推式子的过程有点烦,但我也明白,做数位dp的过程本质上就是个递归,每一步考虑一下自己本身,答案就是对的。
Code
// Problem: 1586:【 例 2】数字游戏
// Contest: SSOIER
// URL: http://ybt.ssoier.cn:8088/problem_show.php?pid=1586
// Memory Limit: 524 MB
// Time Limit: 1000 ms
//
// Powered by CP Editor (https://cpeditor.org)
#include<bits/stdc++.h>
using namespace std;
#define int long long
inline int read(){int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;
ch=getchar();}while(ch>='0'&&ch<='9'){x=(x<<1)+
(x<<3)+(ch^48);ch=getchar();}return x*f;}
//#define M
#define mo 1000000007
//#define N
int n, m, i, j, k;
int x, y, a[30], ten[10010];
int f[30][10][10][2];
int s[30][10][10][2];
int t[30][10][10][2];
int dfs(int n, int sum, int x, int p)
{
if(n==0) return f[n][sum][x][p]=((sum%7>0 && x%7>0) ? 1 : 0);
int i, k=0, F, S, T;
if(f[n][sum][x][p]!=-1) return f[n][sum][x][p];
for(i=0; i<=9; ++i)
{
if(!p && i>a[n]) break;
if(i==7) continue;
k+=dfs(n-1, (sum+i)%7, (x*10+i)%7, p|(i<a[n])); k%=mo;
F=f[n-1][(sum+i)%7][(x*10+i)%7][p|(i<a[n])];
S=s[n-1][(sum+i)%7][(x*10+i)%7][p|(i<a[n])];
T=t[n-1][(sum+i)%7][(x*10+i)%7][p|(i<a[n])];
t[n][sum][x][p]+=(ten[n-1]*i%mo*F%mo+T)%mo;
// printf("t[%lld][%lld][%lld][%lld]+=ten[%lld]*%lld*%lld+%lld\n", n, sum, x, p, n-1, i, F, T);
s[n][sum][x][p]+=(ten[n-1]*ten[n-1]%mo*i%mo*i%mo*F%mo+ten[n-1]*i%mo*T%mo*2%mo+S)%mo;
// printf("%lld %lld %lld %lld\n", n, S, T, s[n][sum][x][p]);
t[n][sum][x][p]%=mo; s[n][sum][x][p]%=mo;
}
return f[n][sum][x][p]=k;
}
int calc(int x)
{
n=0;
memset(f, -1, sizeof(f));
memset(s, 0, sizeof(s));
memset(t, 0, sizeof(t));
while(x) a[++n]=x%10, x/=10;
dfs(n, 0, 0, 0);
// printf("> %lld\n", s[n][0][0][0]);
return s[n][0][0][0];
}
signed main()
{
// freopen("tiaoshi.in", "r", stdin);
// freopen("tiaoshi.out", "w", stdout);
for(i=ten[0]=1; i<=10000; ++i) ten[i]=ten[i-1]*10%mo;
m=read();
while(m--) x=read(), y=read(), printf("%lld\n", ((calc(y)-calc(x-1))%mo+mo)%mo);
return 0;
}
本文来自博客园,作者:zhangtingxi,转载请注明原文链接:https://www.cnblogs.com/zhangtingxi/p/15793528.html
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 微软正式发布.NET 10 Preview 1:开启下一代开发框架新篇章
· 没有源码,如何修改代码逻辑?
· PowerShell开发游戏 · 打蜜蜂
· 在鹅厂做java开发是什么体验
· WPF到Web的无缝过渡:英雄联盟客户端的OpenSilver迁移实战