zoukankan      html  css  js  c++  java
  • CF891C Envy【最小生成树】

    题目链接

    我们知道,根据Kruskal的贪心,对于最小生成树,每一种权值的边数是一样的,而且如果将(leq x)的边做最小生成树,合法方案的联通性是一样的。所以我们可以对于所有边分开考虑。

    对于一组询问,对于所有权值,权值为(x)的有(k)个,那么可以将(<x)的边全部加入,然后将这(k)个边加入,看看能不能全部加入进去。如果有一个成环了,那么肯定是不行的。

    那么(q)组询问,可以离线下来,对于(边权,询问编号)二元组排序,然后对于同一边权的同一组询问,尝试加入,到了下一组询问就撤销,然后处理完之后又全部加入进去。用可撤销并查集就可以做了。

    #include<bits/stdc++.h>
    #define Rint register int
    using namespace std;
    const int N = 500003;
    int n, m, k, q, mx, fa[N], siz[N];
    bool ans[N];
    struct Edge {
    	int u, v, w;
    	inline bool operator < (const Edge &o) const {return w < o.w;}
    } e[N];
    struct Query {
    	int u, v, w, id;
    	inline bool operator < (const Query &o) const {return id < o.id;}
    };
    vector<Query> vec[N];
    inline int getfa(int x){
    	return x == fa[x] ? x : getfa(fa[x]);
    }
    int stk[N], top;
    inline bool comb(int x, int y){
    	x = getfa(x); y = getfa(y);
    	if(x != y){
    		if(siz[x] > siz[y]) swap(x, y);
    		siz[y] += siz[x]; fa[x] = y; stk[++ top] = x;
    		return true;
    	}
    	return false;
    }
    int main(){
    	scanf("%d%d", &n, &m);
    	for(Rint i = 1;i <= m;i ++) scanf("%d%d%d", &e[i].u, &e[i].v, &e[i].w), mx = max(mx, e[i].w);
    	scanf("%d", &q);
    	for(Rint i = 1;i <= q;i ++){
    		scanf("%d", &k);
    		while(k --){
    			int now; scanf("%d", &now);
    			vec[e[now].w].push_back((Query){e[now].u, e[now].v, e[now].w, i});
    		}
    	}
    	for(Rint i = 1;i <= mx;i ++) if(!vec[i].empty())
    		sort(vec[i].begin(), vec[i].end());
    	sort(e + 1, e + m + 1);
    	for(Rint i = 1;i <= n;i ++) siz[i] = 1, fa[i] = i;
    	for(Rint i = 1;i <= q;i ++) ans[i] = true;
    	for(Rint i = 1;i <= m;){
    		int val = e[i].w; top = 0;
    		for(Rint j = 0;j < vec[val].size();j ++){
    			if(!ans[vec[val][j].id]) continue;
    			if(j && vec[val][j].id != vec[val][j - 1].id){
    				while(top){
    					siz[fa[stk[top]]] -= siz[stk[top]];
    					fa[stk[top]] = stk[top]; -- top;
    				}
    			}
    			if(!comb(vec[val][j].u, vec[val][j].v)) ans[vec[val][j].id] = false;
    		}
    		while(e[i].w == val){
    			comb(e[i].u, e[i].v); ++ i;
    		}
    	}
    	for(Rint i = 1;i <= q;i ++) puts(ans[i] ? "YES" : "NO");
    }
    
  • 相关阅读:
    ubuntu上一些好的操作方式 习惯 文件备份
    ubuntu ufw 防火墙的设置
    ubuntu 的SSH 服务
    对一个数组的处理。
    ubuntu 的 crontab 计划任务
    django1.6 apache 项目部署.
    只此荒废的一个半个月。
    myeclipse python插件安装和环境配置(Windows7下)
    ubunt 12.04 64位 english 服务器版安装
    ubuntu sudo自动切换root 无需输入密码
  • 原文地址:https://www.cnblogs.com/AThousandMoons/p/11609246.html
Copyright © 2011-2022 走看看