[ CERC 2014 ] Vocabulary
\(\\\)
\(Description\)
给出三个长度分别为 \(lenA,lenB,lenC\) 的三个字符串 \(A,B,C\) ,其中字符集只包括所有小写字母以及 \(?\) 号。
现在将所有 \(?\) 号改为任意小写字母,问有多少种方案,使得 \(A,B,C\) 三个字符串按照字典序排列\((\)相同不算\()?\)
输出方案数对 \(10^9+9\) 取模后的结果。多组询问。
- 字符串总长度\(\le 10^6\)
\(\\\)
\(Solution\)
有趣的预处理题目。
三个串长度不能确定,不能通过数据组数判断复杂度要求,只好按照上界分析,即只有一组数据,做法接近线性。
首先考虑三个串不等长的问题,字典序中规定,一个串的任意长度前缀都比当前串字典序小,所以在不是最长的串后面补上若干个比 \(a\) 还要小的字符即可。
\(\\\)
考虑为了使得最后合法,三个串的字典序在确定 \(?\) 号的时候合法的情况(小于号表示字典序左侧优于右侧):
- \(1:\ A=B=C\)
- \(2:\ A<B=C\)
- \(3:\ A=B<C\)
- \(4:\ A<B<C\)
其他情况会导致在某一位置某两串字典序相反,这样后面不管怎样设置字典序都是反过来的。
\(\\\)
动态规划。
设\(f[i][1/2/3/4]\)表示,当前位置为\(i\),当前位置及以前的所有 \(?\) 号都确定完了之后,使得三个串的字典序关系变为 \(1/2/3/4\) 的方案数。
为了便于初始化,将字符串都向后推一个,然后有\(f[0][1]=1\),答案为\(f[maxlen][4]\)。
考虑转移的过程,合法的转移有:\(1 \to 1/2/3/4\ ,\ 2\to 2/4\ ,\ 3\to 3/4\ ,\ 4\to4\),想不明白可以手玩一下,当前位置每一种情况的字典序的前提是不同的,例如 \(1\) 类情况必须要求前面的位置都为 \(1\) 类情况。
\(\\\)
然后考虑转移的过程,显然我们的转移影响要素有,前一位置状态,以及当前位置三个字符串的字符。
考虑到每次操作统计方案数非常麻烦,而转移只有 \(28^3\times(4+2+2+1)\) 种\((\)后面括号里对应着状态间的转移\()\),不妨对每个转移处理出来方案数。
设 \(g[i][j][x][y][z]\) 表示,要从状态 \(i\) 转移到 \(j\) ,当前位置三个串的字符分别为 \(x,y,z\) 的方案数,预处理就暴力枚举三个字符,注意从上到下字符区间的限制,然后将可以累加的转移方案数累加就好。
然后我们动规的转移就可以直接借用 \(g\) 数组转移了,单组数据复杂度 \(\text O (16N)\)。
\(\\\)
\(Code\)
#include<cmath>
#include<cstdio>
#include<cctype>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#define N 1000010
#define R register
#define mod 1000000009
using namespace std;
typedef long long ll;
char a[N],b[N],c[N];
ll f[N][4],g[4][4][28][28][28];
int s1[N],s2[N],s3[N],lena,lenb,lenc,len;
inline void prework(){
int l1,r1,l2,r2,l3,r3;
for(R int i=0;i<=27;++i)
for(R int j=0;j<=27;++j)
for(R int k=0;k<=27;++k){
(i==27)?l1=1,r1=26:l1=r1=i;
for(R int x=l1;x<=r1;++x){
(j==27)?l2=1,r2=26:l2=r2=j;
for(R int y=l2;y<=r2;++y){
(k==27)?l3=1,r3=26:l3=r3=k;
for(R int z=l3;z<=r3;++z){
++g[3][3][i][j][k];
if(x==y) ++g[1][1][i][j][k];
if(x<y) ++g[1][3][i][j][k];
if(y==z) ++g[2][2][i][j][k];
if(y<z) ++g[2][3][i][j][k];
if(x==y&&y==z) ++g[0][0][i][j][k];
if(x==y&&y<z) ++g[0][1][i][j][k];
if(x<y&&y==z) ++g[0][2][i][j][k];
if(x<y&&y<z) ++g[0][3][i][j][k];
}
}
}
}
}
inline void init(){
scanf("%s",a); lena=strlen(a);
scanf("%s",b); lenb=strlen(b);
scanf("%s",c); lenc=strlen(c);
len=max(lena,max(lenb,lenc));
for(R int i=0;i<lena;++i) s1[i+1]=(a[i]=='?'?27:a[i]-'a'+1);
for(R int i=lena;i<=len;++i) s1[i+1]=0;
for(R int i=0;i<lenb;++i) s2[i+1]=(b[i]=='?'?27:b[i]-'a'+1);
for(R int i=lenb;i<=len;++i) s2[i+1]=0;
for(R int i=0;i<lenc;++i) s3[i+1]=(c[i]=='?'?27:c[i]-'a'+1);
for(R int i=lenc;i<=len;++i) s3[i+1]=0;
}
int main(){
prework();
int t;
scanf("%d",&t);
while(t--){
init();
for(R int i=0;i<=len;++i)
for(R int j=0;j<4;++j) f[i][j]=0;
f[0][0]=1;
for(R int i=1;i<=len;++i)
for(R int k=0;k<4;++k)
if(f[i-1][k]) for(R int j=0;j<4;++j)
(f[i][j]+=f[i-1][k]*g[k][j][s1[i]][s2[i]][s3[i]])%=mod;
printf("%lld\n",f[len][3]);
}
return 0;
}
[ CERC 2014 ] Vocabulary的更多相关文章
- CERC 2014 (动态树+主席树)
CERC 2014 Pork barrel Problem : n个点m条边有边权的无向图,有q个询问,每次询问权值在[L,R]内的边组成的最小生成树的权值和,强制在线. n <= 1000, ...
- 【CERC 2014 E】2048
题意 2048曾经是一款风靡全球的小游戏. 今天,我们换一种方式来玩这个小游戏. 你有一个双端队列,你只能把元素从左端或从右端放入双端队列中.一旦放入就不得取出.放入后,若队列中有连续两个相同的元素, ...
- ICLR 2014 International Conference on Learning Representations深度学习论文papers
ICLR 2014 International Conference on Learning Representations Apr 14 - 16, 2014, Banff, Canada Work ...
- 漫谈2014年人机交互(CHI)大会
编者按:2014年度以人机交互为主题的顶级会议ACM SIGCHI已经落下帷幕.微软研究院在此次会议的入选论文总数仅次于卡耐基•梅隆大学,位列第二.此次会议中,有哪些创新想法或技术让人眼前一亮?听微软 ...
- 看完SQL Server 2014 Q/A答疑集锦:想不升级都难!
看完SQL Server 2014 Q/A答疑集锦:想不升级都难! 转载自:http://mp.weixin.qq.com/s/5rZCgnMKmJqeC7hbe4CZ_g 本期嘉宾为微软技术中心技术 ...
- SQL Server 2014 新特性——内存数据库
SQL Server 2014 新特性——内存数据库 目录 SQL Server 2014 新特性——内存数据库 简介: 设计目的和原因: 专业名词 In-Memory OLTP不同之处 内存优化表 ...
- 2014年暑假c#学习笔记目录
2014年暑假c#学习笔记 一.C#编程基础 1. c#编程基础之枚举 2. c#编程基础之函数可变参数 3. c#编程基础之字符串基础 4. c#编程基础之字符串函数 5.c#编程基础之ref.ou ...
- SQL Server 2014聚集列存储索引
转发请注明引用和原文博客(http://www.cnblogs.com/wenBlog) 简介 之前已经写过两篇介绍列存储索引的文章,但是只有非聚集列存储索引,今天再来简单介绍一下聚集的列存储索引,也 ...
- Web 前端攻防(2014版)
在百度 FEX 团队时写的.有次让我写点前端开发中注意的安全规范,结果写着写着就跑题了,写了一堆纯前端实现的攻击方式...当然还有防范措施. 文章就懒得粘过来了~ 直接贴个地址算了:) http:// ...
随机推荐
- Spring在Java Filter注入Bean为Null的问题解决
在Spring的自动注入中普通的POJO类都可以使用@Autowired进行自动注入,但是除了两类:Filter和Servlet无法使用自动注入属性.(因为这两个归Web容器管理)可以用init(集承 ...
- list.ensureCapacity竟然会变慢
list.ensureCapacity竟然会变慢 jdk1.8 应该是做了优化了: public class Test10 { public static void main(String[] arg ...
- [Java] 监控java对象回收的原理与实现
监控Java对象回收的原理与实现 一.监控Java对象回收的目的 监控Java对象是否回收的目的是:为了实现内存泄露报警. 内存泄露是指程序中对象生命周期(点击查看详情)已经进入不可见阶段,但因为编码 ...
- UVA10056 - What is the Probability ?(概率)
UVA10056 - What is the Probability ? (概率) 题目链接 题目大意:有n个人玩游戏,一直到一个人胜出之后游戏就能够结束,要不然就一直从第1个到第n个循环进行,没人一 ...
- centos7 安装8188eu驱动小记
最小化安装把lsusb和lspci装上 使用lsusb 和lspci的命令, centos上的安装命令: yum -y install usbutils yum -y install pciutils ...
- Massive Data Mining学习记录
第一周: 学习PageRank, 知识点:每个节点的权值由其他节点的投票决定,所有节点的权值和为1 当节点很多时候必须转换成矩阵运算来计算节点的最终值,由马尔可夫链可以证明,这个值可以迭代得到 问题: ...
- tracert 路由跟踪程序
C:\Users\Administrator>tracert 10.0.0.1 通过最多 30 个跃点跟踪到 10.0.0.1 的路由 1 <1 毫秒 1 ms 3 ms 192.168. ...
- Probabilistic interpretation
Under the previous probabilistic assumptions on the data, least-squares regression corresponds to fi ...
- ubuntu 16.04 Sqoop 安装
1.下载:https://mirrors.tuna.tsinghua.edu.cn/apache/sqoop/1.4.6/ sqoop-1.4.6.bin__hadoop-2.0.4-alpha.ta ...
- 15.extjs tabPanel的用法
转自:https://blog.csdn.net/mezhaha/article/details/78878894 本文导读:TabPanel继承于Ext.Panel,Ext.TabPanel就是有选 ...