题目链接

BZOJ2001

题解

CDQ分治神题。。。

难想难写。。

比较朴素的思想是对于每个询问都求一遍\(BST\),这样做显然会爆

考虑一下时间都浪费在了什么地方

我们每次求\(BST\)实际上就只有一条边不同,我们实际浪费了很多时间在处理相同的边上

那就考虑分治

对于一个待修改的边集,我们将其权值全部设为\(-\infty\),跑一遍\(BST\),此时其它边如果被选中,说明这些边在单独询问时也一定会被选,将这些边连的点缩点

同样,对于一个待修改的边集,我们将其权值全部设为\(\infty\),跑一遍\(BST\),此时其它边没被选中,说明这些边在单独询问时也一定不会被选,将这些边删掉

这样就可以\(A\)了

复杂度我也不知道是什么

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<map>
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define mp(a,b) make_pair<int,int>(a,b)
#define cls(s) memset(s,0,sizeof(s))
#define cp pair<int,int>
#define LL long long int
using namespace std;
const int maxn = 50005,maxm = 50005,INF = 0x3fffffff;
inline int read(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
return out * flag;
}
struct EDGE{int i,a,b,w;}e[50][maxm],d[maxm],b[maxm];
struct Que{int u,v;}q[maxm];
inline bool operator <(const EDGE& a,const EDGE& b){
return a.w < b.w;
}
LL ans[maxm];
int n,m,Q,sum[50],w[maxm],id[maxm],a[maxm];
int pre[maxm];
int find(int u){return u == pre[u] ? u : pre[u] = find(pre[u]);}
void clear(int t){
for (int i = 1; i <= t; i++){
pre[d[i].a] = d[i].a;
pre[d[i].b] = d[i].b;
}
}
void comb(int& t,LL& Ans){
clear(t); int tmp = 0,fa,fb;
sort(d + 1,d + 1 + t);
REP(i,t){
fa = find(d[i].a); fb = find(d[i].b);
if (fa != fb){
pre[fb] = fa;
b[++tmp] = d[i];
}
}
REP(i,tmp) {
pre[b[i].a] = b[i].a;
pre[b[i].b] = b[i].b;
}
REP(i,tmp) if (b[i].w != -INF){
fa = find(b[i].a); fb = find(b[i].b);
if (fa != fb){
pre[fb] = fa;
Ans += b[i].w;
}
}
tmp = 0;
REP(i,t){
fa = find(d[i].a); fb = find(d[i].b);
if (fa != fb){
b[++tmp] = d[i];
id[d[i].i] = tmp;
b[tmp].a = find(b[tmp].a);
b[tmp].b = find(b[tmp].b);
}
}
REP(i,tmp) d[i] = b[i];
t = tmp;
}
void rd(int& t){
clear(t); int tmp = 0,fa,fb;
sort(d + 1,d + 1 + t);
REP(i,t){
fa = find(d[i].a); fb = find(d[i].b);
if (fa != fb){
pre[fb] = fa;
b[++tmp] = d[i];
}
else if (d[i].w == INF){
b[++tmp] = d[i];
}
}
for (int i = 1; i <= tmp; i++) d[i] = b[i];
t = tmp;
}
void solve(int l,int r,int now,LL Ans){
int t = sum[now];
if (l == r) a[q[l].u] = q[l].v; //原标号边权值
for (int i = 1; i <= t; i++)
e[now][i].w = a[e[now][i].i]; //边赋值
for (int i = 1; i <= t; i++)
d[i] = e[now][i],id[d[i].i] = i; //新边对应旧边位置
if (l == r){
ans[l] = Ans; clear(t);
sort(d + 1,d + 1 + t);
int fa,fb;
for (int i = 1; i <= t; i++){
fa = find(d[i].a); fb = find(d[i].b);
if (fa != fb){
pre[fb] = fa; ans[l] += d[i].w;
}
}
return;
}
for (int i = l; i <= r; i++) d[id[q[i].u]].w = -INF;
comb(t,Ans);
for (int i = l; i <= r; i++) d[id[q[i].u]].w = INF;
rd(t);
REP(i,t) e[now + 1][i] = d[i];
sum[now + 1] = t;
int mid = l + r >> 1;
solve(l,mid,now + 1,Ans);
solve(mid + 1,r,now + 1,Ans);
}
int main(){
n = read(); m = read(); Q = read();
for (int i = 1; i <= m; i++){
e[0][i].i = i;
e[0][i].a = read();
e[0][i].b = read();
a[i] = e[0][i].w = read();
}
for (int i = 1; i <= Q; i++){
q[i].u = read();
q[i].v = read();
}
sum[0] = m;
solve(1,Q,0,0);
for (int i = 1; i <= Q; i++)
printf("%lld\n",ans[i]);
return 0;
}

