题目

P4965 薇尔莉特的打字机

快到十二点了正在颓废突然发现了一道好题

虽然毒瘤,但确实是容斥原理的好题啊,做法也特别巧妙(标程

思路

题目大意(怕自己突然忘)

n个初始字符,m个操作(加入或删除),任何一个操作都可能无效,求最后不同的字符方案数\((n,m<=5*10^6)\)

先考虑无删除操作

这里的\(t_{0..i}\)为一个字串,\(\widetilde{t_{0..i}}\)指该字串不同的子序列个数,\(dp_i\)为\(\widetilde{t_{0..i-1}}\)

若\(t_{i}\)这个字符在循环时第一次出现,即\(t_{0...i-1}\)没出现过\(t_{i}\),显然我们考虑的状态有三种

\(~~~~~1.\widetilde{t_{0..i-1}}\),有 \(dp_{i-1}\)个\(\Longrightarrow\)理解:插入操作无效或只考虑前面字符的方案数

\(~~~~~2.\widetilde{t_{0..i-1}} + \{t_i\}\),有\(dp_{i-1}\)个\(\Longrightarrow\)理解:插入操作有效,且与前面字符组合起来的方案数

\(~~~~~3.\widetilde{t_{i..i}}\)\(\Longrightarrow\)理解:插入操作有效\(t_{i}\)单独组成一种方案

综上,\(t_{i}\)这个字符在循环时第一次出现:\(dp_{i}=2*dp{i-1}+1\)

\(\begin{aligned}
\\
\end{aligned}\)

那不是第一次出现呢?显然会出现重复的子序列

\(lst[c]\)表示字符\(c\) 上一次出现的位置

1.\(\widetilde{t_{0..lst[t_i]-1}}+\{t_{lst[t_i]}\}\)与\(\widetilde{t_{0..lst[t_i]-1}}+\{t_i\}\) 重复

2.\(\{t_{lst[t_i]}\}\) 与 \(\{t_i\}\) 重复

综上,\(dp_i\)要去掉\(dp_{lst[t_i]-1}+1\)

\(\therefore dp_i=\begin{cases}2*dp_{i-1}+1\quad(t_i\text{第一次出现})\\2dp_{i-1}-dp_{lst[t_i]-1}\quad (t_i\text{出现过})\end{cases}\)

目前为止,时间复杂度为\(O(m)\),毒瘤的出题人不可能就这样放过我们嘛\(emmm\)

考虑删除操作

其实删除操作只用考虑删除前面的文本串,为什么?删除插入操作无异与:删除与插入两个操作同时无效,而前面的方程已经将此情况考虑进去了

故我们只用考虑文本串与插入操作中间的删除操作

当能作为有效删除操作为\(cnt\)个时,我们枚举有\(k(k<=cnt)\)个有效操作

则此时新增子序列\(s_{0..n-k-1} + \widetilde{t_{pre[p_k]..m-1}}\)

发现没有?从小到大枚举\(k\),时间复杂度瞬间指数加\(1\)成了\(O(m^2)\),那我们就从\(m-1\)~\(0\)逆推,又变成线性的了!!

故\(dp_i\)为\(\widetilde{t_{i..m-1}}\),\(lst[c]\)为倒推时字符\(c\)上一次出现的位置,由于删除操作的存在,方程中的\(dp_{i-1}\) 改为\(dp_{pre[i]}\)(这些应该都好理解吧)

\(\begin{aligned}
\\
\end{aligned}\)

同样地,我们还得考虑重复部分,

若退格所删去的最后一个字符即\(s_{n-k}\)在 \(t_{pre[p_k]..m-1}\) 中出现过,则会产生重复的答案:第\(k\)删除在文本串\(s\)中删除的字符是\(s_{n-k}\),

而在第\(k-1\) 个删除 时,\(s_{n-k}\)不会被删除,则一旦 \(s_{n-k}\) 在 \(t_{pre[p_k]..m-1}\) 中出现过,就意味着在只有\(k-1\)个删除时\(s_{n-k}\)会与后面的发生重复

重复子序列为\(s_{0..n-k}+\widetilde{t_{pre[lst[s_{n-k}]]..m-1}}\)以及\(s_{0..n-k}\),个数为 \(dp_{pre[lst[s_{n-k}]]}+1\),计算答案时要将这部分减去

My complete code

上短得可怜的代码

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
const int p=0x125E591;
const int maxn=5000000+9;
int n,m,pos,cnt,ans;
int lst[maxn],pre[maxn],dp[maxn];
char s[maxn],t[maxn];
inline int Get(char x){
return (lst[x])?p-dp[pre[lst[x]]]:1;
}
int main(){
scanf("%d%d",&n,&m);
scanf(" %s %s",s,t);
for(int i=0;i<m;++i)
cnt+=(t[i]=='u');
for(int i=m-1;i>=0;--i)
if(t[i]=='u'){
if(n-cnt>=0)
ans=(ans+dp[pos]+Get(s[n-cnt]))%p;
--cnt;
}else{
dp[i]=(2*dp[pre[i]=pos]+Get(t[i]))%p;
pos=lst[t[i]]=i;
}
printf("%d",(ans+dp[pos]+1)%p);
return 0;
}

总结

动规啊容斥啊这些真的得完全弄懂再去写代码,几次想写代码了还是回过头自己纯手推了一下

能想到写篇博客竟然花了快一个多小时,凌晨一点半 其实也还早啊,睡觉

P4965 薇尔莉特的打字机的更多相关文章

  1. 洛谷4965 薇尔莉特的打字机(Trie,DP)

    神仙题. 考虑在一棵 Trie 上进行染色,将可能出现的串的末尾染成黑色.答案就是黑点的个数.一开始只有 \(A\) 的末尾点是黑色. 当出现一个字符(不是退格)\(c\) 时,就要将每个黑点的 \( ...

  2. ACM训练联盟周赛(第一场)

    B:Zeratul与Xor 题目描述 Xor(按位异或),对应C++中的“^”运算符. Zeratul给出了一个数列A[n](n≤105),要做q(q≤105)组动作,这些动作包括: 1  a:数列中 ...

  3. 【胡策篇】题解 (UOJ 192 + CF938G + SPOJ DIVCNT2)

    和泉纱雾与烟花大会 题目来源: UOJ 192 最强跳蚤 (只改了数据范围) 官方题解: 在这里哦~(说的很详细了 我都没啥好说的了) 题目大意: 求树上各边权乘积是完全平方数的路径数量. 这种从\( ...

  4. Python语言之面向对象

    Python语言之面向对象 前言 面向对象 -- Object Oriented 简写:OO 面向对象设计 -- Object Oriented Design 简写:OOD 面向对象编程 -- Obj ...

  5. 小贝_mysql select连接查询

    select连接查询 简要: 一.union联合查询 二.左右内连接 一.union联合查询 作用: 把2次或多次查询结果合并起来 具体: (表1查询结果) union (表2查询结果) 运行: 先算 ...

  6. 营口6378.7939(薇)xiaojie:营口哪里有xiaomei

    营口哪里有小姐服务大保健[微信:6378.7939倩儿小妹[营口叫小姐服务√o服务微信:6378.7939倩儿小妹[营口叫小姐服务][十微信:6378.7939倩儿小妹][营口叫小姐包夜服务][十微信 ...

  7. 本溪6397.7539(薇)xiaojie:本溪哪里有xiaomei

    本溪哪里有小姐服务大保健[微信:6397.7539倩儿小妹[本溪叫小姐服务√o服务微信:6397.7539倩儿小妹[本溪叫小姐服务][十微信:6397.7539倩儿小妹][本溪叫小姐包夜服务][十微信 ...

  8. FreeBSD基金会添加新成员,梁莉成为第一位来自微软和中国的基金会董事

    这个月23日FreeBSD基金会很高兴地宣布Philip Paeps和Kylie Liang (梁莉)正式加入董事会. 梁莉,现任微软开源技术部高级项目经理,主要负责FreeBSD在公有云以及私有云的 ...

  9. 【爆料】-《英博夏尔大学毕业证书》BPP一模一样原件

    英博夏尔大学毕业证[微/Q:2544033233◆WeChat:CC6669834]UC毕业证书/联系人Alice[查看点击百度快照查看][留信网学历认证&博士&硕士&海归&a ...

随机推荐

  1. Qt Creator中增加新的ui文件时报错

    原因分析:moc_开头的文件编译过程中没有又一次生成导致. 解决的方法:删除编译产生的build目录.又一次编译就可以. 错误类型截图例如以下: 这个问题的解决.使得能够在不论什么时候都能够在proj ...

  2. amchart 图表设置

    官网:https://www.amcharts.com/demos/ 属性介绍:https://docs.amcharts.com/3/javascriptcharts/AmLegend 安装 bow ...

  3. 跟我一起写 Makefile(二)[转]

    原文链接 http://bbs.chinaunix.net/thread-408225-1-1.html(出处: http://bbs.chinaunix.net/) 一.Makefile里有什么? ...

  4. mac上搭建docker镜像私服

    1.创建私服容器 docker run -d -e SETTINGS_FLAVOR=dev -e STORAGE_PATH=/tmp/registry -v /opt/data/registry:/t ...

  5. 织梦DedeCMS自定义表单提交成功后返回当前页面的教程

    织梦的自定义表单制作的留言,报名等功能,提交成功后会自动返回到首页,那么如何让它返回到当前页面呢? 方法如下: 打开plus/diy.php文件 找到 showmsg($bkmsg, $goto); ...

  6. 使用虚拟环境 virtualenv

    1.安装 $ sudo apt-get install python-virtualenv 2.重命名,一般虚拟环境会被命名为venv $ virtualenv   venv 3. 激活 $ sour ...

  7. strpos 判断字符串是否存在

    strpos    中为什么要用逗号隔开的原因是因为   防止找出相匹配的中 ,  如   查找1    而数组中  存在  12  那么这个结果也是可以找出来的 ,分别在1 前后加个,  就是为了区 ...

  8. 从零开始学android -- 简易的socket通信

    先来介绍下socket,网上摘抄点资料,免得自己打字了 网络中进程之间如何通信? 本地的进程间通信(IPC)有很多种方式,但可以总结为下面4类: 1.消息传递(管道.FIFO.消息队列) 2.同步(互 ...

  9. iOS开发 Xcode8 问题

      一.证书管理 用Xcode8打开工程后,比较明显的就是下图了,这个是苹果的新特性,可以帮助我们自动管理证书.建议大家勾选这个Automatically manage signing(Ps.但是在b ...

  10. RecyclerView的使用(3)之加入Header和Footer

    原创文章.转载请注明 http://blog.csdn.net/leejizhou/article/details/50742544 李济洲的博客 RecyclerView尽管作为ListView的替 ...