考虑树的情况,将其以任意一点为根建树

对于每一个节点,考虑其要与父亲操作几次才能使子树内均为黑色,这可以用形如$(0/1,x)$的二元组来描述,其中0/1即表示其要求操作时父亲是白色/黑色且要操作$x$次

考虑一个叶子,其二元组显然为$(0,1)$,接下来每一个点即可以交替将儿子中的$(0/1,x)$变为$(0/1,x-1)$

(先以白色点操作$(0,x)$,操作后变为黑色,再以黑色点操作$(1,x)$……)

最终,将剩下的合并(第一元必然相同,第二元直接相加),那么该节点即需在另一个状态下与其父亲操作,另外最后还需要额外以白色操作一次

换言之,假设合并后的二元组为$(p,x)$(其中$x\ge 0$,且若$x=0$则不妨令$p=1$),那么根据上述分析该点上的二元组即为$\begin{cases}(1,x-1)&(p=0)\\(0,x+1)&(p=1)\end{cases}$

关于答案,若根节点上的二元组第二维不为0显然无解,否则考虑每一个节点与父亲操作的次数,即为该点上二元组的第二维,显然将对所有节点求和即可

时间复杂度为$o(n)$,可以通过

事实上,上述二元组可以直接用一个整数描述,即将$(0/1,x)$分别看作$\pm x$,则转移即$g_{i}=1-\sum_{son}g_{son}$,最终答案也即$\sum_{i}|g_{i}|$(特别的,若$g_{rt}\ne 0$则无解,其中$rt$为根),两者等价性显然

考虑基环树的情况,将整个基环当作根,并删去基环上的边后得到若干棵子树(仍以基环上的点为根),对每一棵子树仍用上述方式计算得到根的$g_{i}$,最终即可得到一个长为$l$(其中$l$为环长)的序列

记该序列为$a_{i}$,此时问题即将相邻两个$a_{i}$(注意首尾也相连)同时$\pm 1$,并使得最终$a_{i}$均为0

若$\sum_{i=1}^{l}a_{i}\not\equiv 0(mod\ 2)$显然无解(操作不改变奇偶性),否则对$l$的奇偶性分类讨论:

1.若$l$为奇数,考虑奇偶数位的差值,除了首尾操作以外不会改变该值,而首尾操作恰会使该值$\pm 2$,由此不难确定首尾操作形式即次数,进而操作后将首尾断开,从前往后依次使得$a_{i}$为0即可(最终$a_{l}$一定为0)

2.若$l$为偶数,注意到奇数位和偶数位差值不变,因此初始两者必须相同

枚举首尾操作使得$a_{1}$和$a_{l}$同时减$z$,之后即将首尾断开并从前往后依次使得$a_{i}$为0

不难得到(首尾断开后)$a_{i}$清0的代价即$\sum_{i=1}^{l}\abs{\sum_{j=1}^{i}(-1)^{i-j}a_{j}}$,对每个$i$预处理出该值为$S_{i}$(减去$z$前),那么修改的影响即将奇数项减去$z$、偶数项(除$l$以外)加上$z$

将$S_{i}$偶数项取相反数(奇数项不变)得到$S'_{i}$,此时即求$|S_{l}|+\min_{z}\left(\sum_{i=1}^{l-1}|S'_{i}-z|+|z|\right)$

最后一项可以看作$|0-z|$,那么显然取$z$为$S'_{i}$(其中$1\le i\le l-1$)和0的中位数即可

时间复杂度也为$o(n)$,可以通过

 1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 100005