BZOJ2001 [Hnoi2010]City 城市建设 【CDQ分治 + kruskal】的更多相关文章

  1. BZOJ2001 [Hnoi2010]City 城市建设 CDQ分治

    2001: [Hnoi2010]City 城市建设 Time Limit: 20 Sec  Memory Limit: 162 MB Description PS国是一个拥有诸多城市的大国,国王Lou ...

  2. BZOJ2001: [Hnoi2010]City 城市建设

    题目:http://www.lydsy.com/JudgeOnline/problem.php?id=2001 cdq分治+重建图. 可以保留当前一定会被选的非修改边然后把点缩起来.这样的话每次点数至 ...

  3. bzoj 2001 CITY 城市建设 cdq分治

    题目传送门 题解: 对整个修改的区间进行分治.对于当前修改区间来说,我们对整幅图中将要修改的边权都先改成-inf,跑一遍最小生成树,然后对于一条树边并且他的权值不为-inf,那么这条边一定就是树边了. ...

  4. 【BZOJ2001】 [Hnoi2010]City 城市建设

    BZOJ2001 [Hnoi2010]City 城市建设 Solution 我们考虑一下这个东西怎么求解? 思考无果...... 咦? 好像可以离线cdq,每一次判断一下如果这条边如果不选就直接删除, ...

  5. BZOJ 2001: [Hnoi2010]City 城市建设

    2001: [Hnoi2010]City 城市建设 Time Limit: 20 Sec  Memory Limit: 162 MBSubmit: 1132  Solved: 555[Submit][ ...

  6. 【bzoj2001】 Hnoi2010—City 城市建设

    http://www.lydsy.com/JudgeOnline/problem.php?id=2001 (题目链接) 题意 给出一张无向图,$m$组操作,每次修改一条边的权值,对于每次操作输出修改之 ...

  7. bzoj 2001: City 城市建设 cdq

    题目 PS国是一个拥有诸多城市的大国,国王Louis为城市的交通建设可谓绞尽脑汁.Louis可以在某些城市之间修建道路,在不同的城市之间修建道路需要不同的花费.Louis希望建造最少的道路使得国内所有 ...

  8. [HNOI2010]CITY 城市建设

    问题: 给一张图,支持边长度修改,求MST 题解: 自己想就想不到了.. 考虑cdq分治 1.首先求出一定有用的边 对于未处理的边,全部设为-INF,求一次MST,出现在MST上的边一定最终出现在后面 ...

  9. 2001: [Hnoi2010]City 城市建设 - BZOJ

    DescriptionPS国是一个拥有诸多城市的大国,国王Louis为城市的交通建设可谓绞尽脑汁.Louis可以在某些城市之间修建道路,在不同的城市之间修建道路需要不同的花费.Louis希望建造最少的 ...

随机推荐

  1. python3 练习题100例 (二十三)与7相关的数

    与7相关的数:如果一个正整数,它能被7整除或者它的十进制表示法中某个位数上的数字为7,则称之为与7相关的数.(10分) 题目内容: 现在我们给定一个正整数n(n<1000),求所有小于等于n的与 ...

  2. 华为机试 求int型数据在内存中存储时1的个数

    题目描述 输入一个int型的正整数,计算出该int型数据在内存中存储时1的个数. 输入描述: 输入一个整数(int类型) 输出描述: 这个数转换成2进制后,输出1的个数 输入 5 输出 2 普通运算方 ...

  3. 03---Nginx配置文件

    #启动子进程程序默认用户#user nobody;#一个主进程和多个工作进程.工作进程是单进程的,且不需要特殊授权即可运行:这里定义的是工作进程数量worker_processes 1; #全局错误日 ...

  4. linux c fprintf()

    #include<stdio.h> #include<unistd.h> #include<time.h> int main(int argc,char *argv ...

  5. Java测试工具和框架

    个人目前只接触过JUnit以及Powermock,后续会关注更多有关测试这方面的东西 8个超实用的Java测试工具和框架_开发/数据库_IT专家网 http://database.ctocio.com ...

  6. 异常处理中try,else,finally含有return的情况解析

    直接看代码,拿到你的py下运行测试一下就 明白了. 例一: def f(): try: print() finally: print() print(f()) # 若注释掉finally内的retur ...

  7. Vue折腾记 - (2)写一个不大靠谱的面包屑组件

    先看效果图 我把页面标题和面包屑封装到一起..就不用涉及到组件的通讯了,不然又要去监听路由或者依赖状态去获取 这里写图片描述 疑惑解答: 点击父(也就是折叠菜单)为什么会跑到子菜单第一个 因为我第一个 ...

  8. Week2 Teamework from Z.XML 软件分析与用户需求调查(四)Bing桌面及助手的现状与发展

    一.Bing搜索的相关背景 第一,必应搜索前几年的发展重点在于欧美市场,并且取得了一定的成效:根据 Hitwise 的统计数据,Bing 在 2011年3 月份市场占有率突破了 30% 大关,达到 3 ...

  9. eclipse mylyn.tasks.ui

    sudo rm workspace/.metadata/.lock ./Applications/eclipse/Eclipse.app/Contents/MacOS/eclipse -clean - ...

  10. Uva 294 Divisors(唯一分解定理)

    题意:求区间内正约数最大的数. 原理:唯一分解定义(又称算术基本定理),定义如下: 任何一个大于1的自然数 ,都可以唯一分解成有限个质数的乘积  ,这里  均为质数,其诸指数  是正整数.这样的分解称 ...