SP10606题解
本文同步更新于洛谷博客
题目描述
一个数被称为是平衡的数,当且仅当对于所有出现过的数位,每个偶数出现奇数次,每个奇数出现偶数次。给定 请统计出 内所有平衡数的个数。
题解
平衡数与数的大小无关,并且我们要统计一个区间内符合条件的数的个数,不难想到用数位 dp。又因为我们要统计数字出现的次数,所以需要状态压缩。
我们用 来表示数字出现的次数, 表示出现偶数次, 表示出现奇数次。现在问题来了, 也可以表示该数字没出现过,因此需要再用一个变量 表示有没有出现过, 表示没出现过, 表示出现过。根据定义不难知道,转移的时候, 用的是异或, 用的是或。
接下来,我们传五个参数 进入 dfs,分别表示枚举到第 位,当前每个数字出现次数的状态,当前每个数字是否出现的状态,当前位是否为前导 ,以及这一位填的数有没有限制,用 数组记忆化即可。
注意到本题的内存限制是 ,所以我们可以直接开 的数组,没有必要用其他题解说的 map,当然想用也行。
Code
#include<bits/stdc++.h>
using namespace std;
#define ll long long
int t,len,a[20];
ll l,r,f[20][1024][1024];
ll check(int x,int y)
{
for(int i=0;i<=9;i++)
{
if(y&(1<<i))
{
if((i&1)^(x&(1<<i))==0)
return 0;
}
}
return 1;
}
ll dfs(int k,int x,int y,int p,int q)
{
if(!k)
return check(x,y);
if(!p&&!q&&f[k][x][y]!=-1)
return f[k][x][y];
int z=q?a[k]:9,w;
ll res=0;
for(int i=0;i<=z;i++)
{
w=p&&!i;
res+=dfs(k-1,w?0:x^(1<<i),w?0:y|(1<<i),w,q&&(i==z));
}
if(!p&&!q)
f[k][x][y]=res;
return res;
}
ll divide(ll x)
{
len=0;
while(x)
{
a[++len]=x%10;
x/=10;
}
return dfs(len,0,0,1,1);
}
int main()
{
memset(f,-1,sizeof(f));
scanf("%d",&t);
while(t--)
{
scanf("%lld%lld",&l,&r);
printf("%lld\n",divide(r)-divide(l-1));
}
return 0;
}
本文作者:Ginger_he
本文链接:https://www.cnblogs.com/Gingerhe/p/15845827.html
版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步