【BZOJ1023】仙人掌图(仙人掌,动态规划)
题面
BZOJ
求仙人掌的直径(两点之间最短路径最大值)
题解
一开始看错题了,以为是求仙人掌中的最长路径。。。
后来发现看错题了一下就改过来了。。
首先和普通的仙人掌\(dp\)是一样的,
对于没有问题的圆圆边,直接做最长链的转移(同时更新\(ans\))
然后对于一个环,把它拎出来单独考虑
首先要对于这个环,计算能够贡献的答案,
然后再用环上的值更新环的最顶点
先考虑怎么更新,这个直接拿环上的点的\(dp\)值,再计算一下这两点之间的最短路(深度差和环大小减深度差的较小值),相加去更新\(dp\)值。
然后考虑一下如何贡献答案,
要求的相当于是\(max(f[i]+f[j]+dist(i,j))\)
而\(dist(i,j)=min(abs(dep[i]-dep[j]),circle\_size-abs(dep[i]-dep[j]))\)
发现维护一个单调队列,按照深度依次进栈,
这样子距离直接可以用深度做差,没有了绝对值
因为可以通过返祖边回去,因此把所有点按照顺序进两次队就可以了
第二次进队的时候给深度加上一个环大小再进队
然后如何保证是环上的最短路?
如果两个深度差已经大于环大小的一半了,那么最短路就不是这一条了
所以直接弹走队首就行了
#include
#include
#include
#include
#include
#include
#include
#include