正题

题目链接:https://loj.ac/p/6564


题目大意

给两个序列\(a,b\)求它们的最长公共子序列。

\(1\leq n,m,a_i,b_i\leq 7\times 10^4\)


解题思路

无意间看到的一个\(bitset\)科技。

首先设\(f_{i,j}\)表示\(a\)串匹配到第\(i\)个\(b\)串匹配到第\(j\)个时的最长长度,做过\(dp\)套\(dp\)的应该知道\(f_{i,j}\)的性质。

\[0\leq f_{i,j}-f_{i,j-1}\leq 1
\]

基本的思路就是设\(01\)矩阵\(M\)满足\(f_{i,j}=\sum_{k=1}^jM_{i,k}\)然后用\(bitset\)优化转移

然后考虑一下怎么转移,我们先预处理出\(p\)数组其中\(p_i\)表示数字\(i\)出现的位置集合

我们的转移要把\(M\)中的\(1\)尽量的往前移动并且看能否加上一个新的\(1\)。

假设现在的字符是\(c\),那么我们将使用\(p_c\)进行转移。

我们把\(M\)中每个\(1\)作为结尾分成若干段(最后的\(0\)也是一段,顺序是从低位到高位)。

那么对于一段中如果这一段\(p_c\)有\(1\)那么我们就取最前面的那个\(1\),这样因为前面假设有\(j\)个\(1\)那么这次就匹配\(p_c\)最前面的那个作为\(j+1\)。

但是我们显然不可能一段一段做,我们可以考虑怎么把这个操作转成位运算。

考虑一下我们平时是怎么取一个\(01\)串的第一位的,我们有\(lowbit(x)=((x-1)\ xor\ x)\ and\ x\)对吧。

发现这里每段分开取实际上难实现的地方只有\(x-1\),考虑怎么实现这个问题。

因为\(1\)是段的末尾,所以每一段的开头前面都是\(1\),所以如果我们让\(M\)左移一位那么就变成开头是\(1\)了(需要注意补上第一段的\(1\),所以应该是\((M<<1)+1\))

最后来说是

\[M=(X-(M<<1)+1)\ xor\ X\ and\ X
\]

这样我们就完成了\(M\)的转移,因为需要位运算,所以需要手写\(bitset\)。

时间复杂度\(O(\frac{nm}{\omega})\)

我是看这篇博客学的,看不懂的可以去看下:https://www.cnblogs.com/-Wallace-/p/bit-lcs.html


code

#include<cstdio>
#include<cstring>
#include<algorithm>
#define ull unsigned long long
using namespace std;
const int N=7e4+10;
int n,m,L;
struct bitset{
ull t[N/64+5];
bitset(){memset(t,0,sizeof(t));return;}
void set(int p){
t[p>>6]|=(1ull<<(p&63));
return;
}
void shift(){
ull last=0;
for(int i=0;i<L;i++){
ull cur=t[i]>>63;
(t[i]<<=1)|=last;
last=cur;
}
return;
}
int count(){
int ans=0;
for(int i=0;i<L;i++)
{ull x=t[i];while(x)x-=(x&-x),ans++;}
return ans;
}
bitset& operator=(const bitset &b)
{memcpy(t,b.t,sizeof(t));return *this;}
bitset& operator|=(const bitset &b){
for(int i=0;i<L;i++)t[i]|=b.t[i];
return *this;
}
bitset& operator&=(const bitset &b){
for(int i=0;i<L;i++)t[i]&=b.t[i];
return *this;
}
bitset& operator^=(const bitset &b){
for(int i=0;i<L;i++)t[i]^=b.t[i];
return *this;
}
}p[N],f,g;
bitset operator-(const bitset &a,const bitset &b){
bitset tmp;ull last=0;
for(int i=0;i<L;i++){
ull cur=(a.t[i]<b.t[i]+last);
tmp.t[i]=a.t[i]-b.t[i]-last;
last=cur;
}
return tmp;
}
int main()
{
scanf("%d%d",&n,&m);L=(n>>6)+1;
for(int i=1,c;i<=n;i++)
scanf("%d",&c),p[c].set(i);
for(int i=1,c;i<=m;i++){
scanf("%d",&c);
(g=f)|=p[c];
f.shift();f.set(0);
f=g-f;f^=g;f&=g;
}
printf("%d",f.count());
return 0;
}

