A. 组合

将两个字符之间建边,

问题转化为一笔画,也就是欧拉路问题。

所以直接用dfs压栈的方法解决,注意正确使用当前弧优化。

 

void dfs1(int x){
	for(int i=head[x];i;){
		if(vis[i>>1]){
			head[x]=nxt[i];
			i=nxt[i];
			continue;
		}
		vis[i>>1]=1;
		dfs1(to[i]);
		stack[++top]=w[i];
		i=head[x];//如果忽略这句话 复杂度仍为E^2
	}
}

 

 

 

 

B. 统计

实际上消失的逆序对只产生在进行排序的点之间。

所以可以预处理出每个点与它后面的多少个点产生了逆序关系。

对于一个点,问题是将它后面的且权值不大于它的点产生的逆序关系删去。

似乎是一个二维点对问题,显然可以用链表维护然后就AC了。

因为每个点只会删一次,所以用线段树维护区间最值,

当递归到叶子节点或区间最值已经超过当前要求时$return$。

因为删除次数是合法的,其复杂度为$O(nlogn)$

 

 

 

C. 点亮

因为数据保证随机,深度不会很大。

注意到实际上并不关注每个点对之间的关系,

当每层祖先中亮着的点与暗着的点的大小关系确定,每个点的点权是确定的。

所以可以状压每层祖先的状态。预处理出每个点对于每个状态的点权。

设$dp_{i,j,s}$表示点$i$的子树中有$j$个被点亮,祖先链的状态为$s$。

之后可以直接作树上背包转移。