传送门

先考虑树,树是一个二分图。

看到是二分图并且每次是对两边的同色的点反色可以想到转化:让奇数层的点为黑,偶数为白,变成每次可以交换两个点的颜色。

把黑看成 \(-1\),白看成 \(1\),那么求一个子树和,考虑每一条边的贡献可以得到 \(ans=\sum_{i=1}^{n} |sum_i|\)

如果根的 \(sum\) 不为 \(0\),那么肯定是无解的。

对于基环树,先考虑奇环。

断开奇环的一条边 \((u,v)\),变成树,\(u,v\) 肯定是同一边的点。

操作一次 \((u,v)\) 相当于可以两边可以同时新增加白点/黑点,也就是可以把根的 \(sum\) 用这两个点来变成 \(0\),(\(sum\) 必须为偶数)平均分配之后用树的做法即可。

考虑偶环。

断开偶环的一条边 \((u,v)\),变成树,\(u,v\) 肯定不是同一边的点。

操作一次 \((u,v)\) 相当于是让左右的点走了一次捷径。

设用的次数为 \(x\)。

如果一个点同时包含或不包含 \(u,v\) 两个点,那么 \(sum\) 一定不变。

否则加上或者减去 \(x\)。

相当是是要求 \(|x|+\sum |sum_i-x| + \sum |sum_i+x|\)

经典问题,排序之后取中位数即可。

# include <bits/stdc++.h>
using namespace std;
typedef long long ll; const int maxn(1e5 + 5); struct Edge { int to, next; }; int n, m, first[maxn], cnt, sum[maxn], fa[maxn], dsu[maxn], deep[maxn], a, b, ans, val[maxn];
Edge edge[maxn << 1]; inline int Find(int x) { return (dsu[x] ^ x) ? dsu[x] = Find(dsu[x]) : x; } inline void Add(int u, int v) {
edge[cnt] = (Edge){v, first[u]}, first[u] = cnt++;
edge[cnt] = (Edge){u, first[v]}, first[v] = cnt++;
} void Dfs(int u, int ff, int d) {
int e, v;
sum[u] = d;
for (e = first[u]; ~e; e = edge[e].next)
if ((v = edge[e].to) ^ ff) {
deep[v] = deep[u] + 1;
fa[v] = u, Dfs(v, u, -d);
sum[u] += sum[v];
}
} int main() {
int i, u, v, len = 1;
memset(first, -1, sizeof(first));
scanf("%d%d", &n, &m);
if (n & 1) return puts("-1"), 0;
for (i = 1; i <= n; ++i) dsu[i] = i;
for (i = 1; i <= m; ++i) {
scanf("%d%d", &u, &v);
if (Find(u) ^ Find(v)) Add(u, v), dsu[Find(u)] = Find(v);
else a = u, b = v;
}
if (!a) {
Dfs(1, 0, 1);
if (sum[1]) return puts("-1"), 0;
}
else {
Dfs(a, 0, 1);
if (deep[b] & 1) {
if (sum[a]) return puts("-1"), 0;
for (i = b; i ^ a; i = fa[i]) val[++len] = sum[i], sum[i] = 0;
sort(val + 1, val + len + 1), v = val[(len + 1) >> 1];
for (i = 1; i <= len; ++i) ans += abs(val[i] - v);
}
else {
if (sum[a] & 1) return puts("-1"), 0;
v = sum[a] >> 1, ans = abs(v), sum[a] = 0;
for (i = b; i ^ a; i = fa[i]) sum[i] -= v;
}
}
for (i = 1; i <= n; ++i) ans += abs(sum[i]);
printf("%d\n", ans);
return 0;
}