[科技]Loj#6564-最长公共子序列【bitset】的更多相关文章

  1. 【科技】位运算(bitset)优化最长公共子序列算法

    最长公共子序列(LCS)问题 你有两个字符串 \(A,B\),字符集为 \(\Sigma\),求 \(A, B\) 的最长公共子序列. 简单动态规划 首先有一个广为人知的 dp:\(f_{i,j}\) ...

  2. POJ1159——Palindrome(最长公共子序列+滚动数组)

    Palindrome DescriptionA palindrome is a symmetrical string, that is, a string read identically from ...

  3. POJ 1159 Palindrome(最长公共子序列)

    Palindrome [题目链接]Palindrome [题目类型]最长公共子序列 &题解: 你做的操作只能是插入字符,但是你要使最后palindrome,插入了之后就相当于抵消了,所以就和在 ...

  4. 51NOD 1006 最长公共子序列 Lcs 动态规划 DP 模板题 板子

    给出两个字符串A B,求A与B的最长公共子序列(子序列不要求是连续的). 比如两个串为: abcicba abdkscab ab是两个串的子序列,abc也是,abca也是,其中abca是这两个字符串最 ...

  5. [csu/coj 1078]多个序列的最长公共子序列

    题意:给n个序列,同一个序列里面元素互不相同,求它们的最长公共子序列. 思路:任取一个序列,对于这个序列里面的两个数ai,aj(i<j),如果对于其它每一个序列,都出现过ai,aj,且ai在aj ...

  6. 用python实现最长公共子序列算法(找到所有最长公共子串)

    软件安全的一个小实验,正好复习一下LCS的写法. 实现LCS的算法和算法导论上的方式基本一致,都是先建好两个表,一个存储在(i,j)处当前最长公共子序列长度,另一个存储在(i,j)处的回溯方向. 相对 ...

  7. 动态规划之最长公共子序列(LCS)

    转自:http://segmentfault.com/blog/exploring/ LCS 问题描述 定义: 一个数列 S,如果分别是两个或多个已知数列的子序列,且是所有符合此条件序列中最长的,则 ...

  8. [Data Structure] LCSs——最长公共子序列和最长公共子串

    1. 什么是 LCSs? 什么是 LCSs? 好多博友看到这几个字母可能比较困惑,因为这是我自己对两个常见问题的统称,它们分别为最长公共子序列问题(Longest-Common-Subsequence ...

  9. 动态规划求最长公共子序列(Longest Common Subsequence, LCS)

    1. 问题描述 子串应该比较好理解,至于什么是子序列,这里给出一个例子:有两个母串 cnblogs belong 比如序列bo, bg, lg在母串cnblogs与belong中都出现过并且出现顺序与 ...

随机推荐

  1. wpf 自定义 RadioButton.

    <Style TargetType="RadioButton" x:Key="nav"> <Setter Property="Tem ...

  2. pyspark默认使用python2-----更改

    默认使用的竟然是2.7好烦如何解决呢 配置环境变量就行了 vi ~/.bashrc 添加一句话 export PATH=/home/hadoop/app/python3/bin:$PATH 保存退出  ...

  3. servlet中servletContext的五大作用(二)

    1.    获取web的上下文路径 2.    获取全局的参数 3.    作为域对象使用 4.    请求转发 5.    读取web项目的资源文件 package day10.about_serv ...

  4. Linux命令集锦之·正则表达式

    时间:2018-11-15 记录:byzqy 正则表达式与通配符: 正则表达式,用来在文件中匹配符合条件的字符串,正则是包含匹配. grep.awk.sed 等命令可以支持正则表达式. 通配符,用来匹 ...

  5. css之px、em、rem

    rem是css3新定义的设置字体大小属性,常用的两种字体大小设置有下面2种:1. px为单位2.em为单位(百分比用法跟em类似) PX为单位 在Web页面初期制作中,我们都是使用"px&q ...

  6. 菜鸟入门Linux之路(方法论浅谈)

    Linux是为人熟知的OS之王,已"统治"世界.要想学好绝非易事. 作为菜鸟,可以与Linux亲密接触的方法很多,如视频.书籍.各种企培资料等等,如今的在线教育也如火如荼. 总结说 ...

  7. RHCA-openstack学习部署笔记-CL210

    笔记说明:目前考试是基于RHELOSP13(32G)来进行环境部署,引入了容器,而学习是基于RHELOSP6来安装部署的,基于Openstack. 讨论:openstack.虚拟化和云计算的关系以及区 ...

  8. Spring(一)——概述

    一.概述 1.介绍 struts 是 web 框架 (jsp/action/actionfrom).hibernate是orm (Object Relational Mapping) 框架,处于持久层 ...

  9. error : Hooks can only be called inside of the body of a function component. 依赖包和主包加载多个react 引发冲突问题

    1. 结论: 在依赖包和主包的node-modules中,同时install包含react , react-dom 或者react-redux时,跑测试或者启动的时候,显示如下error.当然出现这个 ...

  10. Appium问题解决方案(8)- selenium.common.exceptions.WebDriverException: Message: An unknown server-side error occurred while processing the command. Original error: Could not sign with default certificate.

    背景 运行时代码报错: selenium.common.exceptions.WebDriverException: Message: An unknown server-side error occ ...