洛谷题目页面传送门 & CodeForces题目页面传送门

题意见洛谷里的翻译。

首先我们可以用区间DP算出对于每个子01串,能表示的字母串的个数。

设\(dp_{i,j}\)表示长度为\(i\),起点为\(j\)(下标从\(1\)开始)的子01串能表示的字母串的个数(也许我设的DP状态有点奇怪)。显然,\(dp_{0,i}=1\),因为空子01串能且仅能表示空字母串。

那么转移怎么转移呢?可以从串的首部转移,从首部挖出\(1\sim4\)个字符组成字母,累加上剩下的串的DP值。那么状态转移方程为\(dp_{i,j}=\sum\limits_{k=1}^{\min(4,i)}dp_{i-k,j+k}\),其中\(k=4\)时还要考虑首部\(4\)个字符是否为不能表示字母的那可怜的\(4\)个01串,如果是的话,就不能累加上\(dp_{i-4,j+4}\)。同样的道理,从尾部转移也是可以的。转移的总时间复杂度是\(\mathrm O\!\left(n^2\right)\),因为一共有\(\mathrm O\!\left(n^2\right)\)个子01串,而每次转移是\(\mathrm{O}(1)\)的。

接下来要算答案了。设到当前第\(i\)次的添加字符后,答案为\(ans\),那么容易想到,下一次(第\(i+1\)次)添加字符后,增加了\([1,i+1],[2,i+1],\cdots,[i+1,i+1]\)这\(i+1\)个子01串,答案变成了\(ans+\sum\limits_{j=1}^{i+1}dp_{i+1-j+1,j}\),我们把\(\sum\limits_{j=1}^{i+1}dp_{i+1-j+1,j}\)累加到\(ans\)里。但是,在累加的过程中,可能对于有的\(j\),\([j,i+1]\)这段子01串和之前累加的有重复,也就是说这段子01串所能表示的所有字母串都被算过了,就不能再把\(dp_{i+1-j+1,j}\)累加上了。所以我们需要判重。

我们可以预处理出前缀哈希数组\(Hsh\),这样可以在\(\mathrm{O}(1)\)时间内计算出任意一个子串的哈希值。然后怎么判重呢?用一个\(vis\)数组肯定不行,因为哈希值在\(\left[0,2^{64}\right)\)内,会MLE。用裸的set,要判\(\mathrm O\!\left(n^2\right)\)次,每次\(\mathrm O\!\left(\log n^2\right)=\mathrm{O}(\log n)\),总共\(\mathrm O\!\left(n^2\log n\right)\);将set按子01串长度分成\(n\)组,也还是\(\mathrm O\!\left(n^2\log n\right)\)。这复杂度配上STL的常数,会TLE。

那该怎么办呢?事实上,set的组数其实应该尽可能多,那样每个set的元素个数就尽可能少,复杂度就尽可能小。不妨按哈希值模\(10^7\)的值分为\(10^7\)组。这样分,平均每组只有一两个元素,用set还亏了,直接用链式前向星装即可。

以下是AC代码:

#include<bits/stdc++.h>
using namespace std;
#define N 3000
#define hshmod 10000000//将哈希分为的组数
#define mod 1000000007
#define ull unsigned long long
bitset<N+1> a;//01串
int dp[N+1][N+2];//dp[i][j]表示长度为i,起点为j的子01串能表示的字母串数
inline bool ok(bool a,bool b,bool c,bool d){//是否能表示字母
return !(!a&&!b&&c&&d||!a&&b&&!c&&d||a&&b&&c&&!d||a&&b&&c&&d);//不是那可怜的4个串就OK
}
ull Hsh[N+1]/*前缀哈希值*/,power[N+1]/*power[i]为哈希base(131)的i次方*/;
ull hsh(int l,int r){//[l,r]这段子01串的哈希值
return Hsh[r]-Hsh[l-1]*power[r-l+1];
}
int head[hshmod],nxt[N*N],tot;ull data[N*N];//链式前向星
void add(int x,ull y){//往第x组里添加哈希值y
data[++tot]=y;nxt[tot]=head[x];head[x]=tot;
}
int main(){
int n/*添加字符的次数,即最终01串的长度*/,ans=0/*目前的答案*/,i,j;scanf("%d",&n);
for(i=1;i<=n;i++){int x;scanf("%d",&x);a[i]=x;}
power[0]=1;//131^0=1
for(i=1;i<=n;i++)Hsh[i]=Hsh[i-1]*131ull+a[i]+1,power[i]=power[i-1]*131ull;//预处理
fill(dp[0]+1,dp[0]+n+2,1);//DP边界:dp[i][0]=1
for(i=1;i<=n;i++)for(j=1;j+i-1<=n;j++)//状态转移
dp[i][j]=dp[i-1][j+1],
dp[i][j]+=i>=2?dp[i-2][j+2]:0,dp[i][j]%=mod,
dp[i][j]+=i>=3?dp[i-3][j+3]:0,dp[i][j]%=mod,
dp[i][j]+=i>=4&&ok(a[j],a[j+1],a[j+2],a[j+3])?dp[i-4][j+4]:0,dp[i][j]%=mod;
// for(i=1;i<=n;i++)for(j=1;j+i-1<=n;j++)
// printf("dp[%d][%d]=%d\n",i,j,dp[i][j]);
for(i=1;i<=n;i++){//第i次添加字符
for(j=1;j<=i;j++){//枚举新增子01串的左端点,右端点是i
ull h=hsh(j,i);//此新子01串的哈希值
int hmod=h%hshmod;//它的类别
// cout<<h<<" "<<hmod<<" "<<dp[i-j+1][j]<<"\n";
bool mar=false;//是否和之前的某一个串重复
for(int k=head[hmod];k;k=nxt[k])//枚举此组里的元素
if(data[k]==h)//有和它的哈希值相同的?
{mar=true;/*重复*/break;/*不必再找,退出*/}
if(!mar)//如果无重复,照样累加
ans+=dp[i-j+1][j],ans%=mod,add(hmod,h)/*扔进链表*/;
}
printf("%d\n",ans);//输出目前的答案
// puts("");
}
return 0;
}

