题解【洛谷P2619】[国家集训队2]Tree I
2024-09-06 22:54:24
题目描述
给你一个无向带权连通图,每条边是黑色或白色。让你求一棵最小权的恰好有\(need\)条白色边的生成树。
题目保证有解。
输入输出格式
输入格式
第一行\(V,E,need\)分别表示点数,边数和需要的白色边数。
接下来\(E\)行
每行\(s,t,c,col\)表示这边的端点(点从\(0\)开始标号),边权,颜色(\(0\)白色\(1\)黑色)。
输出格式
一行表示所求生成树的边权和。
输入输出样例
输入样例#1
2 2 1
0 1 1 1
0 1 2 0
输出样例#1
2
说明
\(0:V<=10\)
\(1,2,3:V<=15\)
\(0,..,19:V<=50000,E<=100000\)
所有数据边权为\([1,100]\)中的正整数。
\(By\) \(WJMZBMR\)
题解
\(WQS\)二分入门题。
关于\(WQS\)二分,可以把这里作为教程,做一下这里的练习题。
回到这个题,我们可以将\(WQS\)二分和\(MST\)(最小生成树)配合解题。
考虑给每一条白边减去一个值\(cost\),使得在坐标轴上横坐标为\(k\)的点纵坐标最大。
二分\(cost\),将每一条白边减去\(cost\),最后用\(Kruskal\)跑一遍\(MST\)即可。
代码
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <cctype>
using namespace std;
inline int gi()
{
int f = 1, x = 0; char c = getchar();
while (c < '0' || c > '9') { if (c == '-') f = -1; c = getchar();}
while (c >= '0' && c <= '9') { x = x * 10 + c - '0'; c = getchar();}
return f * x;
}
int n, m, k, sum, tot, cnt, ans;
int uu[100003], vv[100003], ww[100003], cc[100003], fa[100003];
struct Node
{
int u, v, w, c;
} e[100003];
inline bool cmp(Node x, Node y)//将点权排序
{
if (x.w == y.w) return x.c > y.c;
else return x.w < y.w;
}
int getf(int u)//并查集找祖先
{
if (fa[u] == u) return u;
return fa[u] = getf(fa[u]);
}
inline bool check(int t)//二分的check
{
tot = cnt = 0;//将计数器清零
for (int i = 1; i <= n; i++) fa[i] = i;//初始化祖先
for (int i = 1; i <= m; i++)//存图
{
e[i].u = uu[i], e[i].v = vv[i], e[i].c = cc[i], e[i].w = ww[i];
if (!cc[i]) e[i].w = e[i].w - t;//白边减去边权
}
sort(e + 1, e + 1 + m, cmp);//将边排序
for (int i = 1; i <= m; i++)//跑一遍MST
{
int p = getf(e[i].u), q = getf(e[i].v);
if (p != q)
{
fa[p] = q, tot = tot + e[i].w;
if (!e[i].c) ++cnt;//是白色边就计数器+1
}
}
return cnt <= k;//符合题目条件
}
int main()
{
n = gi(), m = gi(), k = gi();
for (int i = 1; i <= m; i++)
{
uu[i] = gi(), vv[i] = gi(), ww[i] = gi(), cc[i] = gi();
++uu[i], ++vv[i];
}//输入
int Left = -105, Right = 105;
while (Left <= Right)//开始二分
{
int mid = (Left + Right) >> 1;
if (check(mid))
{
Left = mid + 1, ans = tot + k * mid;//注意ans的存储
}
else
{
Right = mid - 1;
}
}
printf("%d\n", ans);//输出ans
return 0;//结束
}
最新文章
- View and Data API Tips: Hide elements in viewer completely
- mybatis调用视图和存储过程
- android获取系统通讯录
- 【Leetcode】【Hard】Valid Number
- IOS开发实现录音功能
- Unix网络编程(3)&mdash;&mdash;C/S模型几种情况
- 使用 Acegi 保护 Java 应用程序
- 微软企业库Microsoft Enterprise Library的相关文章链接
- 如何调整iMindMap打印设置
- 【CF 189A Cut Ribbon】dp
- leetcode medium
- 轻松学JVM(二)&mdash;&mdash;内存模型、可见性、指令重排序
- Cannot resolve taglib with uri http://java.sun.com/jsp/jstl/core
- Centos7快速安装docker
- v4l2框架分析
- 人力资源项目中 add_account.php
- hdu 5877 Weak Pair (Treap)
- Confluence 6 如何配置快速导航的同时查找数量
- JsonServer服务环境搭建
- BZOJ 4004 [JLOI2015]装备购买 | 线性基
热门文章
- MYSQL碰到The total number of locks exceeds the lock table size 问题解决记录
- 使用PHP链接MySQL
- url跳转问题
- OpenGL 编程指南 (4)
- Codeforces Round #601 (Div. 2) A	 Changing Volume
- EF简单的CodeFirst示例(自己创建数据库,不使用数据迁移)
- 安装破解pycharm2018版
- DeepLearningDTU: Building a RNN step by step
- python3练习100题——006
- c++指针实例