4 #define ll long long
5 vector<int>v0,v[N];
6 int n,m,x,y,st[N],vis[N];
7 ll sum,ans,a[N],g[N];
8 void dfs(int k,int fa){
9 st[++st[0]]=k,vis[k]=1;
10 for(int i=0;i<v[k].size();i++)
11 if (v[k][i]!=fa){
12 if (!vis[v[k][i]])dfs(v[k][i],k);
13 else{
14 if (vis[v[k][i]]==2)continue;
15 for(int j=st[0];st[j]!=v[k][i];j--)v0.push_back(st[j]);
16 v0.push_back(v[k][i]);
17 }
18 }
19 st[0]--,vis[k]=2;
20 }
21 void dp(int k){
22 vis[k]=g[k]=1;
23 for(int i=0;i<v[k].size();i++)
24 if (!vis[v[k][i]]){
25 dp(v[k][i]);
26 g[k]-=g[v[k][i]];
27 }
28 }
29 int main(){
30 scanf("%d%d",&n,&m);
31 for(int i=1;i<=m;i++){
32 scanf("%d%d",&x,&y);
33 v[x].push_back(y);
34 v[y].push_back(x);
35 }
36 dfs(1,0);
37 memset(vis,0,sizeof(vis));
38 if (v0.empty()){
39 dp(1);
40 if (g[1])printf("-1\n");
41 else{
42 for(int i=1;i<=n;i++)ans+=abs(g[i]);
43 printf("%lld\n",ans);
44 }
45 return 0;
46 }
47 int l=v0.size();
48 for(int i=0;i<l;i++)vis[v0[i]]=2;
49 for(int i=0;i<l;i++){
50 dp(v0[i]);
51 a[i+1]=g[v0[i]];
52 }
53 for(int i=1;i<=n;i++)ans+=abs(g[i]);
54 for(int i=1;i<=l;i++){
55 ans-=abs(a[i]);
56 if (i&1)sum+=a[i];
57 else sum-=a[i];
58 }
59 if (sum&1){
60 printf("-1\n");
61 return 0;
62 }
63 if (l&1){
64 ans+=(abs(sum)>>1);
65 a[1]-=(sum>>1),a[l]-=(sum>>1);
66 for(int i=1;i<l;i++){
67 ans+=abs(a[i]);
68 a[i+1]-=a[i];
69 }
70 printf("%lld\n",ans);
71 return 0;
72 }
73 if (sum){
74 printf("-1\n");
75 return 0;
76 }
77 for(int i=1;i<=l;i++)a[i]-=a[i-1];
78 ans+=abs(a[l]),a[l]=0;
79 for(int i=2;i<=l;i+=2)a[i]=-a[i];
80 sort(a+1,a+l+1);
81 for(int i=1;i<=l;i++)
82 if (i<=(l>>1))ans+=a[l>>1]-a[i];
83 else ans+=a[i]-a[l>>1];
84 printf("%lld\n",ans);
85 return 0;
86 }