Atcoder:AGC004F Namori的更多相关文章

  1. 2017国家集训队作业[agc004f]Namori

    2017国家集训队作业[agc004f]Namori 题意: 给你一颗树或环套树,树上有\(N\)个点,有\(M\)条边.一开始,树上的点都是白色,一次操作可以选择一条端点颜色相同的边,使它的端点颜色 ...

  2. AtCoder AGC004F Namori (图论)

    题目链接 https://atcoder.jp/contests/agc004/tasks/agc004_f 题解 神仙题.. 首先考虑树的情况,树是二分图,因此假设我们对二分图进行黑白染色,那么操作 ...

  3. AtCoder:C - Nuske vs Phantom Thnook

    C - Nuske vs Phantom Thnook https://agc015.contest.atcoder.jp/tasks/agc015_c 题意: n*m的网格,每个格子可能是蓝色, 可 ...

  4. AGC004F Namori 树形DP、解方程(?)

    传送门 因为不会列方程然后只会树上的,被吊打了QAQ 不难想到从叶子节点往上计算答案.可以考虑到可能树上存在一个点,在它的儿子做完之后接着若干颜色为白色的儿子,而当前点为白色,只能帮助一个儿子变成黑色 ...

  5. [agc004f]Namori 贪心

    Description ​ 现在给你一张NN个点MM条边的连通图,我们保证N−1≤M≤NN−1≤M≤N,且无重边和自环. ​ 每一个点都有一种颜色,非黑即白.初始时,所有点都是白色的. ​ 想通过执行 ...

  6. [AGC004F] Namori

    Description 现在给你一张N个点M条边的连通图,我们保证N−1≤M≤N,且无重边和自环. 每一个点都有一种颜色,非黑即白.初始时,所有点都是白色的. "全"想通过执行若干 ...

  7. AtCoder刷题记录

    构造题都是神仙题 /kk ARC066C Addition and Subtraction Hard 首先要发现两个性质: 加号右边不会有括号:显然,有括号也可以被删去,答案不变. \(op_i\)和 ...

  8. 贪心/构造/DP 杂题选做Ⅲ

    颓!颓!颓!(bushi 前传: 贪心/构造/DP 杂题选做 贪心/构造/DP 杂题选做Ⅱ 51. CF758E Broken Tree 讲个笑话,这道题是 11.3 模拟赛的 T2,模拟赛里那道题的 ...

  9. NOIp模拟赛二十九

    又是受虐的一天呢~接下来四天都要打模拟赛QAQ 今日分数:0(100)+100+0=100 A题O(读入)结论题判断结果时没return 0被subtask卡成0分,喜提fstQAQ,B题DP,C题不 ...

随机推荐

  1. MapReducer

    MapReducer    概述        是一个分布式的计算框架(编程模型),最初由由谷歌的工程师开发,基于GFS的分布式计算框架.后来Cutting根据<Google Mapreduce ...

  2. 让PETSc跑得再快一些

    最近做了一个使用PETSc来求解线性方程组(Ax=b)的项目,把其中遇到的一些坑和解决方法记录下来.本文不介绍PETSc如何入门,而是给出一些能让PETSc运行得更快的编程细节.开始我只是简单地修改P ...

  3. spring + mybatis 存取clob

    存的时候会比较麻烦,需要使用select for update的方式更新数据,如果原来没有这一条数据,还需要先新增,新增的时候需要将clob字段存为oracle.sql.CLOB.empty_lob( ...

  4. (转)面向对象(深入)|python描述器详解

    原文:https://zhuanlan.zhihu.com/p/32764345 https://www.cnblogs.com/aademeng/articles/7262645.html----- ...

  5. ubuntu 下配置elasticSearch

    配置JAVA环境 配置jdk 上官网下载x64的和ubuntu匹配的jdk 找到usr/java ,解压下载的文件  tar –xzvf  文件.tar.gz Vim /etc/source  添加配 ...

  6. postgresql的启停和创建

    一.启停方法 两种方法 1.直接运行postgres进程启动: 2.使用pg_ctl命令启动 postgres -D /home/osdba/pgdata  & 二.停止数据库的三种模式 sm ...

  7. Linux系统修改防火墙配置

    防火墙配置文件位置 /etc/sysconfig/iptables 需要开放端口,请在里面添加一条内容即可: 1 -A RH-Firewall-1-INPUT -m state --state NEW ...

  8. B+树原理及mysql的索引分析

    转自:http://blog.csdn.net/qq_23217629/article/details/52512041 B+/-Tree原理 B-Tree介绍 B-Tree是一种多路搜索树(并不是二 ...

  9. Java 并发编程常识 —— by 梁飞

    参考 :梁飞 并发编程常识

  10. Docker基础-Docker数据管理

    1.数据卷 数据卷是一个可供容器使用的特殊目录,它将主机操作系统目录直接映射进容器,类似于Linux中的mount操作. 数据卷可以提供很多有用的特性: 1.数据卷可以在容器之间共享和重用,容器间传递 ...