题目

PS国是一个拥有诸多城市的大国,国王Louis为城市的交通建设可谓绞尽脑汁。Louis可以在某些城市之间修建道路,在不同的城市之间修建道路需要不同的花费。Louis希望建造最少的道路使得国内所有的城市连通。但是由于某些因素,城市之间修建道路需要的花费会随着时间而改变,Louis会不断得到某道路的修建代价改变的消息,他希望每得到一条消息后能立即知道使城市连通的最小花费总和, Louis决定求助于你来完成这个任务。

题解

经典的动态最小生成树问题。

可以采用cdq分治的方式来解决。

核心思想就是:

  1. 对于无论被修改的边修改成什么样都一定会被加入的非修改边进行缩点以减小数据范围。
  2. 对于无论被修改的边修改成什么样都一定不被加入的非修改边进行删除以减小数据范围。

对于两种边的确定可以直接设被修改的边的边权为-inf或inf,然后跑Kruskal确定

复杂度。。。

他们说是\(O(nlog^2n)\)的。。。

反正跑的很快就是了。

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
inline void read(int &x){
x=0;static char ch;static bool flag;flag = false;
while(ch=getchar(),ch<'!');if(ch == '-') ch=getchar(),flag = true;
while(x=10*x+ch-'0',ch=getchar(),ch>'!');if(flag) x=-x;
}
#define rg register int
#define rep(i,a,b) for(rg i=(a);i<=(b);++i)
#define per(i,a,b) for(rg i=(a);i>=(b);--i)
const int maxn = 50010;
const int maxm = 100010;
const int inf = 0x3f3f3f3f;
struct Edge{
int u,v,w,id;
bool friend operator < (const Edge &a,const Edge &b){
return a.w < b.w;
}
}e[30][maxm],L[maxm],tmp[maxm];
int Nn[30],Ne[30];
int fa[maxn];
inline int find(int x){
return x == fa[x] ? x : fa[x] = find(fa[x]);
}
inline bool merge(int u,int v){
int x = find(u);
int y = find(v);
if(x == y) return false;
fa[x] = y;return true;
}
int val[maxm];
struct Node{
int k,w;
}qer[maxm];
bool vis[maxm];
int tmp_n[maxn],tmp_m[maxm],map[maxm];
ll ans[maxm];
inline void step1(int &n,int &m,ll &res){
int N = 0,M = 0;
rep(i,1,n) fa[i] = i;
rep(i,0,m-1) vis[i] = 0;
sort(L,L+m);
rep(i,0,m-1){
if(merge(L[i].u,L[i].v) && L[i].w != -inf){
res += L[i].w;
vis[i] = true;
}else tmp[M ++ ] = L[i];
}
rep(i,1,n) fa[i] = i;
rep(i,0,m-1) if(vis[i]) merge(L[i].u,L[i].v);
rep(i,1,n) if(find(i) == i) tmp_n[i] = ++ N;
rep(i,1,n) tmp_n[i] = tmp_n[find(i)];
rep(i,0,M-1){
L[i] = tmp[i];
map[L[i].id] = i;
L[i].u = tmp_n[L[i].u];
L[i].v = tmp_n[L[i].v];
}
n = N;m = M;
}
inline void step2(int &n,int &m){
int M = 0;
rep(i,1,n) fa[i] = i;
sort(L,L+m);
rep(i,0,m-1){
if(merge(L[i].u,L[i].v) || L[i].w == inf){
map[L[i].id] = M;
L[M++] = L[i];
}
}m = M;
}
inline void solve(int l,int r,int cur,ll res){
int n = Nn[cur],m = Ne[cur];
if(l == r) val[qer[r].k] = qer[r].w;
rep(i,0,m-1){
e[cur][i].w = val[e[cur][i].id];
L[i] = e[cur][i];
map[L[i].id] = i;
}
if(l == r){
rep(i,1,n) fa[i] = i;
sort(L,L+m);
rep(i,0,m-1){
if(merge(L[i].u,L[i].v)) res += L[i].w;
}
ans[l] = res;
return ;
}
rep(i,l,r) L[map[qer[i].k]].w = -inf;step1(n,m,res);
rep(i,l,r) L[map[qer[i].k]].w = inf;step2(n,m);
Nn[cur+1] = n;Ne[cur+1] = m;
rep(i,0,m-1) e[cur+1][i] = L[i];
int mid = l+r >> 1;
solve(l,mid,cur+1,res);
solve(mid+1,r,cur+1,res);
}
int main(){
int n,m,Q;read(n);read(m);read(Q);
rep(i,0,m-1){
read(e[0][i].u);
read(e[0][i].v);
read(e[0][i].w);
val[i] = e[0][i].w;
e[0][i].id = i;
}
rep(i,1,Q){
read(qer[i].k);read(qer[i].w);
-- qer[i].k;
}
Nn[0] = n;Ne[0] = m;
solve(1,Q,0,0);
rep(i,1,Q) printf("%lld\n",ans[i]);
return 0;
}

