[Luogu P4180][BJWC 2010]严格次小生成树
严格次小生成树,关键是“严格”,如果是不严格的其实只需要枚举每条不在最小生成树的边,如果得到边权和大于等于最小生成树的结束就行。原理就是因为Kruskal非常贪心,只要随便改一条边就能得到一个非严格的次小生成树。然而是严格的QAQ,于是得搞点别的东西来实现“严格”,维护个次大值就行。依次枚举每条边,如果这条边和加上这条边构成的环中最大的边边权相等,取次大值,否则取最大值。
参考代码:
#include<cstdio>
#include<algorithm>
#define ll long long
#include<iostream>
#define A(x) cout << #x << " " << x << endl;
#define qwq 900100
const ll inf = ;
using namespace std;
struct Node
{
ll u,v,val,nxt;
bool used;
} a[ * qwq],b[ * qwq];
ll head[qwq];
ll f[qwq];
ll find(ll x) {
if(x == f[x]) return x;
return f[x] = find(f[x]);
}
ll n,m,cnt;
void add(ll u,ll v,ll w) {
cnt++;
b[cnt].nxt = head[u];
head[u] = cnt;
b[cnt].val = w;
b[cnt].v = v;
}
bool cmp(Node a,Node b) {
return a.val < b.val;
}
ll ans = 0ll;
void kruskal() {
ll k = ;
for(ll i = ; i <= n; i++) f[i] = i;
for(ll i = ; i <= m; i++)
{
ll f1 = find(a[i].u);
ll f2 = find(a[i].v);
if(f1 != f2) {
f[f1] = f2;
k++;
ans += a[i].val;
add(a[i].u,a[i].v,a[i].val);
add(a[i].v,a[i].u,a[i].val);
a[i].used = ;
if(k == n - ) break;
}
}
}
ll fa[qwq][],dep[qwq];
ll maxx[qwq][],lmax[qwq][];
void dfs(ll u,ll pa) {
fa[u][] = pa;
for(int i = head[u]; i; i = b[i].nxt) {
ll v = b[i].v;
if(v == pa) continue;
dep[v] = dep[u] + 1ll;
maxx[v][] = b[i].val;
lmax[v][] = -inf;
dfs(v,u);
}
}
void max_set() {
for(ll i = ; i <= ; i++)
for(ll u = ; u <= n; u++) {
fa[u][i] = fa[fa[u][i - ]][i - ];
maxx[u][i] = max(maxx[u][i - ],maxx[fa[u][i - ]][i - ]);
lmax[u][i] = max(lmax[u][i - ],lmax[fa[u][i - ]][i - ]);
if(maxx[u][i - ] > maxx[fa[u][i - ]][i - ])
lmax[u][i] = max(lmax[u][i],maxx[fa[u][i - ]][i - ]);
else if(maxx[u][i - ] < maxx[fa[u][i - ]][i - ])
lmax[u][i] = max(lmax[u][i],maxx[u][i - ]);
}
}
ll lca(ll x,ll y) {
if(dep[x] < dep[y]) swap(x,y);
for(ll i = ; i >= ; i--)
if(dep[fa[x][i]] >= dep[y])
x = fa[x][i];
if(x == y) return x;
for(ll i = ; i >= ; i--)
if(fa[x][i] ^ fa[y][i]) {
x = fa[x][i];
y = fa[y][i];
}
return fa[x][];
}
ll find_max(ll u,ll v,ll qaq) {
ll Ans = -inf;
for(ll i = ; i >= ; i--) {
if(dep[fa[u][i]] >= dep[v]) {
if(qaq != maxx[u][i])Ans = max(Ans,maxx[u][i]);
else Ans = max(Ans,lmax[u][i]);
u = fa[u][i];
}
}
return Ans;
}
int main()
{
scanf("%lld %lld",&n,&m);
for(ll i = ;i <= m;i++)
{
scanf("%lld %lld %lld",&a[i].u,&a[i].v,&a[i].val);
}
sort(a + ,a + + m,cmp);
kruskal();
lmax[][] = -inf;
dep[] = 1ll;
dfs(,-);
max_set();
ll answ = inf;
for(ll i = ; i <= m; i++)
{
if(!a[i].used)
{
ll u = a[i].u;
ll v = a[i].v;
ll d = a[i].val;
ll Lca = lca(u,v);
ll maxu = find_max(u,Lca,d);
ll maxv = find_max(v,Lca,d);
answ = min(answ,ans - max(maxu,maxv) + d);
}
}
printf("%lld",answ);
}
[Luogu P4180][BJWC 2010]严格次小生成树的更多相关文章
- 【Beijing 2010】 次小生成树
[题目链接] 点击打开链接 [算法] 首先,有一个结论 : 一定有一棵严格次小生成树是在最小生成树的基础上去掉一条边,再加上一条边 这个结论的正确性是显然的 我们先用kruskal算法求出最小生成树, ...
- 【luogu P4180 严格次小生成树[BJWC2010]】 模板
题目链接:https://www.luogu.org/problemnew/show/P4180 这个题卡树剖.记得开O2. 这个题inf要到1e18. 定理:次小生成树和最小生成树差距只有在一条边上 ...
- Luogu P4180 【模板】严格次小生成树[BJWC2010]
P4180 [模板]严格次小生成树[BJWC2010] 题意 题目描述 小\(C\)最近学了很多最小生成树的算法,\(Prim\)算法.\(Kurskal\)算法.消圈算法等等.正当小\(C\)洋洋得 ...
- 【题解】洛谷P4180 [BJWC2010] 严格次小生成树(最小生成树+倍增求LCA)
洛谷P4180:https://www.luogu.org/problemnew/show/P4180 前言 这可以说是本蒟蒻打过最长的代码了 思路 先求出此图中的最小生成树 权值为tot 我们称这棵 ...
- [Luogu] 次小生成树
https://www.luogu.org/problemnew/show/P4180#sub 严格次小生成树,即不等于最小生成树中的边权之和最小的生成树 首先求出最小生成树,然后枚举所有不在最小生成 ...
- P4180 【模板】严格次小生成树[BJWC2010]
P4180 [模板]严格次小生成树[BJWC2010] 倍增(LCA)+最小生成树 施工队挖断学校光缆导致断网1天(大雾) 考虑直接枚举不在最小生成树上的边.但是边权可能与最小生成树上的边相等,这样删 ...
- 洛谷P4180【Beijing2010组队】次小生成树Tree
题目描述: 小C最近学了很多最小生成树的算法,Prim算法.Kurskal算法.消圈算法等等.正当小C洋洋得意之时,小P又来泼小C冷水了.小P说,让小C求出一个无向图的次小生成树,而且这个次小生成树还 ...
- P4180 [BJWC2010]严格次小生成树
P4180 [BJWC2010]严格次小生成树 P4180 题意 求出一个无向联通图的严格次小生成树.严格次小生成树的定义为边权和大于最小生成树的边权和但不存在另一棵生成树的边权和在最小生成树和严格次 ...
- 洛谷P4180 [Beijing2010组队]次小生成树Tree(最小生成树,LCT,主席树,倍增LCA,倍增,树链剖分)
洛谷题目传送门 %%%TPLY巨佬和ysner巨佬%%% 他们的题解 思路分析 具体思路都在各位巨佬的题解中.这题做法挺多的,我就不对每个都详细讲了,泛泛而谈吧. 大多数算法都要用kruskal把最小 ...
随机推荐
- LoadRunner 11 error:Cannot initialize driver dll
LoadRunner 11 error:Cannot initialize driver dll 这个错误很容易解决,使用win7系统时,有些程序要以管理员身份才能运行. 解决方案:右键选择:“以管理 ...
- Android开发欢迎页点击跳过倒计时进入主页
没点击跳过自然进入主页,点击跳过之后立即进入主页 1.欢迎页布局activity_sp.xml放一张背景图(图片随你便啦)再放一个盛放倒计时的TextView <?xml versi ...
- SQL瓶颈分析,以及适应最佳执行计划的探讨
原文地址: https://blog.csdn.net/daiqiulong2/article/details/86546446?tdsourcetag=s_pcqq_aiomsg 年纪大了,慢慢 ...
- Linux内存管理 (10)缺页中断处理
专题:Linux内存管理专题 关键词:数据异常.缺页中断.匿名页面.文件映射页面.写时复制页面.swap页面. malloc()和mmap()等内存分配函数,在分配时只是建立了进程虚拟地址空间,并没有 ...
- CF618F Double Knapsack 构造、抽屉原理
传送门 首先,选取子集的限制太宽了,子集似乎只能枚举,不是很好做.考虑加强限制条件:将"选取子集"的限制变为"选取子序列"的限制.在接下来的讨论中我们将会知道: ...
- left join inner join 区别
left 以左表为准,左表在右表没有对应的记录,也为显示(右表字段填空). inner 需要满足两张表都有记录. 不管哪种join 一对多最终的结局 只会是多条记录
- C语言之控制语言:分支和跳转
if语句 #include<stdio.h> int main(void) { const int FREEZING = 0; float temperature; int cold_da ...
- 4月11日java多线程4
继昨天学习了线程池之后,今天学习了多线程内的锁Lock. 定义方法: ReentrantLock queueLock = new ReentrantLock(); //可重入锁 ReentrantRe ...
- php如何解决中文乱码问题?
为什么会出现中文乱码? 很多新手朋友学习PHP的时候,发现程序中的中文在输出的时候会出现乱码的问题,那么为什么会出现这种乱码的情况呢?一般来说,乱码的出现有2种原因,一种是由于编码(charset) ...
- UIAutomator简介
简介 Android 4.3发布的时候包含了一种新的测试工具–uiautomator,uiautomator是用来做UI测试的.也就是普通的手工测试,点击每个控件元素 看看输出的结果是否符合预期.比如 ...