【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,如果直接记录下数字, ...
随机推荐
- 【代码笔记】Web-CSS-CSS样式列表(url)
一,效果图. 二,代码. <!DOCTYPE html> <html> <head> <meta charset="utf-8"> ...
- idea使用事项
1.调用外部jar包 Run/Debug Configurations --- Tomcat Server --- Startup/connection --- Debug 添加“path”变量:** ...
- Centos7 系统下搭建.NET Core2.0+Nginx+Supervisor+Mysql环境
好记性不如烂笔头! 一.简介 一直以来,微软只对自家平台提供.NET支持,这样等于让这个“理论上”可以跨平台的框架在Linux和macOS上的支持只能由第三方项目提供(比如Mono .NET).直到微 ...
- Android 轮播图Banner切换图片的效果
Android XBanner使用详解 2018年03月14日 08:19:59 AND_Devil 阅读数:910 版权声明:本文为博主原创文章,未经博主允许不得转载. https://www. ...
- 微信小程序 从含有tabbar的页面跳转到不含有tabbar的页面
如何离开含有tabbar的页面 在微信小程序开发过程中,我们会碰到从某页跳转到一个含有tabbar的页面的需求, 用 wx.navigateTo({url: '...',}) 不起作用,需要使用 w ...
- HttpWebRequest 改为 HttpClient 踩坑记-请求头设置
HttpWebRequest 改为 HttpClient 踩坑记-请求头设置 Intro 这两天改了一个项目,原来的项目是.net framework 项目,里面处理 HTTP 请求使用的是 WebR ...
- SQL Server 数据库部分常用语句小结(一)
1. 查询某存储过程的访问情况 SELECT TOP 1000 db_name(d.database_id) as DBName, s.name as 存储名字, s.type_desc as 存储类 ...
- C# Debug和release判断用法
C# Debug和release判断用法 #if (!DEBUG) Response.Write("DEBUG下运行");#else Response.Write("re ...
- netstat Recv-Q和Send-Q
通过netstat -anp可以查看机器的当前连接状态: Active Internet connections (servers and established) Proto Recv-Q Se ...
- JAVA常用API的总结(2)
这篇是常用API的结束了,写完的话可以继续往后复习了. 1.基本类型包装类的介绍与相关代码实现 对于数据包装类的特点就是可以将基本数据类型与字符串来回切换,接下来我会通过介绍Integer类的形式,来 ...