CF做题记录
CF1550
观察样例可得:满足条件当且仅当初始序列满足\(a_i=i\) ,然后选择\(\lfloor\dfrac n2\rfloor\) 个位置\(-x\) ,剩下位置\(+x\) ,其中\(x\) 为任意非零整数。对这个计数即可。枚举最靠前的减的位置然后枚举最靠后的加的位置计数,然后用上指标求和优化即可。
二分答案然后用状态压缩dp判断。设\(f_S\) 表示已经完成集合\(S\) 中的要求时最靠前的位置,转移十分显然。需要预处理长度为\(x\) 且全是小写字母\(v\) 的字符串能放的位置。
考虑到可达性,将该问题转化为一个图论问题:在\(u,v\) 之间连接边权为\(||a_u-a_v|-d|\) 的边表示偏移量大小。这样如果存在一条从\(s\) 到\(i\) 的路径满足经过边权最大值\(f(i)\) 不超过\(k\) 则答案为Yes,否则为No。
根据经典结论:任意两点间的任意路径上边权最大值的最小值等于在最小生成树简单路径上的边权最大值。因此只要建出最小生成树即可。这里考虑使用boruvka算法。现在关键在于如何对一个联通块求出以它连向其它连通块的边中权值最小的那一条边。枚举联通块内的每个点\(u\) ,要求最小的\(v\) 使边权尽量小,只要\(a_v\) 距离\(a_u\pm d\) 尽量小。使用set lower_bound即可。
CF1592
CF1592C Bakry and Partitioning
首先题目不要看错!!接下来可以转化为是否存在三个连通块使得其内部点的点权异或和均为sum,其中sum表示所有点权异或和。只有两种情况:1.存在\(v_1,v_2\) 满足其子树内均存在某个子树异或和为sum 2.存在\(u,v\) 满足\(u\) 是\(v\) 的祖先且\(u\) 子树异或和为0,\(v\) 的为sum。dfs判断即可。
需要找的其实是最大边权。根据询问次数考虑二分,关键是如何保证每次二分时点集都是联通的。直接在欧拉序上二分即可。(求欧拉序不要写错)
与大于异或意味着存在某一二进制位满足更高的位都一致但是当前位与是1而异或是0,即这一位全部是1且有偶数个,因此区间长度一定为偶数。考虑前面的位相同,分0/1两种情况:1.存在0且有偶数个1。2.全是1且有奇数个。第二类情况显然不会出现,而对于第一种情况,可以排除掉存在0这一条件,因为这样答案不会变大。于是我们枚举当前位,对数列做前缀异或和,那么前面的位相同意味着值相等。用vector存下值为i,下标奇偶性为0/1的下标,然后扫一遍同时得到最大值即可。
CF1580
CF1580B Mathematics Curriculum
所有包含这个数的区间中,不同的最大值的个数恰有 \(m\) 个意味着\([1,x]\) 的后缀最大值和\([x,n]\) 的前缀最大值个数,而这就是这个数在排列建成的笛卡尔树中的深度(注意这个转化)。于是记\(f_{i,j,k}\) 表示长度为\(i\) 的排列中深度为\(j\) 的点有\(k\) 个的方案数,dp即可。
根号分治。首先先进行差分答案数组\(s\) ,最后再前缀和回来。对于\(x_i+y_i>B\) 的点暴力做。对于\(x_i+y_i\le B\) 的点,\(x_i+y_i\) 的取值只有\(B\) 种,因此可以用数组\(D_i\) 维护\(s\) 数组中下标除以\(x_i+y_i\) 余\(i\) 位置的答案,那么每次的贡献就是区间加,仍然可以转化为差分+前缀和。复杂度为\(\mathcal O(n\sqrt n)\) 。
CF1601
与运算的一个性质是每个二进制位相互独立。对每个二进制位进行考虑,假设该位总共有\(d\) 个1,那么容易发现\(k\) 合法当且仅当\(k\mid d\) 。因此最终答案就是所有\(d\) 的最大公约数的约数。
可以直接dp,然后使用数据结构进行优化。也可以直接建图后跑01bfs,但是需要线段树优化建边。
使用如下策略必然能够达到最优解:对\(b\) 的每个元素\(b_i\) 放到满足放入后新产生的逆序对(即前面比\(b_i\) 小的数的个数和后面比\(b_i\) 大的数的个数)最少的位置。
考虑证明。关键在于每个\(b_i\) 都能取到最小值,无关插入顺序。我们考虑将\(b\) 从小到大地排序依次插入。令\(c_i\) 表示在\(i\) 位置后插入当前的\(b\) 时新产生的逆序对。那么对于\(a_i\) ,假设它从原来的大于\(b\) 变成了小于\(b\) ,那么相当于对于\(x\in[1,i],c_x\leftarrow c_x+1\) ,对于\(x\in[i+1,n],c_x\leftarrow c_x-1\) 。而初始时\(c=[1,2,3,\cdots,n]\) ,每次相当于前缀加后缀减,那么最小值的位置一定是单调右移,即\(b\) 越大,其最优位置越靠右。这样不仅每个\(b\) 均能取到最小值,同时\(b\) 内部也不会新产生任何逆序对,因此这样一定是最优的。具体实现时使用线段树进行维护即可。
结论:按照第一关键字为\(\max(s,a)\) ,第二关键字为\(s\) ,第三关键字为\(a\) 从小到大进行排序。依次考虑是否能选即能获得最大值。考虑证明。对于两个元素\((s_1,a_1),(s_2,a_2)\) 。分三种情况:1.\(s_1<a_1,s_2<a_2\) 2.\(s_1>a_1,s_2<a_2\) 3.\(s_1>a_1,s_2>a_2\) 分别讨论即可。
令\(mn_i=\min_{t=i}^{i+k}a_t\) ,那么对于区间\([l,r]\) :
按照\(\bmod k\) 进行分类后,可以将问题转化为\(k=1\) 的情况。此时从右到左进行考虑,对于每个位置能做出贡献的一定是一个单调递减的序列。维护这个单调栈,然后维护两个数组\(go_{i,j},s_{i,j}\) 分别表示从\(i\) 出发跳\(2^j\) 步后到达的点,以及经过的数的和。每次询问时倍增即可。
CF1603
当\(x>y\) 时答案可以为\(x+y\) 。否则考虑令\(n=ax+b,y=cn+b\) ,然后将\(n\) 代入得\(y=acx+b(c+1)\) 。大胆令\(b(c+1)=y\bmod x\) ,然后由于\(2\mid y\bmod x\) ,因此再大胆令\(c+1=2\) ,则\(b=\dfrac{y\bmod x}2\) ,发现满足\(b\in[0,x-1],b\in[0,y-1]\) ,因此答案即为\(y-\dfrac{y\bmod x}2\) 。
先考虑对一个序列求出其“extreme value”。考虑从后往前找到第一个\(i\) 满足\(a_i>a_{i+1}\) 。容易发现至少需要将\(a_i\) 分为\(\lceil\dfrac{a_i}{a_{i+1}}\rceil\) 尽量平均的份才能满足条件。考虑到需要让操作总数尽量少,因此需要让操作后最小的数尽量大,因此恰好将\(a_i\) 分为\(\lceil\dfrac{a_i}{a_{i+1}}\rceil\) 就是最优的。令\(a_i=\sum_{t=1}^kb_t(b_1\le b_2\le\cdots\le b_k)\) ,则\(k=\lceil\dfrac{a_i}{a_{i+1}}\rceil,b_1=\lfloor\dfrac{a_i}{k}\rfloor,b_k=\lceil\dfrac{a_i}{k}\rceil\) 。然后\(ans\leftarrow k-1\) ,将\(a_i\) 替换为\(b_1\) ,继续进行上述操作即可。最终的\(ans\) 就是答案。
对于原问题,直接做可以做到\(\mathcal O(n^2)\) 。优化考虑dp。令\(f_{i,x}\) 表示有多少\(j\) 满足\(j\ge i\) 且从\(j\) 开始操作到\(i\) 时该序列的第一个数为\(x\) 。转移十分trivial,按照上面的过程模拟即可,即\(f_{i+1,x}\rightarrow f_{i,\lfloor\frac{a_i}{\lceil\frac{a_i}{x}\rceil}\rfloor}\) 。可以发现对于状态\(f_{i+1,x}\) ,其对答案的贡献为\(i\cdot f_{i+1,x}\cdot(\lceil\frac{a_i}{x}\rceil-1)\) 。容易发现状态数事实上仅有\(\mathcal O(\sqrt{10^5}n)\) 个,直接dp即可。
考虑\(c\) 到底是什么。
\(S\) 可以线性预处理。然后我们对于每个\(r\in[1,n]\) 求出每个\(\lfloor \dfrac rd\rfloor\) 处的\(S\) 的前缀和,那么\(c(l,r)\) 就可以做到\(\mathcal O(1)\) 求出了。
对于\(f(n,k)\) 我们显然有转移方程:\(f(n,k)=\min_{i=1}^n(f(i-1,k-1)+c(i,n))\) 。直接做是\(\mathcal O(n^3)\) 的,考虑优化。首先是减少状态数。可以发现\(c(l,r)\ge r-l+1\) ,而当\(r\le 2l-1\) 时\(c(l,r)\) 取得下界。对于最小的\(k\) 满足\(2^k>n\) ,我们考虑取\(x_i=2^{i-1}-1\) ,在这样的情况下所有的\(c\) 同时取得最小值,显然\(f(n,k)\) 是最小的,答案就是\(n\) ,不需要dp。同时当\(k\) 更大时答案仍然能取得最小值\(n\) 。
现在只需要考虑\(k\le \log_2 n\) 的\(k\) 了,但复杂度仍然是\(\mathcal O(n^2\log n)\) 的。考虑优化转移。观察转移方程可以发现其形式非常经典,于是我们可以大胆猜想\(c\) 满足四边形不等式。事实上的确如此,按照定义即可证明。那么令\(g(n,k)\) 表示\(f(n,k)\) 的最优转移位置,就有\(g(n-1,k)\le g(n,k)\le g(n+1,k)\) ,即决策点单调不左移。因此采用分治的做法就可以对于一个\(k\) 在\(\mathcal O(n\log n)\) 的复杂度求出所有的\(f(n,k)\) 了。
总时间复杂度\(\mathcal O(n\sqrt n+n\log^2 n+t)\) 。
ps:即使每次转移时暴力根号地计算\(c(l,r)\) ,不知道为何也可以通过。
CF1588
首先通过二分可以在30次询问下找到 \(k\) 的值。接下来不妨设 \(k-j+1=m,j-i=n\) ,那么我们再询问 \([1,k],[1,k-1]\) 这两个区间,设其答案分别为 \(p,q\) ,那么可以得到 \(\dbinom n2+\dbinom m2=p,\dbinom n2+\dbinom {m-1}2=q\) ,联立即可解出 \(n,m\) ,从而求出 \(i,j\) 。
容易发现一个序列 \(a_1,a_2,\cdots ,a_n\) 是winning的当且仅当 \(a_1+a_2+\cdots +a_n=0\) 且对于所有奇数 \(p\) 都有\(a_1-a_2+a_3-\cdots+a_p\ge 0\) ,对于所有偶数 \(q\) 都有 \(-a_1+a_2-a_3+\cdots+a_q\ge 0\) 。证明可以考虑 \(a_1\) 一定和 \(a_2\) 一起被减去,然后以此类推下去。
对这个东西计数方法很多,在这里介绍一种比较简便的做法,即枚举右端点,同时使用两个set分别维护当前情况下的所有奇左端点和偶左端点。右端点向右移动一步时,加上此位置的贡献,然后排除掉不满足前缀交错和不小于0的左端点,因为在后面的右端点中它们也不可能合法了。最后统计其中0的个数,贡献到答案中即可。
可以发现点 \(P\) 到线段 \(AB\) 的距离是 \(P\) 到射线 \(AB\) 和到射线 \(BA\) 距离的较大值。于是我们可以枚举点 \(P\) ,然后 \(\mathcal O(n)\) 求出所有的 \(Q\) 使得所有点距离射线 \(PQ\) 的距离都不大于 \(R\) 。最后 \(\mathcal O(n^2)\) 枚举统计答案即可。
考虑以 \(A\) 为半径为 \(R\) 圆心作圆,那么如果某个点在圆内,那么其到射线的距离一定不大于 \(R\) 。否则我们考虑点 \(C\) ,如下:
如图所示,\(CD,CE\) 为切线, \(F\) 为线段 \(CE\) 上一点满足 \(\ang FAC=\ang FCA\) ,\(CG\) 为 \(AF\) 垂线, \(F',G'\) 类似。
根据三角形全等可以发现 \(CG=CG'=R\) 。因此 \(C\) 距离射线 \(AB\) 不大于 \(R\) 当且仅当射线 \(AB\) 在 \(\ang G'AG\) 以内。注意 \(\ang G'AG\) 是一个定角,于是可以将所有点对应的角求交,然后看哪些射线在该公共角内即可。
CF1588F Jumping Through the Array
请参见wlzhouzhuan的博客。