Luogu 3665 [USACO17OPEN]Switch Grass 切换牧草
BZOJ 4777 被权限了。
这道题的做法看上去不难,但是感觉自己yy不出来。
首先是两个结论:
1、答案一定是连接着两个异色点的一条边。
2、答案一定在最小生成树上。
感觉看到了之后都比较显然,自己想……算了吧……想不出来的……
那么我们可以对每一个点开一个以颜色为下标的线段树,对这棵树存一存它儿子的颜色到它的距离,然后在叶子结点维护一个$multiset$,把所有颜色相同的点都丢进去,然后维护一个最小值$lst_x = min(query(1, 1, k, 1, col_x - 1), query(1, 1, n, col_x + 1, k))$。
再全局维护一个$multiset$,每一次更改颜色的时候加入删除就好了。
时间复杂度$O(nlog^2n)$。
在Luogu上需要氧气。
Code:
#include <cstdio>
#include <cstring>
#include <set>
#include <algorithm>
using namespace std;
typedef long long ll; const int N = 2e5 + ;
const int inf = << ; int n, m, k, qn, col[N], ufs[N], tot = , head[N];
int lst[N], idCnt = , id[N * ], fa[N], eVal[N];
multiset <int> ans, w[N * ]; struct Pathway {
int u, v, val; friend bool operator < (const Pathway &x, const Pathway &y) {
return x.val < y.val;
} } pat[N]; struct Edge {
int to, nxt, val;
} e[N << ]; inline void add(int from, int to, int val) {
e[++tot].to = to;
e[tot].val = val;
e[tot].nxt = head[from];
head[from] = tot;
} inline void addEdge(int x, int y, int v) {
add(x, y, v), add(y, x, v);
} inline void read(int &X) {
X = ; char ch = ; int op = ;
for(; ch > '' || ch < ''; ch = getchar())
if(ch == '-') op = -;
for(; ch >= '' && ch <= ''; ch = getchar())
X = (X << ) + (X << ) + ch - ;
X *= op;
} int find(int x) {
return x == ufs[x] ? x : ufs[x] = find(ufs[x]);
} inline void kruskal() {
sort(pat + , pat + + m);
int cnt = ;
for(int i = ; i <= n; i++) ufs[i] = i;
for(int i = ; i <= m; i++) {
int u = find(pat[i].u), v = find(pat[i].v);
if(u == v) continue;
ufs[u] = v;
addEdge(pat[i].u, pat[i].v, pat[i].val);
++cnt;
if(cnt >= n - ) break;
}
} inline int min(int x, int y) {
return x > y ? y : x;
} inline void chkMin(int &x, int y) {
if(y < x) x = y;
} namespace SegT {
struct Node {
int lc, rc, mn;
} s[N * ]; int root[N], nodeCnt = ; #define lc(p) s[p].lc
#define rc(p) s[p].rc
#define mn(p) s[p].mn
#define mid ((l + r) >> 1) inline void up(int p) {
mn(p) = min(mn(lc(p)), mn(rc(p)));
} void ins(int &p, int l, int r, int x, int v) {
if(!p) mn(p = ++nodeCnt) = inf;
if(l == r) {
if(!id[p]) id[p] = ++idCnt;
w[id[p]].insert(v);
mn(p) = *(w[id[p]].begin());
return;
} if(x <= mid) ins(lc(p), l, mid, x, v);
else ins(rc(p), mid + , r, x, v);
up(p);
} void del(int &p, int l, int r, int x, int v) {
if(l == r) {
w[id[p]].erase(w[id[p]].find(v));
if(w[id[p]].empty()) mn(p) = inf;
else mn(p) = *(w[id[p]].begin());
return;
} if(x <= mid) del(lc(p), l, mid, x, v);
else del(rc(p), mid + , r, x, v);
up(p);
} int query(int p, int l, int r, int x, int y) {
if(!p) return inf;
if(x <= l && y >= r) return mn(p); int res = inf;
if(x <= mid) chkMin(res, query(lc(p), l, mid, x, y));
if(y > mid) chkMin(res, query(rc(p), mid + , r, x, y));
return res;
} } using namespace SegT; inline int queryMin(int x) {
int res = inf;
if(col[x] != ) chkMin(res, query(root[x], , k, , col[x] - ));
if(col[x] != k) chkMin(res, query(root[x], , k, col[x] + , k));
return res;
} void dfs(int x, int fat) {
fa[x] = fat;
for(int i = head[x]; i; i = e[i].nxt) {
int y = e[i].to;
if(y == fat) continue;
eVal[y] = e[i].val;
ins(root[x], , k, col[y], e[i].val);
dfs(y, x);
}
if(root[x]) {
lst[x] = queryMin(x);
ans.insert(lst[x]);
}
} int main() {
// freopen("2.in", "r", stdin);
// freopen("my.out", "w", stdout); read(n), read(m), read(k), read(qn);
for(int i = ; i <= m; i++)
read(pat[i].u), read(pat[i].v), read(pat[i].val);
kruskal(); for(int i = ; i <= n; i++) read(col[i]); mn() = inf;
dfs(, ); for(int x, v; qn--; ) {
read(x), read(v);
int pre = col[x];
col[x] = v;
if(root[x]) {
ans.erase(ans.find(lst[x]));
lst[x] = queryMin(x);
ans.insert(lst[x]);
}
if(fa[x]) {
ans.erase(ans.find(lst[fa[x]]));
del(root[fa[x]], , k, pre, eVal[x]);
ins(root[fa[x]], , k, v, eVal[x]);
lst[fa[x]] = queryMin(fa[x]);
ans.insert(lst[fa[x]]);
} printf("%d\n", *(ans.begin()));
}
return ;
}
Luogu 3665 [USACO17OPEN]Switch Grass 切换牧草的更多相关文章
- P3665 [USACO17OPEN]Switch Grass
题目描述 N个点M条边的无向图,每个点有一个初始颜色,每次改变一个点的颜色,求改变后整张图上颜色不同的点之间的距离最小值. 思路 考虑整张图的距离最小值一定是一条边,而不可能是一条路径,那么显然这条边 ...
- BZOJ 4777: [Usaco2017 Open]Switch Grass
4777: [Usaco2017 Open]Switch Grass Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 46 Solved: 10[Su ...
- js使用Switch达到切换不同颜色的效果
实现的效果,点击哪个,哪个变颜色,效果如下. 代码如下: <!DOCTYPE html> <html> <head> <meta charset=" ...
- BZOJ 4777 Usaco2017 Open Switch Grass Kruskal+替罪羊树+权值线段树
这道题首先可以看出答案一定是一条边,而且答案一定在最小生成树上,那么我们就可以在这个最小生成树上维护他与异色儿子的边最小值,所以我们就可以已通过Kruskal和一棵平衡树来解决,时间复杂度是O(n*l ...
- BZOJ 3887/Luogu P3119: [Usaco2015 Jan]Grass Cownoisseur (强连通分量+最长路)
分层建图,反向边建在两层之间,两层内部分别建正向边,tarjan缩点后,拓扑排序求一次1所在强连通分量和1+n所在强联通分量的最长路(长度定义为路径上的强联通分量内部点数和).然后由于1所在强连通分量 ...
- BZOJ4777 [Usaco2017 Open]Switch Grass[最小生成树+权值线段树套平衡树]
标题解法是吓人的. 图上修改询问,不好用数据结构操作.尝试转化为树来维护.发现(不要问怎么发现的)最小生成树在这里比较行得通,因为最近异色点对一定是相邻的(很好想),所以只要看最短的一条两端连着异色点 ...
- switch host 切换本地host
百度网盘提取地址 提取码: 753r 下载后放到软件目录即可使用
- [bzoj4777]Switch Grass
结论:最短路径一定是单独的一条边且在最小生成树上,可以用反证法证明.那么求出最小生成树,对于每一个点建立一棵权值线段树,再对每一个权值线段树上的叶子节点开一个multiset,维护所有儿子中该种颜色的 ...
- flat ui switch 改变状态而不响应事件
Flat UI是一套精美的扁平风格 UI 工具包,基于 Twitter Bootstrap实现.这套界面工具包含许多基本的和复杂的 UI 部件,例如按钮,输入框,组合按钮,复选框,单选按钮,标签,菜单 ...
随机推荐
- 剑指offer-第四章解决面试题的思路(从上往下打印二叉树)
题目:从上往下打印二叉树的每一个节点,同一层的节点按照从左到右的顺序打印 思路:这是一个层序遍历的问题,因此要借用到队列.我们可以在打印第一个节点的同时将这个节点的左右子节点都放入队列,同样打印左右子 ...
- C# 用wps(api v9) 将word转成pdf
我们不产生代码只是代码的搬运工 我们先来看一段跑不起来的代码 ..各种未将对象应用到实例.. using System; using System.Collections.Generic; usin ...
- Python学习系列(五)(文件操作及其字典)
Python学习系列(五)(文件操作及其字典) Python学习系列(四)(列表及其函数) 一.文件操作 1,读文件 在以'r'读模式打开文件以后可以调用read函数一次性将文件内容全部读出 ...
- (转)TextView 设置背景和文本颜色的问题
在做一个项目,突然遇到如下问题 比如:在color.xml中定义了几个颜色 <color name="white">#FFFFFF</color> < ...
- 配置动态ip为静态ip qq交流总结
修改 /etc/sysconfig/network-scripts/ifcfg-etho 修改dhcp 为 static 修改后的样例 这三个ip该怎么对应 ifconfig 123各自对应 修改/e ...
- java流类,快速统计出字符次数+++
总结:学会给一个合适的命名,不要总是abc..虽然简单,但是不容易看懂,和方便去理解 package com.aini; import java.io.File; import java.io.Fil ...
- Python web框架 Tornado(二)异步非阻塞
异步非阻塞 阻塞式:(适用于所有框架,Django,Flask,Tornado,Bottle) 一个请求到来未处理完成,后续一直等待 解决方案:多线程,多进程 异步非阻塞(存在IO请求): Torna ...
- AngularJS绑定数据
绑定数据总共有三种方式1.{{}}最常用2.ngbind3.ng-model 主要用在input标签
- Linux性能监测:磁盘IO篇
磁盘通常是计算机最慢的子系统,也是最容易出现性能瓶颈的地方,因为磁盘离 CPU 距离最远而且 CPU 访问磁盘要涉及到机械操作,比如转轴.寻轨等.访问硬盘和访问内存之间的速度差别是以数量级来计算的,就 ...
- SSD知识
不管什么接口的SSD,一般都由以下部分组成:主控,Flash,板,壳,品牌.下面本佬就这些部分一一发帖,仅供娱乐参考,不作任何推荐和偏向,有不同见解请直接发表,有任何错误,请直接指正,不为吵架,只为娱 ...