bzoj 3669: [Noi2014]魔法森林 (LCT & kruskal)
这道题呢,
首先按照关键字a排序,然后不断地加边,用lct维护这个过程
具体实现: 先按照关键字a排序,枚举每一条边,判断两点是否已经联通(kruskal 部分)如果联通,就在两点路径间寻找最大的b, 和这条边的b值相比较,如果更大一些,就切断u,v之间的路径, 并连上这条边;
如果不联通,就让它联通(好随意啊= =
最后寻找路径之间最大的b + 当前的a,和原来的答案相比较。
剩下的用lct维护
注意联通的方式是把把边看做点, 然后连接(所以加上n) ->link(u, i + n); link(v, i + n);
(由于我一直不理解怎么保证当前的a一定在路径上,所以这是自己yy的证明(可能有很大的bug,勿喷):当前的a如果不在(1,n)路径之中,但(1,n)已联通,那么最优值在之前已经被计算过,否则由于a是递增的,那么此时的a就是路径中最大的a)
下面是代码
/**************************************************************
Problem: 3669
User: cminus
Language: C++
Result: Accepted
Time:5104 ms
Memory:6128 kb
****************************************************************/
#include <cstdio>
#include <algorithm>
using namespace std;
#define l(x) ch[x][0]
#define r(x) ch[x][1]
#define kd(x) (r(fa[x]) == x)
#define setc(f, c, k) (ch[fa[c] = f][k] = c)
#define isRoot(x) (r(fa[x]) != x && l(fa[x]) != x)
const int N = ;
struct Edge{
int u, v, a, b;
inline void init(){
scanf("%d %d %d %d", &u, &v, &a, &b);
}
inline bool operator <(const Edge &rhs)const{
return a < rhs.a;
}
}e[];
int fa[N], ch[N][], rev[N], val[N], maxp[N], f[];
int INF = 0x7f7f7f7f;
inline void update(int x) {
maxp[x] = x;
if (val[maxp[x]] < val[maxp[l(x)]]) maxp[x] = maxp[l(x)];
if (val[maxp[x]] < val[maxp[r(x)]]) maxp[x] = maxp[r(x)];
}
inline void push(int x) {
if (rev[x] and x){
rev[x] = ;
if (l(x)) rev[l(x)] ^= , swap(l(l(x)), r(l(x)));
if (r(x)) rev[r(x)] ^= , swap(l(r(x)), r(r(x)));
}
}
inline void pushDown(int x) {
if (! isRoot(x)) pushDown(fa[x]);
push(x);
}
inline void rotate(int x) {
int y = fa[x], t = kd(x);
setc(y, ch[x][t^], t);
if (isRoot(y)) fa[x] = fa[y];
else setc(fa[y], x, kd(y));
setc(x, y, t^);
update(y); update(x);
}
inline void splay(int x){
pushDown(x);
while(! isRoot(x)){
if (! isRoot(fa[x]))
if (kd(x) == kd(fa[x])) rotate(fa[x]);
else rotate(x); rotate(x);
}
}
inline void access(int x){
int t = ;
while(x) {
splay(x);
r(x) = t; update(x);
t = x; x = fa[x];
}
}
inline void makeRoot(int x){
access(x); splay(x);
rev[x] ^= ; swap(l(x), r(x));
}
inline void link(int u, int v) { makeRoot(u); fa[u] = v; }
inline void cut(int u, int v){
makeRoot(u);
access(v); splay(v);
fa[u] = l(v) = ;
}
inline int query(int u, int v){
makeRoot(u);
access(v); splay(v);
return maxp[v];
}
int find(int x) { return f[x] == x ? x : f[x] = find(f[x]); }
int main(){
int n, m;
scanf("%d %d", &n, &m);
int ans = INF;
for (int i = ; i <= m; i++) e[i].init();
for (int i = ; i <= n; i++) f[i] = i;
sort(e + , e + m + );
for (int i = ; i <= m; i++){
int u = e[i].u, v = e[i].v, rt1 = find(u), rt2 = find(v);
val[i + n] = e[i].b; maxp[i + n] = i + n;
if (rt1 == rt2){
int p = query(u, v);
if (val[p] > e[i].b) cut(u, p), cut(v, p);
else continue;
}
else f[rt1] = rt2;
link(u, i + n); link(v, i + n);
if (find() == find(n)) ans = min(ans, val[query(, n)] + e[i].a);
}
printf("%d\n", ans == INF ? - : ans);
return ;
}
bzoj 3669: [Noi2014]魔法森林 (LCT & kruskal)的更多相关文章
- BZOJ 3669: [Noi2014]魔法森林 [LCT Kruskal | SPFA]
题目描述 为了得到书法大家的真传,小 E 同学下定决心去拜访住在魔法森林中的隐 士.魔法森林可以被看成一个包含 n 个节点 m 条边的无向图,节点标号为 1,2,3,…,n,边标号为 1,2,3,…, ...
- BZOJ 3669: [Noi2014]魔法森林( LCT )
排序搞掉一维, 然后就用LCT维护加边MST. O(NlogN) ------------------------------------------------------------------- ...
- bzoj 3669: [Noi2014]魔法森林 (LCT)
链接:https://www.lydsy.com/JudgeOnline/problem.php?id=3669 题面: 3669: [Noi2014]魔法森林 Time Limit: 30 Sec ...
- bzoj 3669: [Noi2014] 魔法森林 LCT版
Description 为了得到书法大家的真传,小E同学下定决心去拜访住在魔法森林中的隐士.魔法森林可以被看成一个包含个N节点M条边的无向图,节点标号为1..N,边标号为1..M.初始时小E同学在号节 ...
- BZOJ 3669: [Noi2014]魔法森林(lct+最小生成树)
传送门 解题思路 \(lct\)维护最小生成树.我们首先按照\(a\)排序,然后每次加入一条边,在图中维护一棵最小生成树.用并查集判断一下\(1\)与\(n\)是否联通,如果联通的话就尝试更新答案. ...
- bzoj 3669: [Noi2014]魔法森林
bzoj 3669: [Noi2014]魔法森林 Description 为了得到书法大家的真传,小E同学下定决心去拜访住在魔法森林中的隐士.魔法森林可以被看成一个包含个N节点M条边的无向图,节点标号 ...
- bzoj 3669: [Noi2014]魔法森林 动态树
3669: [Noi2014]魔法森林 Time Limit: 30 Sec Memory Limit: 512 MBSubmit: 363 Solved: 202[Submit][Status] ...
- bzoj 3669: [Noi2014]魔法森林 -- 动点spfa
3669: [Noi2014]魔法森林 Time Limit: 30 Sec Memory Limit: 512 MB 动点spfa Description 为了得到书法大家的真传,小E同学下定决心 ...
- [BZOJ 3669] [Noi2014] 魔法森林 【LCT】
题目链接:BZOJ - 3669 题目分析 如果确定了带 x 只精灵A,那么我们就是要找一条 1 到 n 的路径,满足只经过 Ai <= x 的边,而且要使经过的边中最大的 Bi 尽量小. 其实 ...
- bzoj 3669: [Noi2014]魔法森林(并查集+LCT)
Description 为了得到书法大家的真传,小E同学下定决心去拜访住在魔法森林中的隐士.魔法森林可以被看成一个包含个N节点M条边的无向图,节点标号为1..N,边标号为1..M.初始时小E同学在号节 ...
随机推荐
- LeetCode刷题 --杂篇 --数组,链表,栈,队列
武汉加油,中国加油.希望疫情早日结束. 由于疫情,二狗寒假在家不能到处乱逛,索性就在家里系统的刷一下算法的内容,一段时间下来倒也有些小小的收获.只是一来家中的小破笔记本写起博客来实在不是很顺手,二来家 ...
- 剑指offer-面试题7-重建二叉树-二叉树
/* 题目: 输入二叉树的前序遍历和中序遍历的结果,重建二叉树.假设输入的前序遍历和中序遍历的结果中不包含重复的数字. */ /* 思路: 使用前序遍历找到根节点,再通过中序遍历找到左子树和右子树. ...
- CF776D The Door Problem [2sat]
考虑 \(\texttt{2-SAT}\) 首先每个门 \(i\) 都有一个初始状态 \(a_i\) 题目条件每个门只被两个开关控制,那么很显然的 \(\texttt{2-SAT}\) 用 \(b_{ ...
- Java文件与流测试
import java.io.File; import java.io.InputStreamReader; import java.math.BigDecimal; import java.io.B ...
- vue.js事件传值之子组件传向父组件以及$emit的使用
在项目开发中,有时候会遇到一种需求比如是:在子组件中,通过一个事件,比如点击事件,去改变父组件中的某个值,下面来看看是怎么个流程 还是先截图目录结构 父组件为app.vue,components中的文 ...
- DVA知识集合
react与dva 原文地址:https://github.com/dvajs/dva-knowledgemap 1.变量声明 const DELAY = 1000 let count = 0 cou ...
- dev 控件用法2 之repositoryItemSearchLookUpEdit
repositoryItemSearchLookUpEdit var y = userinfo.Select.ToList( a => new { userid = a.userid, code ...
- react-native简单使用
基本组件的使用介绍 View: Text: TextInput: Image: Button: ActivityIndicator: ScrollView:这是一个列表滚动的组件 ListView:也 ...
- MySQL的聚合函数
MySQL的聚合函数 网站:https://www.runoob.com/mysql/mysql-functions.html 函数名 功能描述 count() 计数 sum() 求和 avg() 平 ...
- PATA-1151 LCA in a Binary Tree
题意:根据前序和中序建立树,寻找两个点的LCA. 我在之前的博客中写了关于LCA的多种求法. https://www.cnblogs.com/yy-1046741080/p/11505547.html ...