[atAGC004F]Namori的更多相关文章

  1. [Arc079F] Namori Grundy

    [Arc079F] Namori Grundy 题目大意: 一个有向弱联通环套树. 一个点的sg值等于出边连向点的sg值的mex. 试问是否有办法给每个点分配sg值? 试题分析 题目大意把一些难点跳过 ...

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

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

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

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

  4. 洛谷AT2046 Namori(思维,基环树,树形DP)

    洛谷题目传送门 神仙思维题还是要写点东西才好. 树 每次操作把相邻且同色的点反色,直接这样思考会发现状态有很强的后效性,没办法考虑转移. 因为树是二分图,所以我们转化模型:在树的奇数层的所有点上都有一 ...

  5. AT2046 Namori 图论

    正解: 解题报告: 传送门! 首先看数据范围可以发现要么是棵树要么是个奇环要么是个偶环 然后就分类讨论分别看下这几个情况 首先是棵树的 首先可以想到树的情况就是个二分图,所以不妨把颜色重定义,让奇数层 ...

  6. Atcoder:AGC004F Namori

    传送门 先考虑树,树是一个二分图. 看到是二分图并且每次是对两边的同色的点反色可以想到转化:让奇数层的点为黑,偶数为白,变成每次可以交换两个点的颜色. 把黑看成 \(-1\),白看成 \(1\),那么 ...

  7. 【ARC079F】Namori Grundy

    Description 题目链接 大意:给一张基环外向树.要求给每一个点确定一个值,其值为所有后继点的\(\text{mex}\).求是否存在确定权值方案. Solution 首先,对于叶子节点,其权 ...

  8. [arc079f] Namori Grundy 分类讨论

    Description 给给全有一个NN个点NN条边的有向图,点的的编号从11到NN 给给全的图有NN条边,形如:(p1,1),(p2,2),...,(pN,N)(p1,1),(p2,2),...,( ...

  9. 【agc004F】Namori

    Portal -->agc004F Solution  好神仙的转化qwq ​  首先我们可以先考虑\(m=n-1\)的情况下,也就是树的情况下要怎么做  我们可以将这个问题转化一下:我们对这颗 ...

随机推荐

  1. mysql从零开始之MySQL 删除数据库

    MySQL 删除数据库 使用普通用户登陆 MySQL 服务器,你可能需要特定的权限来创建或者删除 MySQL 数据库,所以我们这边使用 root 用户登录,root 用户拥有最高权限. 在删除数据库过 ...

  2. Java 开发最容易写的 10 个bug

    原文链接:10 个让人头疼的 bug 那个谁,今天又写 bug 了,没错,他说的好像就是我...... 作为 Java 开发,我们在写代码的过程中难免会产生各种奇思妙想的 bug ,有些 bug 就挺 ...

  3. Django对表单进行增删改查

    查 首先在url中写好路径 其次在后面参数的views里写函数类xxxxxxx的基本逻辑 定义一个函数xxxxxxx,继承request,注意这个request对数据库操作结果都会存放在request ...

  4. 题解 [BJOI2019]奥术神杖

    题目传送门 题目大意 给出一个残缺的字符串,每个位置都 \(\in[0,9]\).有 \(m\) 中贡献,即 \(s,k\),表示该字符串中没出现一次 \(s\),贡献便乘上 \(k\).最后对贡献求 ...

  5. iOS自动化之WDA(WebDriverAgent)安装

    1.WDA介绍 WebDriverAgent 在 iOS 端实现了一个 WebDriver server ,借助这个 server 我们可以远程控制 iOS 设备.你可以启动.杀死应用,点击.滚动视图 ...

  6. MyBatis 中实现SQL语句中in的操作 (11)

    MyBatis 中实现SQL语句中in的操作 概括:应用myBatis实现SQL查询中IN的操作 1.数据库结构及其数据 2.mapper.xml文件 <?xml version="1 ...

  7. 【转载】[经验] 嵌入式stm32实用的排序算法 - 交换排序

    Ⅰ.写在前面 前面写了关于ADC采集电压的文章,大家除了求平均的方式来处理采样值,还有没有使用到其他的方式来处理采集值呢? 在某些情况下就需要对一组数据进行排序,并提取头特定的数据出来使用. 排序的应 ...

  8. 【二食堂】Beta - 事后分析

    事后分析 设想和目标 我们的软件要解决什么问题?是否定义得很清楚?是否对典型用户和典型场景有清晰的描述? Beta阶段我们首先要对文本标注方式进行优化,其次时添加好友系统,实现邀请好友共同标注的功能. ...

  9. [对对子队]会议记录5.25(Scrum Meeting11)

    今天已完成的工作 吴桐雨 ​ 工作内容:设计第10.11关 ​ 相关issue:设计额外关卡 ​ 相关签入:level10 level11 吴昭邦 ​ 工作内容:写测试代码 ​ 相关issue:测试初 ...

  10. mysql分表之后怎么平滑上线?

    分表的目的 项目开发中,我们的数据库数据越来越大,随之而来的是单个表中数据太多.以至于查询数据变慢,而且由于表的锁机制导致应用操作也受到严重影响,出现了数据库性能瓶颈. 当出现这种情况时,我们可以考虑 ...