CodeForces 1129C Morse Code的更多相关文章

  1. morse code

    morse code,摩斯电码,是一种时通时断的信号代码,通过不同的排列顺序来表达不同的英文字母.数字和标点符号. 摩斯电码,是一种早期的数字化通信形式,但是它不同于现代只使用0和1两种状态的二进制代 ...

  2. 摩尔斯电码(Morse Code)Csharp实现

    摩尔斯电码,在早期的"生产斗争"生活中,扮演了很重要的角色,作为一种信息编码标准,摩尔斯电码拥有其他编码方案无法超越的长久生命.摩尔斯电码在海事通讯中被作为国际标准一直使用到199 ...

  3. Leetcode 804. Unique Morse Code Words 莫尔斯电码重复问题

    参考:https://blog.csdn.net/yuweiming70/article/details/79684433 题目描述: International Morse Code defines ...

  4. [Swift]LeetCode804. 唯一摩尔斯密码词 | Unique Morse Code Words

    International Morse Code defines a standard encoding where each letter is mapped to a series of dots ...

  5. [LeetCode] Unique Morse Code Words 独特的摩斯码单词

    International Morse Code defines a standard encoding where each letter is mapped to a series of dots ...

  6. (string 数组) leetcode 804. Unique Morse Code Words

    International Morse Code defines a standard encoding where each letter is mapped to a series of dots ...

  7. 804. Unique Morse Code Words

    Description International Morse Code defines a standard encoding where each letter is mapped to a se ...

  8. Unique Morse Code Words

    Algorithm [leetcode]Unique Morse Code Words https://leetcode.com/problems/unique-morse-code-words/ 1 ...

  9. LeetCode - 804. Unique Morse Code Words

    International Morse Code defines a standard encoding where each letter is mapped to a series of dots ...

随机推荐

  1. 在eclipse中使用git创建本地库,以及托管项目到GitHub超详细教程

    关于安装git的教程,由于比较简单,并且网上教程特别多,而且即使不按照网上教程,下载好的windows版本git,安装时候一路默认设置就行. 安装好之后,在桌面上有git图标:右键菜单中有Git Ba ...

  2. QRowTable表格控件-支持hover整行、checked整行、指定列排序等

    目录 一.开心一刻 二.嘴一嘴 三.效果展示 四.浅谈实现 五.自定义数据源 1.data函数 2.flags函数 六.自定义视图 1.目的 2.问题分析 七.测试 八.相关文章 原文链接:QRowT ...

  3. 计算机组成原理第五章(中央处理器CPU)

    ---恢复内容开始--- 指令周期(取指令.分析指令到执行完该指令所需的全部时间) 机器周期通常又称CPU周期 通常把一条指令周期分成若干个机器周期,每个机器周期完成一个基本操作 以主存的工作周期(存 ...

  4. 浅谈 Attention 机制的理解

    什么是注意力机制? 注意力机制模仿了生物观察行为的内部过程,即一种将内部经验和外部感觉对齐从而增加部分区域的观察精细度的机制.例如人的视觉在处理一张图片时,会通过快速扫描全局图像,获得需要重点关注的目 ...

  5. PostgreSQL 窗口函数 ( Window Functions ) 如何使用?

    一.为什么要有窗口函数 我们直接用例子来说明,这里有一张学生考试成绩表testScore: 现在有个需求,需要查询的时候多出一列subject_avg_score,为此科目所有人的平均成绩,好跟每个人 ...

  6. 深入理解Java虚拟机一 阅读笔记

    xl_echo编辑整理.欢迎添加echo微信(微信号:t2421499075)交流学习. 百战不败,依不自称常胜,百败不颓,依能奋力前行.--这才是真正的堪称强大!! --- > 以下内容摘抄自 ...

  7. SSAS Tabular 数据表关系与join的映射

    才想起来总结这两天发现的一个有趣的现象: 在SSAS Tabular 模型中建立了关系之后,在excel中分析发现: 产品库龄作为量值:Aging Stock:=sum([DISTRIBUTOR_ST ...

  8. py+selenium+IE 批量执行脚本10几分钟,IE会卡住【无解,提供绕过方法】

    问题:py+selenium+IE 批量执行单个脚本10几分钟,IE会卡住 一个脚本文件里有20几个用例,跑起来10多分钟,每次跑10分钟后(即第22条用例左右时)IE就会卡住,程序就会在那傻等,最后 ...

  9. 1.低权限的程序向高权限的程序发消息 2.慎用setcurrentdirectory

    1.低权限的程序向高权限的程序发消息 2.慎用setcurrentdirectory

  10. 实验吧--web--天下武功唯快不破

    ---恢复内容开始--- 英文翻译过来嘛,就是:天下武功无快不破嘛.(出题者还是挺切题的) 看看前端源码: 注意这里 please post what you find with parameter: ...