1.P3780 [SDOI2017] 苹果树
首先要转化一下 \(t-h\le k\) 这个限制,考虑它的实际意义,就是我们取出所选的点中最深的一个,记为 \(x\) ,从根到它每个点都能免费领一个苹果。那我们反过来枚举 \(x\) ,尝试计算答案。按普通的方式进行 树形 dp 是 \(O(nk^2)\) 的,瓶颈在于我们合并两个背包时要 \(O(k^2)\) 地枚举。考虑换一种 dp。我们记 \(f_{x,j}\) 表示考虑了所有 \(dfn_u\le dfn_x\) 的点 \(u\) 得到的背包,其中根到 \(x\) 的点只能取 \(a_u-1\) 个。再计算 \(g_{x,j}\) 表示考虑了所有 \(dfn_u\geq dfn_x\) 的点 \(u\) 的得到的背包,其中根到 \(x\) 的点都不能取。统计答案时合并 \(f_x,g_x\) 即可。我们只关心 \(k\) 这一单点的值,所以合并是 \(O(k)\) 的。
现在的问题就是计算 \(f,g\) 。以 \(f\) 为例,我们记 \(f'_{x,j}\) 表示:考虑了 \(dfn_u<dfn_x+sz_x\) 的点,其中 \(fa_x\) 到根只能取 \(a_u-1\) 个,且 \(x\) 必取的背包。
那我们的转移是这样进行的:初始使 \(f'_x:=f_x\) 。依次遍历 \(x\) 的儿子 \(v\) ,\(f_v:=f'_x+\{a_v-1个苹果\},dfs(v),f'_x:=\max(f'_x,f'_v)\) 。
遍历完儿子之后,再在 \(x\) 强制取一个苹果,即 \(f'_{x,i}:=f'_{x,i-1}+z_x\) 。
考虑设计到的操作,对应取 max 是容易的;合并 \(f'_x\) 和 \(a_v-1\) 个苹果,相当于是做一个 \(G_x=xk+\max\limits_{x-y<a_v}F_y-yk\) ,单调队列即可。
总复杂度 \(O(nk)\) 。
这个 trick 做树上依赖性背包非常有用。
2.gym103860D Tree Partition
\(O(nk\log n)\) 是容易的:思考 \([l,r]\) 内的点形成一个联通块的条件:边-点=-1。
我们维护 \(c_l =[l,r]内的边数 +l\) ,则这个等于 \(边-点-(r-1)\) 。支持区间加,查询全局最大值的信息即可。
但是该题存在 \(O(nk+n\log n)\) 的做法:
考虑连边 \(0-1\) ,以 \(0\) 为根把边定向,让边全都指向 \(0\)。这样的话 \([l,r]\) 形成联通块的条件就是只有一条出边。
仍然是枚举 \(r\) 并考虑哪些 \(l\) 是合法的。把边 \(u\rightarrow v\) 分成两种:\(u<v\) ,\(u>v\) 。对于一类边,我们找到 \(u\le r<v\) 的边中 \(u\) 最大和次大的,分别记为 \(x_1,x_2\) 。则对于 \(l\in [x_1+1,r]\) ,我们要求只能有一条二类边;对于 \(l\in [x_2+1,x_1]\) ,不能有二类边。
现在考虑记 \(c_l\) 表示当前 \([l,r]\) 往外的二类边个数。则每次从 \(r-1\) 更新到 \(r\) 时,对于 \(u=r\) 的二类边,我们有 \(l\in (v,r]\) 的 \(c_l\) 都 +1。于是我们维护两个栈,分别记下 \(c_l=0\) 和 \(c_l=1\) 的位置,同时维护 dp 数组的前缀和。每次修改相当于是若干次弹栈,入栈操作,而每个点至多会被修改两次,所以是均摊 \(O(1)\) 的。
\(x_1,x_2\) 用 set 计算即可。复杂度 \(O(n\log n+nk)\) 。
3.gym103640D Daily Turnovers
先转化问题。为了方便,将所有变化取反。
计算前缀和 \(s_l\) 后,我们相当于对 \(l\in [0,n]\) ,求出 \(R_l\) 表示 \(l\) 右边第一个大于它的数,统计 \(\sum R_l\) ,再减掉 \(\tbinom{n+2}{2}\) 即可(设 \(s_{n+1}=+\infty\))。
现在考虑操作:在 \(p\) 上单点加 \(x\) ,反映到前缀和上就是后缀加 \(x\) 。思考哪些 \(R\) 会发生变化:维护 \([0,p)\) 的单调栈,则只有单调栈中的点的 \(R_i\) 会变化。再维护 \([p,n+1]\) 的单调栈,则变化了的 \(R_i\) 只可能变成这个栈中的点。具体的,令 \(F(j)\) 是前面的栈中 \(s_i\le j\) 的数个数,令 \(s'_i=s_i+x-1\) ,\(A\) 表示后面的栈,则减去会变化的 \(R_i\) ,再加上 \(\sum\limits_{i=1}^{top}A_i(F(s'_{A_i})-F(s'_{A_{i+1}}))\) 即可。
维护 BIT 以计算 \(F\) 。每次 \(p\) 改成 \(p+1\) 时,两个栈都有均摊 \(O(1)\) 次修改,对前一个栈修改是容易的: 加入 \(i\) 时,维护好 BIT,再找到最小的 \(s'_p\ge s_i\) ,让总和加上 \(p\) 。对后一个栈修改,就是加入一个数 \(x\) 时它一定是栈顶,我们加上 \((x-A_{top})F(s'_x)\) 即可。删除也是类似的,总复杂度 \(O(n\log n)\) 。
4.gym101309J Jungle Outpost
二分答案,做 \(p_i\rightarrow p_{i+mid}\) 的半平面交,看是否为空即可。
5.gym100520K Kabbalah for Two
判圆在凸包->判凸包往内缩了 \(R\) 之后,点是否在凸包内。
那我们二分答案之后,为了取的两个圆相离且在凸包内,就先求出往内缩 \(mid\) 后的凸包,再用旋转卡壳求最远点对,看距离是否 \(\geq 2mid\) 即可。
6.gym102896G Geometrical Combinatorics
直接计算困难,考虑将整个坐标系转 90 度并往内缩一下,杨辉三角就都在第一象限内了,也就是说 \((i,j)\) 处放的是 \(\tbinom{i+j}{i}\) 。那枚举 \(x\) 坐标,求 \(x=a\) 和三角形的交点,问题就转化成上指标求和。
7.P6925 [ICPC2016 WF] Spin Doctor
先转化成:求出 \(t_i=1\) 的点构成的凸包,取一个方向,用两条线卡住凸包,看中间有多少 \(t_i=0\) 的点。反过来看 \(t_i=0\) 的点在哪些时候不会有贡献:首先在凸包内肯定寄了;否则,求出对凸包的两条切线,则方向要取在这两个之间才不会有贡献。最后离散化,差分搞搞就能算出答案了。
问题是怎么求点对凸包的切线:如果点的横坐标比凸包上所有点都小或者大,那么可以把上下凸壳中分别二分。否则可以按照这个点的横坐标把凸包分成左右两半,再在两半中分别二分即可。(By Kubic)
8.gym102576D Clique
先思考不是环怎么做,显然需要所选区间的交非空,那我们一定是枚举一个点 \(x\),取所有 \(l\le x\le r\) 的区间。那现在是环了,仍然是枚举中间的一个点 \(x\),要求不包含 \(0\) 的区间都包含 \(x\) 。于是我们只需考虑跨过 \(0\) 和跨过 \(x\) 的区间。不难转化成:你有一些黑点和白点,然后要取一些点,让黑点右上角没有白点。从左往右取黑点,在 dp 状态记下来当前最低的黑点,那能取的白点个数就能顺便得到了。再用线段树优化这个 dp 即可。
9.gym103415B Sweeping Robots
对每次询问,取出 \([l,r]\) 中 \(c\) 最小的位置 \(k\) ,拆成 \([l,k)\) 和 \((k,r]\) 。
那就转化成了对笛卡尔树某个节点代表区间的某个前缀/后缀求答案。
前缀为例。现在对于点 \(x\) ,先求出 \(ls_x,rs_x\) 的结果,合并一下。然后考虑跨过 \(x\) 且在范围内的区间。那 \(l\) 端点是固定的;根据 \(s_r-s_x\) 和 \(s_x-s_l\) 的大小关系把 \(r\) 分成两个线段,每个线段中 \(r\) 的权值都能写成 \(ks_r+b\) 。那尝试用李超树维护这个东西。插入线段是容易的,关键是我们能否合并?发现也是可以的,因为两个子树的 \(s_x\) 是不交的。但查询时是前缀 max,于是和以往不同,我们还需要在李超树的每个节点记下来子树内的最大权值。
upd:似乎并不需要合并线段树。
复杂度 \(O(n\log^2n+q\log n)\) 。
10.gym102879J Japanese Knowledge
啊?有结论:恰好 \(k\) 个位置满足 \(x_i=a_i\) 的答案等于:\(a_{k+1},a_{k+2},\ldots,a_n\) 在 \(k=0\) 时的答案,构造双射易得。然后格路计数一下就好了。复杂度 \(O(n\log^2n)\) 。
11.P5853 [USACO19DEC] Tree Depth P
\(i\) 是 \(j\) 在笛卡尔树上的祖先,等同于 \(\max(a_i,\ldots,a_j)=a_i\) 。考虑算逆序对的过程,就是我们一个个插入数,第 \(i\) 个数对逆序对的贡献是 \(0\) 到 \(i-1\) 。你发现我们不止能每次从末尾插一个数,从前面插也是可以的,而且贡献是没有区别的。
现在不妨令 \(j<i\) ,我们就先插入 \(j+1\) 到 \(i\) ,再插入 \(j\) ,再插入剩下的数即可。你发现预处理前后缀的数组拼起来即可,复杂度 \(O(n^3)\) 。
12.P5044 [IOI2018] meetings 会议
省流版题意是令 \(f(x,y)\) 是 \(\min(x,y)\) 到 \(\max(x,y)\) 中 \(h_i\) 的 \(\max\) 值,然后每次询问要取一个点 \(x\) 使得 \(\sum\limits_{i=l}^r f(i,x)\) 尽量小。
考虑 \(l=1,r=n\) 的做法,就是建出笛卡尔树 dp ,我们有 \(dp_x=\min(dp_{ls}+h_x(sz_{rs}+1),dp_{rs}+h_x(sz_{ls}+1))\) 。
那现在还是建出全局的笛卡尔树,然后考虑先找到 \(l,r\) 的 \(lca\) 为 \(t\) ,拆成 \((t,r],[l,t)\) 分别询问再拼起来。那现在要考虑的区间一定是一个节点代表区间的前缀/后缀。
以前缀为例,我们从下往上计算,令 \(f(x,y)\) 表示 \([L_x,y]\) 的答案,你发现对于 \(y<x\) ,\(f(x,y)=f(ls_x,y)\) ,对于 \(y\geq x\) ,\(f(x,y)=\min(yk+b,f(rs_x,y)+t)\) ,其中 \(k,b,t\) 是可以计算出的三个常数。那我们就完全让 \(f(x)\) 直接继承子树的信息,即直接用数组 \(g_y\) 记下当前 \(f(x,y)\) 的值。现在涉及到的操作就是区间加和区间对 \(kx+b\) 求 \(\min\) ,看似需要高级数据结构,实际上你发现,这里是满足 \(k=h_x\) 的,而容易发现 \(g_p\le g_{p-1}+k\) 。所以 \(yk+b\le g_y+t\) 的 \(y\) 一定是一段前缀!于是线段树二分找出分界点后,直接打覆盖 tag 即可。复杂度 \(O(n\log n)\) 。
13.CF1718D Permutation for Burenka
判定 \(a\) 合法:建出的笛卡尔树与给定排列 \(p\) 的笛卡尔树相同。
那首先我们可以对每个 \(a_i=0\) 的点求出它能填的数的上下界:上界是它最近的已填祖先的 \(a_x-1\) ;下界是它子树中已填的最大 \(a_x+1\) 。你发现如果保证上下界了之后,乱填也能调整成合法的,因为如果出现一对 \(a_x<a_v\) ,那由于我们已经限制了空位和已填位之间的关系, \(x\) 和 \(v\) 一定原来都是空位。然后你发现交换它们俩一定是不会破界的,第一 \(R_x=R_v\),第二 \(L_x\geq L_v\) ,调整后更容易满足。
现在就是有一些区间和一些点,区间要和点做匹配,区间要包含点才能匹配上。
可以贪心的做,就是从把区间按 \(r\) 排序后,每次取在区间内的最小点即可。
但现在点集里有一个是不固定的,怎么办呢。
我们仍然搞一遍原来的贪心,然后会出现一个区间,它内部匹配不上点。此时我们就一定需要 \(d\le r\) 。再往后若又有一个不合法区间,肯定就寄了,因为 \(d\) 只能满足前一个区间,扫到这里就还是不合法了。
倒着扫一遍又可以得到 \(d\) 的下界,这两个界就是充分的了。证明比较复杂。
复杂度 \(O(n\log n+q)\) ,瓶颈在于贪心时找区间内最小点。
14.qoj2570 Maximal Subsequence
设 \(f_i\) 是以 \(i\) 为结尾的最长上升子序列,则我们能建出最小割:若 \(i<j,a_i<a_j\) 且 \(f_i+1=f_j\) 则连边 \((i+n,j,+\infty)\) ;\(f_i=1\) 则连边 \((S,i,+infty)\) ,\(f_i=\max f_j\) 则连边 \((i+n,T,+infty)\) 。再连边 \((i,i+n,1)\) 。最小割=最大流,发现问题转化成:在序列中取尽量多的 LIS。这等于杨表最后一列的长度,可以 \(O(n\sqrt{n}log n)\) ;但事实上能直接模拟这个流的过程,可以证明我们先尝试流编号更小的点就不会退流。具体就是走到 \(u\) 时先弹掉下一层 \(<u\) 的点;再依次枚举下一层每个点 \(v\),如果 \(a_u\geq a_v\) 就结束,否则尝试流 \(v\) ,如果 \(v\) 走不到 \(T\) 就直接删掉 \(v\) 即可,否则结束。这样就是 \(O(n\log n)\) 了,瓶颈在求 \(f\) 。
15.qoj5246 Nawiasowe podziały [B]
首先设 \((\) 是 \(+1\) ,\()\) 是 \(-1\) ,令 \(s_i\) 是括号权值的前缀和。
那 \((l,r]\) 合法等同于 \(s_l=s_r=\min\limits_{i=l}^r s_i\) 。
场上的想法就是,我们可以进行染色,使得合法的 \((l,r]\) 满足 \(col_l=col_r\) 。
先 wqs 二分,发现具有决策单调性,然后发现 \(val(l,r)\) 似乎没有办法直接求,但移动指针,\(O(1)\) 更新 \(val(l,r)\) 是容易的,于是通过分治套分治做到 \(O(n\log^2n\log V)\) 。场上就止步于此了。
原因在于我把条件强化了,事实上可以发现对于两个颜色,它们要么不交要么包含。
事实上,注意到 \(s_l=s_r=\min\limits_{i=l}^r s_i\) 是和笛卡尔树相关的,尝试建小根笛卡尔树。但普通笛卡尔树是一个点代表一个元素,这样在区间有多个最小值的时候只能随便选一个,会丢失一些信息。于是使用广义的笛卡尔树,一个点会维护这个区间所有最值的位置。
这样建之后,你发现每个颜色都对应了一个节点。那预处理欧拉序的 ST 表后, \(val(l,r)\) 就能 \(O(1)\) 求了,利用 deque+二分的手法就能做到 \(O(n\log n\log V)\) 。
但想优化到单 log 的话,考虑 wqs 二分基本是无法避免的,于是只能避免决策单调性。但 \(val(l,r)\) 似乎也无法直接拆成一个简单的形式供我们优化,于是想到直接进行树形 dp,设 \(dp_u\) 是在 \(u\) 内砍至少一刀,能得到的最小权值,注意这里为了方便,可以在节点的左边/右边切一刀。令 \(n_u\) 是 \(u\) 儿子的个数。令 \(t_u\) 是 \(u\) 子树内所有点的 \(\sum \tbinom{n_u}{2}\) 之和。
那计算当前 \(dp_u\) 的值,就先求出儿子的 dp 值。不妨令儿子的编号为 \(1,2,\dots,n_u\) ,再求出 \(s_i\) 表示儿子的 \(t_i\) 前缀和,令 \(f_i\) 为:从左往右考虑到第 \(i\) 个儿子,且儿子 \(i\) 一定被劈的最小代价,有 \(f_j=\min\limits_{i<j} f_i+\tbinom{j-i}{2}+s_{j-1}-s_i\) 。斜率优化 dp 即可。
这样总复杂度就是 \(O(n\log V)\) 的了。
16.AGC041F Histogram Rooks
考虑容斥,指定一些点不会被覆盖,我们称这些点是黑格子。还是建出笛卡尔树从下往上 dp ,但是你发现这个会出问题:如果之前某列上有一些格子还能填,结果往上走时这一列突然出现了黑格子,那这个系数就不好算了。。
那你考虑我们每次加入新的一列时就指定:这个列中是否有黑格子。这样就好算了,把子树内被指定的列数记下来计算即可。
但你发现还是有问题,因为你指定了一列有黑格子,但算到最后可能并没有黑格子,这怎么办呢。那我们再次容斥,钦定一些指定有黑格子的列其实没有黑格子。
那现在考虑计算系数,先加入中间一列,然后再在当前节点进行填充,设有 \(t\) 行 \(x\) 列。那首先这 \(t\) 行的系数都是相同的,我们只需要计算 \(1\) 行的系数。第一个 case 是没有填黑格子,那方案数是 \(2^{x-i}\) ,其中 \(i\) 是指定有黑格子的列个数。第二个 case 是填了至少一个黑格子,方案数是 \(\sum\limits_{k=1}^{i-j}(-1)^k\tbinom{i-j}{k}=-[i=j]\) ,其中 \(j\) 是其实没有黑格子的列个数。注意到系数只和 \(i,j\) 是否相等有关,在状态中记下 \(i,[i=j]\) 即可,复杂度 \(O(n^2)\) 。
17.AGC336G 16 Integers
考虑把这个序列当成是一条点数为 \(n+1\) 的路径,其中第 \(i\) 个点是 \(a_ia_{i+1}a_{i+2}\) 拼成的点。 \(x_{a,b,c,d}\) 相当于是这条路径中 \(abc\) 到 \(bcd\) 的边数。
于是问题转化成:从任意一点出发,任意一点结束,经过所有边的方案数。
我们有 BEST 定理:从 \(s\) 为起点和终点的欧拉回路个数是 \(A\prod (out_i-1)!\) ,其中 \(A\) 是以 \(s\) 为根的内向生成树个数。
回到这个题上,有两种可能的情况:第一种是所有点都满足出边=入边,那枚举起点,直接套 BEST 定理即可;第二种是有个点 \(x\) 满足出边=入边+1,另一个点 \(y\) 满足入边=出边+1,剩下的平衡,那建一个虚点 \(s\) 并连边 \((s,x),(y,s)\) ,再用 BEST 定理计算从 \(s\) 出发的回路个数即可。
注意最后要除以 \(\prod x_{a,b,c,d}!\) ,因为上面的计算过程把一些相同的边当成不同的了。
令 \(N=2^k\) ,时间复杂度 \(O((\sum x_{a,b,c,d})+N^4)\) , \(k\) 在本题中 \(=3\) 。上面是我赛时的做法。
考虑优化:第一,我们在第一种情况中无需枚举起点,我们先挖掉一条边变成第二种,再让答案乘上 \(n\) 即可。这样就是 \(O(N^3)\) 的,可以开到 \(k=8/9\) 了 。
第二,考虑加快行列式的计算,注意到我们的矩阵中只有 \(O(N)\) 个位置是非零的 ,考虑以此优化计算。我们知道矩阵的特征多项式是 \(f(x)=|Ix-A|\) 。那么若能求出这个特征多项式, \((-1)^Nf(0)\) 就是想要的答案。但是怎么计算特征多项式呢?不好计算。但现在,假设我们能快速求出矩阵的最小多项式。
如果最小多项式项数 \(=N\) ,说明这就是特征多项式;否则随机一个对角矩阵 \(D\) ,有结论:\(AD\) 的最小多项式项数大概率 \(=N\) 。计算 \(\frac{det(AD)}{det(D)}\) 即可。
现在问题变成计算最小多项式。我们要知道,如果已知最小多项式 \(\le n\) ,那通过序列的前 \(2n\) 项即可唯一确定最小多项式。我们分三步进行:
-
算序列 \(a\) 的最小多项式,即找到 \(c_i\) 使得对任意 \(x\) 有 \(\sum c_ia_{x+i}=0\) 。使用 Berlekamp-Massey 算法即可做到 \(O(n^2)\)。
-
算一组向量 \(b\) 的最小多项式,其中向量有 \(m\) 维。结论:随机 \(m\) 维向量 \(u\) ,找到序列 \(p_i=u^Tb_i\) 的最小多项式即可。复杂度 \(O(n(n+m))\) 。
-
算矩阵 \(A\) 的最小多项式。仍然是随机 \(n\) 维向量 \(u\) ,令 \(b_i=A^iu\) 。再找到 \(b\) 的最小多项式即可。复杂度 \(O(n(n+T))\) ,瓶颈在于算 \(b\) ,我们要进行 \(n\) 次向量乘矩阵的操作。其中 \(T\) 是 \(A\) 中非零元素的个数。
结合上面的步骤,我们得到 \(det(A)\) 可以在 \(O(n(n+T))\) 的时间求出。于是原题可以优化到 \(O(N^2)\) 的复杂度!可以开到 \(k=12/13\) 。
18.P6929 [ICPC2017 WF] Airport Construction
思路不难,就是我们选取的线段一定穿过了至少两个顶点。\(O(n^2)\) 枚举两个端点得到一条直线,然后找到多边形所有边和这条直线的交点,把直线拆成 \(O(n)\) 段,判断每段是否合法即可。
但代码实现很有技巧:对于多边形的一条边,它影响这条直线有两种可能:第一,一个端点恰好在线上,另一个不在。根据方向,我们把这个交点的权值设为 \(1/-1\) ;否则,若这条边和直线严格相交,则把交点的权值设为 \(2/-2\) 。处理完后从左往右扫这个直线,看当前权值的前缀和是否等于0,即可判断当前这一段是否还在多边形内了,它的巧妙之处就在于,考虑 \(A_iA_{i+1}A_{i+2}\) ,若 \(A_{i+1}\) 在线上,则:若 \(A_i,A_{i+2}\) 在同侧,则 \(A_{i+1}\) 被一个 \(+1\) ,一个 \(-1\) 抵消了;若在异侧,则根据方向会变成 \(+2/-2\) ,这就和严格相交的情况相同了。这就避免了复杂的讨论。
19.P9265 [PA 2022] Chodzenie po linie
首先排除掉无解的部分,根据 \(p_{1}\) 到 \(p_x\) 重排后是否为 \([1,x]\) ,把排列分成若干段分别处理即可。
考虑手玩一下 bfs,我们一层一层的看当前能走到那些点。第一步后 \((i,p_i)\) 的左上角右下角都走到了;而右上角和左下角是互不影响的,于是只看左下角。令 \(a_i\) 是 \(p\) 的后缀 min ,\(b_i\) 是 \(p_j\geq i\) 的点中最小的 \(j\) 。 你发现我们相当于是从 \((i,p_i)\) 出发,每步从 \((x,y)\) 走到 \((b_y,a_x)\) ,并沿途计算 \(S(x-1,y-1)\) 的和(\(S(x,y)=\sum\limits_{i=1}^x [i\le y]\)) 。
这样就得到了一个 \(O(n^2)\) 的做法,考虑优化。猜测经过的点数是不多的,事实上可以证明是 \(O(n\sqrt{n})\) 的,原因是你发现每条一步, 要么 \(x+y\) 减少 \(\sqrt{n}\) ,要么 \(|x-q_y|\le \sqrt{n}\) 。这样记忆化一下就是 \(O(n\sqrt{n}\log n)\) 的,瓶颈在于求 \(S\) ;事实上离线下来,用分块即可平衡至 \(O(n\sqrt{n})\) 。
20.关路灯,但是 O(n)
考虑经典的 dp 状态所以只能两维是因为:如果我们走到一端,不记另一端关了多少灯,会导致重复计算。
考虑我们每一步其实是在干这样一件事:令 \(E(x)\) 表示我们认为的 \(x\) 最小可能被关的时间,则每走一步后计算 \(E(x)\) 的增量,最后 \(E(x)\) 全都固定成真实值了,也就算出答案了。传统的做法就是认为,对于还未关的灯, \(E(x)\) 等于当前的时刻 \(T\)。
但我们可以这样设计:设 \(E(x)=T+|x-X|\) ,其中 \(X\) 是你现在所在的位置。
现在再看我们走的过程,比如说从起点左边的 \(i\) ,走到起点右边的 \(j\)。考察 \(E\) 的变化,你发现右端的这些点 \(E\) 无论状态如何,都是不变的!改变的只有左端未被推的点,改变量即 \(2s_{i-1}(x_j-x_i)\) ,其中 \(s\) 是功率的前缀和。
这样就容易设计 dp 了,我们只需设 \(dp_i\) 表示走到 \(i\) 时的最小势能和,输出 \(dp_1+\sum |x_c-x_i|s_i\) 即可,原因是走到 \(1\) 时即使右部灯没有关完,我们走过去关了也不影响势能。
发现可以斜率优化,最后复杂度 \(O(n)\) 。注意转移顺序,类似于 dij ,先用更小的 \(dp_i\) 来转移,具体维护两个指针即可。而且也得维护两个凸包。
21.gym104022J The Answer!
显然有 \(\gcd(a^x-1,a^y-1)=a^{\gcd(x,y)}-1\) ,而 \(\gcd(F_x,F_y)=F_{gcd(x,y)}\) 。
于是原问题可以转化成计算 \(\frac{a^{F_{xk}}-1}{a^{F_x}-1}\) 。模数是合数,拆成 \(\prod p^k\) 分别计算再 CRT 合并。
先分别计算分子分母,注意这里要用扩展欧拉定理处理一下 \(F_x\) 和 \(F_{xk}\)。(\(\varphi(p^k)=(p-1)p^{k-1}\)) 。
最希望看到的是分母 \(Y\) 与 \(p\) 互质,这样直接求逆就好了。但很多时候并不互质。考虑求出最大的 \(t\) 满足 \(p^t|Y\) ,然后我们再计算一遍分子和分母模 \(p^{k+t}\) 的结果。这样分子和分母就都是 \(p^t\) 的倍数了,同时除掉 \(p^t\) ,这样分母就和 \(p\) 互质了。
但 \(t\) 也有可能很大。如果 \(t\geq k\) ,我们就无法直接找出 \(t\) 了。但注意到,\(\frac{a^{F_{xk}}-1}{a^{F_x}-1}=\sum\limits_{j=0}^{\frac{F_{xk}}{F_x}-1}a^{jF_{x}}\) 。而如果 \(t\geq k\) ,\(a^{F_x}\) 在模意义下 \(=1\) ,于是就变成了求 \(\frac{F_{xk}}{F_x}\) !
而观察发现 \(F_0,F_n,F_{2n},\dots\) 也是二阶递推,具体的有 \(F_{ni}=q_nF_{n(i-1)}+F_{n(i-2)}\) ,其中 \(q_n=q_{n-1}+q_{n-2}\) ,\(q_0=2,q_1=1\) 。这就做完了,复杂度瓶颈在于把模数质因数分解。
22 GDKOI
D1T1:先任意跑出一组最大匹配,然后考虑调整,调整就是在残量网络上找到一个简单环,满足黑边个数为奇数。直接在可能的环中取最小的,可以证明一定是简单环,否则能拆成两个更小的环。复杂度 \(O(nm)\) 。
D1T2:考虑倒着枚举操作。如果当前的操作是询问,则没有影响;如果是染色,它会使得后面的一些询问操作权值发生变化。具体的用 ODT 来维护,则我们当前会有均摊 \(O(1)\) 次操作,就是把一个区间 \([l,r]\) 的颜色从 \(col\) 变成 \(i\) ,注意这里的”颜色“是指它被染的时刻。
那对于 \(i\in [l,col-1]\) ,如果是询问操作,我们会把 \(a_i\) 加上 \(|[x,y]\cap[l_i,r_i]|\) 。
然后我们处理左端点是 \(i\) 的查询,就是算 \(a\) 的区间和。
施加容斥后就转化成以下问题:给定序列 \(b_i\) ,修改操作是让一个区间的 \(a_i\) 加上 \(\min(b_i,z)\) , 查询操作就是区间和。发现 poly 不能,但分块容易,复杂度 \(O(n\sqrt{n})\) 。
D1T3:太复杂了咕咕咕。
23.gym103439D LIS Counting
发现 \(LIS=n ,LDS=m\) 的条件等价于杨表有 \(n\) 列 \(m\) 行。我们知道,(杨表,记录表) 和 可能的排列形成双射,但这里要求 \(p_{X}=Y\) ,就不能直接套记录表,考虑设计另一种表:设 \(B_{i,j}=x\) 满足以 \(x\) 为末尾,\(LIS=i,LDS=j\) 。设杨表为 \(A\) ,我们能注意到 \(p_{A_{i,j}}=B_{i,j}\) !而 \((A,B)\) 和可能的排列仍然形成双射。只要算出 \(g_{x,y,i}\) 表示 \(A_{x,y}=i\) 的可能 \(A\) 个数 ,然后计算 \(\sum\limits_{x,y} g_{x,y,Y}g_{n+1-x,y,X}\) 即可。现在设 \(dp_{S}\) 表示我们从按值小到大给 \(A\) 的每个格子染黑,当前染黑的格子的轮廓线是 \(S\) 的方案数,算出 \(dp\) 就容易算 \(g\) 了。注意到
状态数 \(\tbinom{n+m}{n}\) ,直接转移即可。复杂度 \(O(\tbinom{n+m}{m})\) 。
标签:2024.1,log,复杂度,即可,考虑,我们,dp From: https://www.cnblogs.com/cwhfy/p/17966595