bzoj 2001: City 城市建设 cdq的更多相关文章

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

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

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

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

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

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

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

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

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

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

  6. 【刷题】BZOJ 2001 [Hnoi2010]City 城市建设

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

  7. BZOJ2001 [Hnoi2010]City 城市建设 【CDQ分治 + kruskal】

    题目链接 BZOJ2001 题解 CDQ分治神题... 难想难写.. 比较朴素的思想是对于每个询问都求一遍\(BST\),这样做显然会爆 考虑一下时间都浪费在了什么地方 我们每次求\(BST\)实际上 ...

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

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

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

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

随机推荐

  1. 为什么是kafka(二)

    回答几个网友提出的问题,不清楚的能够看上一篇内容. 1.  kafka的删除策略应该怎么配置?为了提升性能.我是不是应该1小时删除一次消费过的数据. 全然能够依据磁盘大小配置.仅仅要磁盘足够用,全然不 ...

  2. php修改密码

      为了让页面更为好看一些,我一般会选择bootstrap,写起来虽然看着麻烦,但是我们真正需要的只有中间的内容  下面是html的内容 <div id="tbx"" ...

  3. 【BZOJ4928】第二题 树hash+倍增

    [BZOJ4928]第二题 Description 对于一棵有根树,定义一个点u的k-子树为u的子树中距离u不超过k的部分. 注意,假如u的子树中不存在距离u为k的点,则u的k-子树是不存在的. 定义 ...

  4. 白昼夢 / Daydream(模拟)

    C - 白昼夢 / Daydream Time limit : 2sec / Memory limit : 256MB Score : 300 points Problem Statement You ...

  5. WCF基础之消息协定

    通常定义消息的架构,使用数据协定就够了,但是有时必须将类型精确映射到soap消息,方法两种:1.插入自定义soap标头:2.另一种是定义消息的头和正文的安全属性.消息协定通过MessageContra ...

  6. 【学员管理系统】0x02 学生信息管理功能

    [学员管理系统]0x02 学生信息管理功能 写在前面 项目详细需求参见:Django项目之[学员管理系统] Django框架大致处理流程 捋一下Django框架相关的内容: 浏览器输入URL到页面展示 ...

  7. Android系统移植与调试之------->如何修改Android设备的桌面背景图片

    1.切换到~/mx0831-0525/device/other/TBDG1073/overlay/frameworks/base/core/res/res目录 2.准备好一张相应尺寸的图片并且命名为d ...

  8. Jquery点击事件出发顺序

    鼠标点击触发事件执行顺序: mouse down -> mouse up -> click 键盘点击出发事件执行顺序: 点击后马上抬起:key down -> key press - ...

  9. ssm框架与shiro的整合小demo,用idea开发+maven管理

    shiro安全框架是目前为止作为登录注册最常用的框架,因为它十分的强大简单,提供了认证.授权.加密和会话管理等功能 . shiro能做什么? 认证:验证用户的身份 授权:对用户执行访问控制:判断用户是 ...

  10. DAS、NAS、SAN

    目前磁盘存储市场上,存储分类(如下表一)根据服务器类型分为:封闭系统的存储和开放系统的存储,封闭系统主要指大型机,AS400等服务器, 开放系统指基于包括Windows.UNIX.Linux等操作系统 ...