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 切换牧草的更多相关文章

  1. P3665 [USACO17OPEN]Switch Grass

    题目描述 N个点M条边的无向图,每个点有一个初始颜色,每次改变一个点的颜色,求改变后整张图上颜色不同的点之间的距离最小值. 思路 考虑整张图的距离最小值一定是一条边,而不可能是一条路径,那么显然这条边 ...

  2. BZOJ 4777: [Usaco2017 Open]Switch Grass

    4777: [Usaco2017 Open]Switch Grass Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 46  Solved: 10[Su ...

  3. js使用Switch达到切换不同颜色的效果

    实现的效果,点击哪个,哪个变颜色,效果如下. 代码如下: <!DOCTYPE html> <html> <head> <meta charset=" ...

  4. BZOJ 4777 Usaco2017 Open Switch Grass Kruskal+替罪羊树+权值线段树

    这道题首先可以看出答案一定是一条边,而且答案一定在最小生成树上,那么我们就可以在这个最小生成树上维护他与异色儿子的边最小值,所以我们就可以已通过Kruskal和一棵平衡树来解决,时间复杂度是O(n*l ...

  5. BZOJ 3887/Luogu P3119: [Usaco2015 Jan]Grass Cownoisseur (强连通分量+最长路)

    分层建图,反向边建在两层之间,两层内部分别建正向边,tarjan缩点后,拓扑排序求一次1所在强连通分量和1+n所在强联通分量的最长路(长度定义为路径上的强联通分量内部点数和).然后由于1所在强连通分量 ...

  6. BZOJ4777 [Usaco2017 Open]Switch Grass[最小生成树+权值线段树套平衡树]

    标题解法是吓人的. 图上修改询问,不好用数据结构操作.尝试转化为树来维护.发现(不要问怎么发现的)最小生成树在这里比较行得通,因为最近异色点对一定是相邻的(很好想),所以只要看最短的一条两端连着异色点 ...

  7. switch host 切换本地host

    百度网盘提取地址 提取码: 753r 下载后放到软件目录即可使用

  8. [bzoj4777]Switch Grass

    结论:最短路径一定是单独的一条边且在最小生成树上,可以用反证法证明.那么求出最小生成树,对于每一个点建立一棵权值线段树,再对每一个权值线段树上的叶子节点开一个multiset,维护所有儿子中该种颜色的 ...

  9. flat ui switch 改变状态而不响应事件

    Flat UI是一套精美的扁平风格 UI 工具包,基于 Twitter Bootstrap实现.这套界面工具包含许多基本的和复杂的 UI 部件,例如按钮,输入框,组合按钮,复选框,单选按钮,标签,菜单 ...

随机推荐

  1. JDBC 1 利用Statement对数据库进行增删改查

    准备工作 1新建po类:User private int id; private String name; private String pwd; set,get方法省略 2  新建UserDao类, ...

  2. 文件上传smart

    package com.bdqn.zhp.util; import java.text.SimpleDateFormat; import java.util.Date; import javax.se ...

  3. 为什么新生代内存需要有两个Survivor区?

    对于常见的GC算法,我们都应该知道,例如:标记清除算法.复制算法.标记整理算法等.标记清除算法由于回收之后存在大量的内存碎片,存在效率和空间问题!为了解决效率问题,引出了复制算法!熟悉GC算法的小伙伴 ...

  4. Java-Runoob:Java 教程

    ylbtech-Java-Runoob:Java 教程 1.返回顶部 1. Java 教程 Java 是由Sun Microsystems公司于1995年5月推出的高级程序设计语言. Java可运行于 ...

  5. 1107 Social Clusters

    题意:给出n个人(编号为1~n)以及每个人的若干个爱好,把有一个或多个共同爱好的人归为一个集合,问共有多少个集合,每个集合里有多少个人? 思路:典型的并查集题目.并查集的模板init()函数,unio ...

  6. mysql 无意重启 [Note] /usr/sbin/mysqld: Normal shutdown

    情况: 今早发现,昨天下午安装的4台mysql服务器,突然出现,由于在shell窗口 (root@localhost:mysql.sock) [(none)]> 190102 18:12:16 ...

  7. mysql数据恢复 insert\update\delete 工具MyFlash

    一.简介MyFlash是由美团点评公司技术工程部开发维护的一个回滚DML操作的工具.该工具通过解析v4版本的binlog,完成回滚操作.相对已有的回滚工具,其增加了更多的过滤选项,让回滚更加容易. 该 ...

  8. django-settings.py配置

    django settings 详细资料 ############ 开始项目 python3.5 pip -m install django==1.11.7 指定版本安装 pip3 install d ...

  9. class_create(),class_device_create()创建/dev/xxx 名字

    在刚开始写Linux设备驱动程序的时候,很多时候都是利用mknod命令手动创建设备节点,实际上Linux内核为我们提供了一组函数,可以用来在模块加载的时候自动在/dev目录下创建相应设备节点,并在卸载 ...

  10. 什么是闭包?在js中的作用是什么?

    闭包就是讲函数内部生成的变量保存到内存中,进行下次调用:也可以说函数外不可以调用函数内部的变量: 当函数内部返回一个函数时,闭包搭建了方法内部与方法外部的桥梁,使得外部也可以任意的获取到方法内部的资源 ...