SPOJ10707 COT2 - Count on a tree II 【树上莫队】
题目分析:
考虑欧拉序,这里的欧拉序与ETT欧拉序的定义相同而与倍增LCA不同。然后不妨对于询问$u$与$v$让$dfsin[u] \leq dfsin[v]$,这样对于u和v不在一条路径上,它们可以改成询问$dfsin[u]$到$dfsin[v]$。否则改成$dfsout[u]$到$dfsin[v]$,并加上LCA上的影响,如果在询问过程中没有加入就加入,否则删除。
代码:
- #include<bits/stdc++.h>
- using namespace std;
- const int maxn = ;
- const int maxm = ;
- const int srt = ;
- int n,m;
- int a[maxn],tb[maxn],arr[maxn],fa[maxn],pm[maxm],ans[maxm];
- int dfsin[maxn],dfsout[maxn],Num,pre[maxn];
- vector <int> g[maxn];
- vector <pair<int,int> > Qy[maxn];
- struct query{int l,r,bel,rem;}Q[maxm];
- int found(int x){
- int rx = x; while(pre[rx] != rx) rx = pre[rx];
- while(pre[x] != rx){int tmp = pre[x]; pre[x] = rx; x = tmp;}
- return rx;
- }
- void dfs(int now,int ff){
- arr[now] = ;pm[++Num] = now;fa[now] = ff;
- dfsin[now] = Num;
- for(int i=;i<Qy[now].size();i++){
- if(!arr[Qy[now][i].first])continue;
- Q[Qy[now][i].second].bel=found(Qy[now][i].first);
- }
- for(int i=;i<g[now].size();i++){
- if(g[now][i]==fa[now])continue;
- dfs(g[now][i],now);
- }
- pre[now] = fa[now];
- pm[++Num] = now; dfsout[now] = Num;
- }
- void read(){
- scanf("%d%d",&n,&m);
- for(int i=;i<=n;i++) scanf("%d",&a[i]),tb[i] = a[i];
- sort(tb+,tb+n+); int num=unique(tb+,tb+n+)-tb-;
- for(int i=;i<=n;i++) a[i]=lower_bound(tb+,tb+num+,a[i])-tb;
- memset(tb,,sizeof(tb));
- for(int i=;i<n;i++){
- int u,v; scanf("%d%d",&u,&v);
- g[u].push_back(v); g[v].push_back(u);
- }
- for(int i=;i<=m;i++) scanf("%d%d",&Q[i].l,&Q[i].r);
- }
- int cmp(query alpha,query beta){
- if(alpha.l/srt == beta.l/srt) return alpha.r < beta.r;
- else return alpha.l/srt < beta.l/srt;
- }
- void init(){
- for(int i=;i<=m;i++){
- Qy[Q[i].l].push_back(make_pair(Q[i].r,i));
- Qy[Q[i].r].push_back(make_pair(Q[i].l,i));
- }
- for(int i=;i<=n;i++) pre[i] = i;
- dfs(,);
- for(int i=;i<=m;i++){
- if(dfsin[Q[i].l] > dfsin[Q[i].r]) swap(Q[i].l,Q[i].r);
- if(Q[i].bel == Q[i].l || Q[i].bel == Q[i].r){
- Q[i].l = dfsin[Q[i].l]; Q[i].r = dfsin[Q[i].r];
- Q[i].bel = i; Q[i].rem = ;
- }else{
- Q[i].l = dfsout[Q[i].l]; Q[i].r = dfsin[Q[i].r];
- Q[i].rem = Q[i].bel; Q[i].bel = i;
- }
- }
- sort(Q+,Q+m+,cmp);
- }
- void work(){
- int L = ,R = ,as=; tb[a[]]++;
- for(int i=;i<=m;i++){
- int nowl = Q[i].l,nowr = Q[i].r;
- while(R < nowr){
- R++;
- if(dfsin[pm[R]]==R||dfsin[pm[R]]<L){
- if(!tb[a[pm[R]]])as++;
- tb[a[pm[R]]]++;
- }else{
- if(tb[a[pm[R]]] == ) as--;
- tb[a[pm[R]]]--;
- }
- }
- while(L > nowl){
- L--;
- if(dfsout[pm[L]]==L||dfsout[pm[L]]>R){
- if(!tb[a[pm[L]]])as++;
- tb[a[pm[L]]]++;
- }else{
- if(tb[a[pm[L]]] == ) as--;
- tb[a[pm[L]]]--;
- }
- }
- while(R > nowr){
- if(dfsin[pm[R]]==R||dfsin[pm[R]]<L){
- if(tb[a[pm[R]]] == ) as--;
- tb[a[pm[R]]]--;
- }else{
- if(!tb[a[pm[R]]])as++;
- tb[a[pm[R]]]++;
- }
- R--;
- }
- while(L < nowl){
- if(dfsout[pm[L]]==L||dfsout[pm[L]]>R){
- if(tb[a[pm[L]]] == ) as--;
- tb[a[pm[L]]]--;
- }else{
- if(!tb[a[pm[L]]])as++;
- tb[a[pm[L]]]++;
- }
- L++;
- }
- if(Q[i].rem && !tb[a[Q[i].rem]])ans[Q[i].bel]=as+;
- else ans[Q[i].bel] = as;
- }
- for(int i=;i<=m;i++) printf("%d\n",ans[i]);
- }
- int main(){
- read();
- init();
- work();
- return ;
- }
SPOJ10707 COT2 - Count on a tree II 【树上莫队】的更多相关文章
- spoj COT2 - Count on a tree II 树上莫队
题目链接 http://codeforces.com/blog/entry/43230树上莫队从这里学的, 受益匪浅.. #include <iostream> #include < ...
- SP10707 COT2 - Count on a tree II (树上莫队)
大概学了下树上莫队, 其实就是在欧拉序上跑莫队, 特判lca即可. #include <iostream> #include <algorithm> #include < ...
- SP10707 COT2 - Count on a tree II [树上莫队学习笔记]
树上莫队就是把莫队搬到树上-利用欧拉序乱搞.. 子树自然是普通莫队轻松解决了 链上的话 只能用树上莫队了吧.. 考虑多种情况 [X=LCA(X,Y)] [Y=LCA(X,Y)] else void d ...
- SPOJ COT2 Count on a tree II 树上莫队算法
题意: 给出一棵\(n(n \leq 4 \times 10^4)\)个节点的树,每个节点上有个权值,和\(m(m \leq 10^5)\)个询问. 每次询问路径\(u \to v\)上有多少个权值不 ...
- [SPOJ]Count on a tree II(树上莫队)
树上莫队模板题. 使用欧拉序将树上路径转化为普通区间. 之后莫队维护即可.不要忘记特判LCA #include<iostream> #include<cstdio> #incl ...
- 【SPOJ10707】 COT2 Count on a tree II
SPOJ10707 COT2 Count on a tree II Solution 我会强制在线版本! Solution戳这里 代码实现 #include<stdio.h> #inclu ...
- SPOJ COT2 - Count on a tree II(LCA+离散化+树上莫队)
COT2 - Count on a tree II #tree You are given a tree with N nodes. The tree nodes are numbered from ...
- COT2 - Count on a tree II(树上莫队)
COT2 - Count on a tree II You are given a tree with N nodes. The tree nodes are numbered from 1 to N ...
- spoj COT2 - Count on a tree II
COT2 - Count on a tree II http://www.spoj.com/problems/COT2/ #tree You are given a tree with N nodes ...
随机推荐
- H5 71-网易注册界面4
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- Johnson算法
用于求稀疏图上的全局最短路. 考虑将带负权的图变为不带负权的图,再跑\(n\)次Dijkstra. 方法:新建点S,向所有点连边权为\(0\)的边,然后以S为起点跑SPFA.然后将每条边的权值重新赋为 ...
- Magic Stones CodeForces - 1110E (思维+差分)
E. Magic Stones time limit per test 1 second memory limit per test 256 megabytes input standard inpu ...
- mysql数据从windows导出,再导入到linux
从windows导出时,要注意字符集最好和linux的一致,如linux字符集一般为utf8,则导出时可以加上参数--default-character-set=utf8指定字符集,然后导入到linu ...
- gnuplot画折线图
之前尝试用jfreechart画自定义横坐标的折线图或时序图,发现很复杂,后来改用gnuplot了. gnuplot在网上一搜就能找到下载地址. 安装完成后,主要是命令行形式的交互界面,至少比jfre ...
- Redis教程(Windows)
安装 1)下载redis压缩包并 , 推荐地址:https://github.com/MSOpenTech/redis/releases Redis 支持 32 位和 64 位.这个需要根据你系统平 ...
- SQL性能优化-order by语句的优化
原文:http://bbs.landingbj.com/t-0-243203-1.html 在某些情况中,MySQL可以使用一个索引来满足ORDER BY子句,而不需要额外的排序.where条件和or ...
- MySQL数据库性能优化思路与解决方法(二转)
原文:http://bbs.landingbj.com/t-0-242512-1.html 1.锁定表 尽管事务是维护数据库完整性的一个非常好的方法,但却因为它的独占性,有时会影响数据库的性能,尤其是 ...
- day 7-21 pymysql模块
一.安装的两种方法 第一种: #安装 pip3 install pymysql 第二种: 二.链接,执行sql,关闭(游标) import pymysql user = input("use ...
- SAP配置BOM的适用范围
配置BOM中定义属性,单纯的编码要搞死人: 适合小批量周期短多品种