TG-WC2021/2022 笔记

DAY1

DP

判断\(a\)的第\(i\)为(从低到高)是否为\(1\)

(1<<(-1))&a

\(a\)\(i\)位改成\(1\)

a|(a<<(i-1))

\(a\)\(i\)位改成\(0\)

a&(~(1<<i))

\(a\)最后一个\(i\)去掉

a&(a-1)

关于四边形不等式

之前说区间\(DP\)死了,但是它可以被四边形不等式优化
关于它,不需完全会证明,只需知道三条引理:

  • 1 若两个区间的vul值中,大区间的vul更大,那么vul满足四边形不等式
  • 2 形如石子合并的区间DP转移方程,与vul有关的,也满足四边形不等式,另外,该定理只对min生效
  • 3 记sij为fij的中间决策点,若f满足不等式,那么sij-1<=sij<=si+1j
    另:\(LH\)大佬给出了一个好方法检验

第一步,造一个较大的数据(但暴力能通过)
第二步,跑暴力,存好答案
第三步,跑不等式,存好答案
第四步,比对

看起来傻,实则很好

期望

求连续抛n个正面的期望次数

\[E_x=E_{x-1}+\frac{1}{2}*(1+E_x+1) \]

\[E_x=2^{x+1}+2 \]

DAY2

网络流

没讲什么有趣的,不过重新学习了dicnic的证明方法,把牛顿迭代看了(摸鱼!)

字符串

扩展KMP

模板
结合一下马拉车和KMP,就是说,我们想求对于母串的所有后缀与模式串的最长前缀长度
首先不要被带进误区,对于母串我们正向扫,发现若一个点往后能扩展很长,就能覆盖后面的一些点
很像马拉车,我们发现,要是一个点在覆盖区,我们能向前找到一个点,像kmp一样,我们往前找,若新点和1号点一样,那么我们看看能不能把他往后延展的部分搬到前面来
这样就用马拉车的套路结合了kmp的操作

#include<iostream>
#include<cstdio>
#include<cstring>
#define ll long long
using namespace std;
const ll N=2e7+30;
ll n,m,nxt[N],Nxt[N],ans1,ans2;
char c[N],s[N];

inline void get_nxt(){
	nxt[1]=m;
	for(ll r=0,mid=0,i=2;i<=m;i++){
		if(i<=r)nxt[i]=min(nxt[i-mid+1],r-i+1);
		while(i+nxt[i]<=m&&s[i+nxt[i]]==s[nxt[i]+1])++nxt[i];
		if(i+nxt[i]-1>r)mid=i,r=i+nxt[i]-1;
	}
}

inline void exkmp(){
	get_nxt();
	for(ll i=1,r=0,mid=0;i<=n;i++){
		if(i<=r)Nxt[i]=min(nxt[i-mid+1],r-i+1);
		while(i+Nxt[i]<=n&&c[i+Nxt[i]]==s[Nxt[i]+1])++Nxt[i];
		if(i+Nxt[i]-1>r)mid=i,r=i+Nxt[i]-1;
	}
}

int main(){
	cin>>c+1;
	cin>>s+1;
	n=strlen(c+1);
	m=strlen(s+1);
	exkmp();
	for(ll i=1;i<=m;i++)ans1^=i*(nxt[i]+1);
	for(ll i=1;i<=n;i++)ans2^=i*(Nxt[i]+1);
	printf("%lld\n%lld",ans1,ans2);
}

SA

我们用一个长得很像分治的东西,将长为n的后缀们一半一半拆解,然后比较,用最小的两个举例
ab bc cd da as
12 23 34 41 15
对他们基数排序,然后大的同理,贴个vector版伪代码
yuSmCQ.png

DAY3

树论

直径

树的所有直径叫交于一点

DAY1

链表

舞蹈链

跳跃表

posted @ 2021-02-01 09:53  蒟蒻丁  阅读(73)  评论(0编辑  收藏  举报