【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,如果直接记录下数字, ...
随机推荐
- NET Core应用中使用缓存
NET Core应用中使用缓存 .NET Core针对缓存提供了很好的支持 ,我们不仅可以选择将数据缓存在应用进程自身的内存中,还可以采用分布式的形式将缓存数据存储在一个“中心数据库”中.对于分布式缓 ...
- java新知识系列 六
sleep和wait的区别有: Servlet方法的使用 方法重写的规则,以及两同两小一大原则: DispatcherServlet的解析 依赖注入DU和控制反转Ioc AOP和OOP的区别 Spri ...
- 07-Vue的基础使用
vue的介绍 前端框架和库的区别 nodejs的简单使用 vue的起步 指令系统 组件的使用 过滤器的使用 watch和computed 钩子函数 渐进式的JavaScript框架 vue react ...
- .NET CORE学习笔记系列(2)——依赖注入[8]: .NET Core DI框架[服务消费]
原文:https://www.cnblogs.com/artech/p/net-core-di-08.html 包含服务注册信息的IServiceCollection对象最终被用来创建作为DI容器的I ...
- audio
// media.cpp : 定义控制台应用程序的入口点. // https://wenku.baidu.com/view/e910c474c5da50e2524d7fb4.html https:// ...
- java.util.NoSuchElementException问题定位
Iterator 迭代器越界 例子如下: Iterator i = set.iterator(); while (i.hasNext()) { System.out.println(i.next()) ...
- $.extend()浅拷贝深拷贝
参考网址:http://bijian1013.iteye.com/blog/2255037 jQuery.extend() 函数用于将一个或多个对象的内容合并到目标对象. 注意:1. 如果只为$.ex ...
- 从开始到头皮炸裂的python第5天
头皮炸裂的一天从学到一个新的数据类型开始,这个数据类型的新成员叫做字典,基本的格式为data={键:值,键:值},info.keys()表示所有的键,info.values()表示所有的值,info. ...
- SQL NOT NULL 约束
SQL NOT NULL 约束 NOT NULL 约束强制列不接受 NULL 值. NOT NULL 约束强制字段始终包含值.这意味着,如果不向字段添加值,就无法插入新记录或者更新记录. 下面的 SQ ...
- Python爬虫-爬取豆瓣电影Top250
#!usr/bin/env python3 # -*- coding:utf-8-*- import requests from bs4 import BeautifulSoup import re ...