hdu 5458 Stability(树链剖分+强连通缩点+线段树)

2024-06-05 00:48

本文主要是介绍hdu 5458 Stability(树链剖分+强连通缩点+线段树),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目链接:hdu 5458 Stability

解题思路

先将操作处理一遍,获得最终图,然后对图进行双联通缩点,剩下的肯定是一棵树,然后将操作逆着做一遍,遇到删边等于是加一条边,加的这条边u,v等于是将两节点路径上的点联通起来变成一个新的双联通分量,在同一个双联通分量中,明显ans=0。所以我们用线段树维护树的每条边权,一开始全为1,每次添加一条边,就将这条路径上的边权值置为0。

代码

#include <cstdio>
#include <cstring>
#include <vector>
#include <set>
#include <algorithm>using namespace std;
const int maxn = 3 * 1e4 + 5;
const int maxm = 1e5 + 5;
typedef pair<int,int> pii;#define lson(x) ((x)<<1)
#define rson(x) (((x)<<1)|1)namespace SegTree {int lc[maxn<<2], rc[maxn<<2], s[maxn<<2], f[maxn<<2];void maintain(int u, int v) {f[u] = v;s[u] = f[u] * (rc[u] - lc[u] + 1);}void pushup(int u) { s[u] = s[lson(u)] + s[rson(u)]; }void pushdown(int u) {if (f[u] != -1) {maintain(lson(u), f[u]);maintain(rson(u), f[u]);f[u] = -1;}}void build (int u, int l, int r) {lc[u] = l, rc[u] = r, f[u] = -1, s[u] = 1;if (l == r) return;int mid = (l+r)>>1;build(lson(u), l, mid);build(rson(u), mid+1, r);pushup(u);}void modify(int u, int l, int r, int v) {if (l <= lc[u] && rc[u] <= r) {maintain(u, v);return;}pushdown(u);int mid = (lc[u] + rc[u]) >> 1;if (l <= mid) modify(lson(u), l, r, v);if (r > mid) modify(rson(u), l, r, v);pushup(u);}int query(int u, int l, int r) {if (l <= lc[u] && rc[u] <= r)return s[u];pushdown(u);int mid = (lc[u] + rc[u]) >> 1, ret = 0;if (l <= mid) ret += query(lson(u), l, r);if (r > mid) ret += query(rson(u), l, r);pushup(u);return ret;}
};/* 树链剖分*/
namespace TreeChain {int E, first[maxn], jump[maxn<<1], link[maxn<<1];int id, idx[maxn], dep[maxn], top[maxn], far[maxn], son[maxn], cnt[maxn];inline void addEdge(int u, int v) {link[E] = v;jump[E] = first[u];first[u] = E++;}void dfs (int u, int pre, int d) {far[u] = pre;dep[u] = d;cnt[u] = 1;son[u] = 0;for (int i = first[u]; i + 1; i = jump[i]) {int v = link[i];if (v == pre) continue;dfs(v, u, d + 1);cnt[u] += cnt[v];if (cnt[son[u]] < cnt[v])son[u] = v;}}void dfs (int u, int rot) {top[u]= rot;idx[u] = ++id;if (son[u])dfs(son[u], rot);for (int i = first[u]; i + 1; i = jump[i]) {int v = link[i];if (v == far[u] || v == son[u]) continue;dfs(v, v);}}void init (int n, int m, const vector<pii>& edges) {SegTree::build(1, 1, n);int u, v;id = E = 0;memset(first, -1, sizeof(first));for (int i = 0; i < m; i++) {if (edges[i].first == edges[i].second) continue;addEdge(edges[i].first, edges[i].second);}dfs(1, 0, 0);dfs(1, 1);}void modify (int u, int v, int k) {int p = top[u], q = top[v];while (p != q) {if (dep[p] < dep[q]) {swap(p, q);swap(u, v);}SegTree::modify(1, idx[p], idx[u], k);u = far[p];p = top[u];}if (dep[u] > dep[v])swap(u, v);if (u != v)SegTree::modify(1, idx[son[u]], idx[v], k);}int query (int u, int v) {int p = top[u], q = top[v], ret = 0;while (p != q) {if (dep[p] < dep[q]) {swap(p, q);swap(u, v);}ret += SegTree::query(1, idx[p], idx[u]);u = far[p];p = top[u];}if (dep[u] > dep[v])swap(u, v);if (u != v)ret += SegTree::query(1, idx[son[u]], idx[v]);return ret;}
};/* Edge biconneted */
int dfsclock, pre[maxn], iscut[maxm<<1], bccno[maxn], cntbcc;
multiset<pii> chains;
vector<int> G[maxn];
vector<pii> edges;int dfs (int u, int fa) {  int lowu = pre[u] = ++dfsclock;for (int i = 0; i < G[u].size(); i++) { int e = G[u][i];int v = edges[e].second;if (!pre[v]) {int lowv = dfs(v, u);lowu = min(lowu, lowv);if (lowv > pre[u]) iscut[e] = iscut[e^1] = 1;} else if (pre[v] < pre[u] && v != fa)lowu = min(lowu, pre[v]);}return lowu;
} void dfs (int u) {bccno[u] = cntbcc;for (int i = 0; i < G[u].size(); i++) { int e = G[u][i];int v = edges[e].second;if (iscut[e] || bccno[v]) continue;dfs(v);}
}void findEdge (int n) {dfsclock = cntbcc = 0;memset(pre, 0, sizeof(pre));memset(iscut, 0, sizeof(iscut));for (int i = 1; i <= n; i++)if (!pre[i]) dfs(i, -1);memset(bccno, 0, sizeof(bccno));for (int i = 1; i <= n; i++)if (!bccno[i]) {++cntbcc;dfs(i);}
} int N, M, Q, T[maxm], U[maxm], V[maxm];
void init () {chains.clear();scanf("%d%d%d", &N, &M, &Q);int u, v;for (int i = 1; i <= M; i++) {scanf("%d%d", &u, &v);if (u > v) swap(u, v);chains.insert(make_pair(u, v));}for (int i = 1; i <= Q; i++) {scanf("%d%d%d", &T[i], &U[i], &V[i]);if (U[i] > V[i]) swap(U[i], V[i]);if (T[i] == 2) continue;multiset<pii>::iterator it = chains.find(make_pair(U[i], V[i]));chains.erase(it);//chains.erase(make_pair(U[i], V[i]));}/* get final graph */edges.clear();for (int i = 1; i <= N; i++) G[i].clear();for (multiset<pii>::iterator i = chains.begin(); i != chains.end(); i++) {int u = i->first, v = i->second;//printf("%d %d\n", u, v);for (int j = 0; j < 2; j++) {edges.push_back(make_pair(u, v));G[u].push_back(edges.size()-1);swap(u, v);}}/* edge biconnected */findEdge(N);for (int i = 0; i < edges.size(); i++) {edges[i].first = bccno[edges[i].first];edges[i].second = bccno[edges[i].second];}sort(edges.begin(), edges.end());int n = unique(edges.begin(), edges.end()) - edges.begin();TreeChain::init(cntbcc, n, edges);
}int main () {int cas;scanf("%d", &cas);for (int kcas = 1; kcas <= cas; kcas++) {init();printf("Case #%d:\n", kcas);vector<int> ans;for (int i = Q; i; i--) {if (T[i] == 1)TreeChain::modify(bccno[U[i]], bccno[V[i]], 0);elseans.push_back(TreeChain::query(bccno[U[i]], bccno[V[i]]));}for (int i = ans.size()-1; i >= 0; i--)printf("%d\n", ans[i]);}return 0;
}

这篇关于hdu 5458 Stability(树链剖分+强连通缩点+线段树)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1031609

相关文章

poj3468(线段树成段更新模板题)

题意:包括两个操作:1、将[a.b]上的数字加上v;2、查询区间[a,b]上的和 下面的介绍是下解题思路: 首先介绍  lazy-tag思想:用一个变量记录每一个线段树节点的变化值,当这部分线段的一致性被破坏我们就将这个变化值传递给子区间,大大增加了线段树的效率。 比如现在需要对[a,b]区间值进行加c操作,那么就从根节点[1,n]开始调用update函数进行操作,如果刚好执行到一个子节点,

hdu1394(线段树点更新的应用)

题意:求一个序列经过一定的操作得到的序列的最小逆序数 这题会用到逆序数的一个性质,在0到n-1这些数字组成的乱序排列,将第一个数字A移到最后一位,得到的逆序数为res-a+(n-a-1) 知道上面的知识点后,可以用暴力来解 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#in

hdu1689(线段树成段更新)

两种操作:1、set区间[a,b]上数字为v;2、查询[ 1 , n ]上的sum 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>#include<map>#include<stdio.h>#include<stdl

usaco 1.3 Mixing Milk (结构体排序 qsort) and hdu 2020(sort)

到了这题学会了结构体排序 于是回去修改了 1.2 milking cows 的算法~ 结构体排序核心: 1.结构体定义 struct Milk{int price;int milks;}milk[5000]; 2.自定义的比较函数,若返回值为正,qsort 函数判定a>b ;为负,a<b;为0,a==b; int milkcmp(const void *va,c

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

hdu 2093 考试排名(sscanf)

模拟题。 直接从教程里拉解析。 因为表格里的数据格式不统一。有时候有"()",有时候又没有。而它也不会给我们提示。 这种情况下,就只能它它们统一看作字符串来处理了。现在就请出我们的主角sscanf()! sscanf 语法: #include int sscanf( const char *buffer, const char *format, ... ); 函数sscanf()和

hdu 2602 and poj 3624(01背包)

01背包的模板题。 hdu2602代码: #include<stdio.h>#include<string.h>const int MaxN = 1001;int max(int a, int b){return a > b ? a : b;}int w[MaxN];int v[MaxN];int dp[MaxN];int main(){int T;int N, V;s

hdu 1754 I Hate It(线段树,单点更新,区间最值)

题意是求一个线段中的最大数。 线段树的模板题,试用了一下交大的模板。效率有点略低。 代码: #include <stdio.h>#include <string.h>#define TREE_SIZE (1 << (20))//const int TREE_SIZE = 200000 + 10;int max(int a, int b){return a > b ? a :

hdu 1166 敌兵布阵(树状数组 or 线段树)

题意是求一个线段的和,在线段上可以进行加减的修改。 树状数组的模板题。 代码: #include <stdio.h>#include <string.h>const int maxn = 50000 + 1;int c[maxn];int n;int lowbit(int x){return x & -x;}void add(int x, int num){while

hdu 3790 (单源最短路dijkstra)

题意: 每条边都有长度d 和花费p,给你起点s 终点t,要求输出起点到终点的最短距离及其花费,如果最短距离有多条路线,则输出花费最少的。 解析: 考察对dijkstra的理解。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstrin