【XSY3139】预言家 数位DP NFA
题目描述
有一个定义在 \(\{0,1,2,3,4,5,6,7,8,9\}\) 上的合规表达式,包含三种基本的操作:
结合:\(E_1E_2\)
分配:\((E_1|E_2|\ldots|E_n),n\geq 2\)
重复:\((E_1)* ,n\geq 0\)
给你 \(l,r\),问你有多少个 \([l,r]\) 之间不含前导零的整数能匹配这个合规表达式。
\(1\leq l\leq r\leq {10}^{18}\)
题解
直接建出这个合规表达式对应的 NFA,在上面跑数位 DP 即可。
记录 \(f_{i,0/1,j}\) 表示还需要确定后 \(i\) 位,前面这几位是否比 \(n\) 小,在 NFA 上面可以达到的状态集合是 \(j\) 时的方案数。
时间复杂度:\(O(???)\)
代码
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<map>
using namespace std;
typedef long long ll;
char s[100];
int c[100];//另一个括号的位置
int n;
int st[100];
int top;
int b[100][100];//epsilon
int nxt[100];
int move(int x,int v)
{
int res=0;
for(int i=1;i<=n+1;i++)
if(((x>>(i-1))&1)&&s[i]-'0'==v)
res|=nxt[i+1];
return res;
}
void init()
{
memset(c,0,sizeof c);
top=0;
for(int i=1;i<=n;i++)
if(s[i]=='(')
st[++top]=i;
else if(s[i]==')')
{
c[i]=st[top];
c[st[top]]=i;
top--;
}
memset(b,0,sizeof b);
for(int i=1;i<=n+1;i++)
b[i][i]=1;
for(int i=1;i<=n;i++)
if(s[i]=='(')
{
if(s[c[i]+1]=='*')
{
b[i][c[i]]=1;
b[i][i+1]=1;
}
else
{
b[i][i+1]=1;
int cnt=0;
for(int j=i;j<=c[i];j++)
{
if(s[j]=='(')
cnt++;
else if(s[j]==')')
cnt--;
if(s[j]=='|'&&cnt==1)
{
b[i][j+1]=1;
b[j][c[i]]=1;
}
}
}
}
else if(s[i]==')')
{
b[i][i+1]=1;
if(s[i+1]=='*')
b[i][c[i]]=1;
}
else if(s[i]=='*')
b[i][i+1]=1;
for(int k=1;k<=n+1;k++)
for(int i=1;i<=n+1;i++)
for(int j=1;j<=n+1;j++)
b[i][j]|=b[i][k]&&b[k][j];
memset(nxt,0,sizeof nxt);
for(int i=1;i<=n+1;i++)
for(int j=1;j<=n+1;j++)
if(b[i][j])
nxt[i]|=1<<(j-1);
}
int len;
int a[100];
map<int,ll> f[100][2];
ll calc(ll m)
{
if(!m)
return 0;
len=0;
while(m)
{
a[++len]=m%10;
m/=10;
}
for(int i=0;i<=len;i++)
{
f[i][0].clear();
f[i][1].clear();
}
for(int i=1;i<=a[len];i++)
f[len-1][i==a[len]][move(nxt[1],i)]++;
for(int i=len-1;i>=1;i--)
for(int j=1;j<=9;j++)
f[i-1][0][move(nxt[1],j)]++;
for(int i=len;i>=1;i--)
{
for(auto v:f[i][1])
if(v.first)
{
for(int j=0;j<=a[i];j++)
f[i-1][j==a[i]][move(v.first,j)]+=v.second;
}
for(auto v:f[i][0])
if(v.first)
{
for(int j=0;j<=9;j++)
f[i-1][0][move(v.first,j)]+=v.second;
}
}
ll res=0;
for(int i=0;i<=1;i++)
for(auto v:f[0][i])
if((v.first>>n)&1)
res+=v.second;
return res;
}
void solve()
{
ll l,r;
scanf("%lld%lld",&l,&r);
scanf("%s",s+1);
n=strlen(s+1);
init();
ll ans1=calc(r);
ll ans2=calc(l-1);
ll ans=ans1-ans2;
printf("%lld\n",ans);
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("a.in","r",stdin);
freopen("a.out","w",stdout);
#endif
int t;
scanf("%d",&t);
while(t--)
solve();
return 0;
}
【XSY3139】预言家 数位DP NFA的更多相关文章
- 【BZOJ1662】[Usaco2006 Nov]Round Numbers 圆环数 数位DP
[BZOJ1662][Usaco2006 Nov]Round Numbers 圆环数 Description 正如你所知,奶牛们没有手指以至于不能玩"石头剪刀布"来任意地决定例如谁 ...
- bzoj1026数位dp
基础的数位dp 但是ce了一发,(abs难道不是cmath里的吗?改成bits/stdc++.h就过了) #include <bits/stdc++.h> using namespace ...
- uva12063数位dp
辣鸡军训毁我青春!!! 因为在军训,导致很长时间都只能看书yy题目,而不能溜到机房鏼题 于是在猫大的帮助下我发现这道习题是数位dp 然后想起之前讲dp的时候一直在补作业所以没怎么写,然后就试了试 果然 ...
- HDU2089 不要62[数位DP]
不要62 Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submis ...
- 数位DP GYM 100827 E Hill Number
题目链接 题意:判断小于n的数字中,数位从高到低成上升再下降的趋势的数字的个数 分析:简单的数位DP,保存前一位的数字,注意临界点的处理,都是套路. #include <bits/stdc++. ...
- 数位dp总结
由简单到稍微难点. 从网上搜了10到数位dp的题目,有几道还是很难想到的,前几道基本都是模板题,供入门用. 点开即可看题解. hdu3555 Bomb hdu3652 B-number hdu2089 ...
- 数位DP入门
HDU 2089 不要62 DESC: 问l, r范围内的没有4和相邻62的数有多少个. #include <stdio.h> #include <string.h> #inc ...
- 数位DP之奥义
恩是的没错数位DP的奥义就是一个简练的dfs模板 int dfs(int position, int condition, bool boundary) { ) return (condition ? ...
- 浅谈数位DP
在了解数位dp之前,先来看一个问题: 例1.求a~b中不包含49的数的个数. 0 < a.b < 2*10^9 注意到n的数据范围非常大,暴力求解是不可能的,考虑dp,如果直接记录下数字, ...
随机推荐
- Dynamics CRM项目实例之八:CRM 2015的产品系列,克隆,修订
关注本人微信和易信公众号: 微软动态CRM专家罗勇,回复139或者20150106可方便获取本文,同时可以在第一时间得到我发布的最新的博文信息,follow me! 今天的博客主要是关于D ...
- SAP MM 可以不用创建盘点凭证直接录入盘点结果?
SAP MM 可以不用创建盘点凭证直接录入盘点结果? 可以.SAP标准功能就支持这么做. 事务代码 MI09 (Enter Count w/o Reference to Document), 输入pl ...
- Xamarin 学习笔记 - Layout(布局)
本文翻译自CodeProject文章:https://www.codeproject.com/Articles/1227733/Xamarin-Notes-Xamarin-Forms-Layouts ...
- 使用ip开头的工具,而不是只会ifconfig
结论: 1.使用ip 开头的工具,比ifconfig显示的信息更多,并且支持的功能更强大. 2.常用的功能有: 显示接口基本信息: ip link show dev eth0 设置端口up/down: ...
- Lua在Redis中的应用
转载至笑松小站http://blog.seoui.com/2018/01/27/redis-lua/ redis从2.6版本开始内置支持Lua解释器,解释器提供了3个函数来处理redis的命令redi ...
- 关于Python, ftplib模块中的cwd()进入含中文目录失败的问题
使用Python的ftplib模块连接ftp服务器时, 使用cwd()连接含中文的目录, 报错 : UnicodeEncodeError: 'latin-1' codec can't encode c ...
- 数据库H2学习
本文转载自:https://www.cnblogs.com/xdp-gacl/p/4171024.html 一.H2数据库介绍 常用的开源数据库有:H2,Derby,HSQLDB,MySQL,Post ...
- CADisplayLink以及定时器的使用
第一种: 用CADisplayLink可以实现不停重绘. - (CADisplayLink *)link { if (!_link) { // 创建定时器,一秒钟调用rotation方法60次 _li ...
- 【转载】xilinx 高速收发器Serdes深入研究
此篇文章深入浅出介绍了关于高速串行收发器的几个重要概念和注意事项,为方便知识点复习总结和后续查阅特此转载,原文标题及链接为:xilinx 高速收发器Serdes深入研究 - CSDN博客 http ...
- man -f/-k [keyword]在fedora 29 中报错nothing appropriate
我们在使用 man 手册的时候,可以使用man -f [keyword]去查询keyword的在线文档,但是这时候会报错:(图来源自网络) 这是因为我们还没有建立 man 手册的索引缓存: 我们可以使 ...