2019.02.15 codechef Favourite Numbers(二分+数位dp+ac自动机)
传送门
题意:
给444个整数L,R,K,nL,R,K,nL,R,K,n,和nnn个数字串,L,R,K,数字串大小≤1e18,n≤65L,R,K,数字串大小\le1e18,n\le65L,R,K,数字串大小≤1e18,n≤65
问[L,R][L,R][L,R]中第KKK小的拥有nnn个数字串中至少一个串作为子串的数。
思路:
一看就要二分答案,现在考虑统计[L,R][L,R][L,R]中有多少个满足题意的数。
不妨考虑数位dpdpdp,然后发现没法很好的转移,为了优化转移可以对于所有的数字串构建一个acacac自动机来优化转移。
代码:
#include<bits/stdc++.h>
#define ri register int
using namespace std;
const int N=2005;
typedef long long ll;
namespace acam{
int ch[N][10],fail[N],tot=0,q[N],hd,tl;
bool exi[N];
#define idx(x) ((x)^48)
inline void init(){memset(ch[0],0,sizeof(ch[0])),memset(fail,0,sizeof(fail)),exi[0]=0,tot=0;}
inline int build(){return exi[++tot]=0,memset(ch[tot],0,sizeof(ch[tot])),tot;}
inline void insert(char s[]){
int p=0,n=strlen(s);
for(ri x,i=0;i<n;++i){
if(!ch[p][(x=idx(s[i]))])ch[p][x]=build();
p=ch[p][x];
}
exi[p]=1;
}
inline void getfail(){
hd=1,tl=0;
for(ri i=0;i<10;++i)if(ch[0][i])q[++tl]=ch[0][i];
while(hd<=tl){
int p=q[hd++];
for(ri i=0,v;i<10;++i){
if((v=ch[p][i]))fail[v]=ch[fail[p]][i],q[++tl]=v,exi[v]|=exi[fail[v]];
else ch[p][i]=ch[fail[p]][i];
}
}
}
#undef idx
}
namespace dDP{
ll f[20][N][2];
vector<int>Up,Down;
inline vector<int>init(ll x){
vector<int>ret;
ret.clear();
if(!x)return ret.push_back(x),ret;
while(x)ret.push_back(x-x/10*10),x/=10;
return ret;
}
inline ll dfs(int pos,int sta,bool flag,bool down,bool up,bool zero){
if(pos==-1)return flag;
if(!up&&!down&&!zero&&~f[pos][sta][flag])return f[pos][sta][flag];
ll ret=0;
for(ri l=down?Down[pos]:0,r=up?Up[pos]:9,i=l;i<=r;++i){
if(zero&&!i&&pos)ret+=dfs(pos-1,0,flag,down&&i==l,up&&i==r,1);
else ret+=dfs(pos-1,acam::ch[sta][i],flag|acam::exi[acam::ch[sta][i]],down&&i==l,up&&i==r,zero&&!i);
}
if(!up&&!down&&!zero)f[pos][sta][flag]=ret;
return ret;
}
inline ll solve(ll l,ll r){
int len1,len2;
Down=init(l),len1=Down.size();
Up=init(r),len2=Up.size();
for(ri i=len1+1;i<=len2;++i)Down.push_back(0);
return dfs(Up.size()-1,0,0,1,1,1);
}
}
char s[100];
int main(){
ll a,b,K,l,r,ans;
int n;
cin>>a>>b>>K>>n;
acam::init();
while(n--)scanf("%s",s),acam::insert(s);
acam::getfail();
l=a,r=b+1,ans=b+1;
memset(dDP::f,-1,sizeof(dDP::f));
while(l<=r){
ll mid=l+r>>1;
if(dDP::solve(a,mid)<K)l=mid+1;
else r=mid-1,ans=mid;
}
if(ans==b+1)puts("no such number");
else cout<<ans;
return 0;
}
2019.02.15 codechef Favourite Numbers(二分+数位dp+ac自动机)的更多相关文章
- 咕咕(数位dp+AC自动机)
咕咕(数位dp+AC自动机) 若一个字符串的字符集合是0~m-1,那么称它为m进制字符串.给出n个m进制字符串\(s_i\),每个字符串的权值为\(v_i\).对于另一个m进制字符串\(S\),设\( ...
- CF 434C Tachibana Kanade's Tofu[数位dp+AC自动机]
Solution //本代码压掉后两维 #include<cstdio> #define max(a,b) (a<b?b:a) using namespace std; inline ...
- [Sdoi2014]数数[数位dp+AC自动机]
3530: [Sdoi2014]数数 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 834 Solved: 434[Submit][Status][ ...
- shuoj 1 + 2 = 3? (二分+数位dp)
题目传送门 1 + 2 = 3? 发布时间: 2018年4月15日 22:46 最后更新: 2018年4月15日 23:25 时间限制: 1000ms 内存限制: 128M 描述 埃森哲是 ...
- POJ3208 Apocalypse Someday(二分 数位DP)
数位DP加二分 //数位dp,dfs记忆化搜索 #include<iostream> #include<cstdio> #include<cstring> usin ...
- Codeforces Round #460 (Div. 2) B Perfect Number(二分+数位dp)
题目传送门 B. Perfect Number time limit per test 2 seconds memory limit per test 256 megabytes input stan ...
- CodeForces 55D "Beautiful numbers"(数位DP+离散化处理)
传送门 参考资料: [1]:CodeForces 55D Beautiful numbers(数位dp&&离散化) 我的理解: 起初,我先定义一个三维数组 dp[ i ][ j ][ ...
- Balanced Numbers (数位dp+三进制)
SPOJ - BALNUM 题意: Balanced Numbers:数位上的偶数出现奇数次,数位上的奇数出现偶数次(比如2334, 2出现1次,4出现1次,3出现两次,所以2334是 Balance ...
- Codeforces #55D-Beautiful numbers (数位dp)
D. Beautiful numbers time limit per test 4 seconds memory limit per test 256 megabytes input standar ...
随机推荐
- 干掉hao123劫持浏览器主页
原因可能是安装某个软件流氓捆绑了IE主页导致的,建议这样尝试: 一.如果安装有三方安全防护类软件,排查流氓软件,建议运行系统自带的Windows Defender或者MSE程序扫描系统. 二.如果有检 ...
- springboot 的war包在Tomcat中启动失败
springboot 默认是通常是打包成jar的,里面会内置一个tomcat容器 有时候我们需要使用以前打成war包的方式部署到对应的tomcat中, 具体springboot 怎么从jar改成war ...
- docker 批量删除
杀死所有正在运行的容器docker kill $(docker ps -a -q) 删除所有已经停止的容器docker rm $(docker ps -a -q) 删除所有未打 dangling 标签 ...
- .net 调用 Matlab生成dll出现的问题(The type initializer for 'MathWorks.MATLAB.NET.Utility.MWMCR' threw an exception.)
https://cn.mathworks.com/matlabcentral/answers/278399-i-get-an-error-saying-the-type-initializer-for ...
- 深入了解scanf()/getchar()和gets()/cin等函数
转:http://www.cnblogs.com/FCWORLD/archive/2010/12/04/1896511.html 转:问题描述一:(分析scanf()和getchar()读取字符) s ...
- day32基于tcp协议的远程执行命令
客户端 from socket import *import structimport json client = socket(AF_INET, SOCK_STREAM)client.connect ...
- uva-10400-搜索
题意:题意很简单,就是输入数字,对数字进行加减乘除,然后能不能得到最后的数字. wa了很多次,memcpy(dst,src,sizeof(dst))才对,最后一个参数写错,最后一个参数是要拷贝的字节数 ...
- 2008nian元旦
我要做---->> 控制财务状况---1记账 2 炒股或基金 3预算 锻炼---每次要达到锻炼的效果 学习计划-- 1报班架构 2 项目管理 3读书 爱好---做饭,收拾整理 职业规划. ...
- ORA-01219: 数据库未打开: 仅允许在固定表/视图中查询解决之道
参考文章:https://blog.csdn.net/Trigl/article/details/50933495 解决.
- PAT甲级
https://www.cnblogs.com/jlyg/p/7525244.html 哇咔咔,瞧人家都刷完甲级130道题了,哼,有什么了不起的,考前刷完,再刷的比你多,哼,照样吊打. 沙耶加油! 一 ...