对树上的路径进行操作是十分难处理的事情。一开始的思路主要针对于\(a_i<=15\)这一特殊性质上。于是考虑了\(a_i<=1\)的情况,然而除了糊出一个适用范围极小的结论外,并没有什么用。

于是我瞄了一眼题解。令每一个点的值为所有与它相邻的边的权值的异或和。那么,我们发现,对于路径上的点,如果它不是端点,那么有两条与它相邻的的边异或上了相同的值,它的值不变;否则,它的值异或上这个值。并且,容易证明所有边权为零与所有点权为零是等价的。这样,各个结点的值都是无关的,树的结构是无意义的。问题转化成了在一个数列中,每次选取两个数异或上一个相同的值,以最少的操作次数使得数列中所有数为零。

然后,我们发现,对于每一次操作,所选取的数的异或和是不变的。假设这个数列是一个\(n\)个节点的图,而每一次操作都是往里面连边,那么最终每一个联通分量内的数异或和为零。并且,对于所有异或和为零的联通分量,一定存在操作次数为其大小减一的方案,即一个个异或过来,这正对应树的边数。那么,我们得到答案就是\(n-\)联通分量数。

那么,我们可以贪心地把所有值为零的结点单独分为一个联通分量,把两个值相同的结点分为一个联通分量。那么,一共就只有15种取值,每种取值最多1个,总状态数是\(2^{15}\)。于是可以状压dp,转移时我们枚举子集就可以了。

时间复杂度\(O(3^{15})\)。

#include <bits/stdc++.h>
using namespace std;
const int TOT = 1 << 15, N = 100010;
int dp[TOT],n,val[N],cnt[16],ans,sta,res[TOT];
int main() {
int a,b,c,tmp;
scanf("%d",&n);
for (int i = 1 ; i < n ; ++ i) {
scanf("%d%d%d",&a,&b,&c);
a++, b++;
val[a] ^= c;
val[b] ^= c;
}
for (int i = 1 ; i <= n ; ++ i)
++ cnt[val[i]];
ans = n;
ans -= cnt[0];
for (int i = 1 ; i <= 15 ; ++ i)
ans -= cnt[i]/2, sta |= (cnt[i]&1) << i >> 1;
for (int i = 0 ; i < TOT ; ++ i) {
tmp = 0;
for (int j = 1 ; j <= 15 ; ++ j)
if ((i >> j-1)&1) tmp ^= j;
res[i] = tmp;
}
for (int i = 1 ; i < TOT ; ++ i) {
for (int j = i ; j ; j = (j-1) & i)
if (!res[j]) dp[i] = max(dp[i],dp[i^j] + 1);
}
ans -= dp[sta];
printf("%d\n",ans);
return 0;
}

小结:这种代码简单但思想巧妙的题目是十分惊艳的,可惜为数不多。

【做题】apc001_f-XOR Tree——巧妙转化及dp的更多相关文章

  1. 【做题】spoj4060 A game with probability——dp

    赛前做题时忽然发现自己概率博弈类dp很弱,心好慌.(获胜概率或最优解期望) 于是就做了这道题,续了特别久. 一开始列dp式子的时候就花了很长时间,首先搞错了两次,然后忘记了根据上一轮dp值直接确定选什 ...

  2. 【做题】arc070_f-HonestOrUnkind——交互+巧妙思维

    做的第一道交互题-- 首先,有解的一个必要条件是\(a>b\).否则,即当\(a<=b\)时,可以有\(a\)个unkind的人假装自己就是那\(a\)个honest的人.(彼此之间都说是 ...

  3. 【做题】CSA49F - Card Collecting Game——思维&dp

    原文链接 https://www.cnblogs.com/cly-none/p/CSA49F.html 题意:Alice和Bob在玩游戏.有\(n\)种卡牌,每种卡牌有\(b_i\)张,保证\(\su ...

  4. 【做题】TCSRM592 Div1 500 LittleElephantAndPermutationDiv1——计数&dp

    题意:定义函数\(f(A,B) = \sum_{i=1}^n \max(A_i,B_i)\),其中\(A\)和\(B\)都是长度为\(n\)的排列.给出\(n\)和\(k\),问有多少对\((A,B) ...

  5. AtCoder Grand Contest 11~17 做题小记

    原文链接https://www.cnblogs.com/zhouzhendong/p/AtCoder-Grand-Contest-from-11-to-20.html UPD(2018-11-16): ...

  6. AtCoder Grand Contest 1~10 做题小记

    原文链接https://www.cnblogs.com/zhouzhendong/p/AtCoder-Grand-Contest-from-1-to-10.html 考虑到博客内容较多,编辑不方便的情 ...

  7. (转)poj算法做题顺序

    初期: 一.基本算法: (1)枚举. (poj1753,poj2965) (2)贪心(poj1328,poj2109,poj2586) (3)递归和分治法. (4)递推. (5)构造法.(poj329 ...

  8. 「AGC035C」 Skolem XOR Tree

    「AGC035C」 Skolem XOR Tree 感觉有那么一点点上道了? 首先对于一个 \(n\),若 \(n\equiv 3 \pmod 4\),我们很快能够构造出一个合法解如 \(n,n-1, ...

  9. LCT做题笔记

    最近几天打算认真复习LCT,毕竟以前只会板子.正好也可以学点新的用法,这里就用来写做题笔记吧.这个分类比较混乱,主要看感觉,不一定对: 维护森林的LCT 就是最普通,最一般那种的LCT啦.这类题目往往 ...

随机推荐

  1. keras可视化pydot graphviz问题

    Keras中提供了一个神经网络可视化的函数plot,并可以将可视化结果保存在本地.plot使用方法如下: from keras.utils.visualize_util import plot plo ...

  2. Codeforces Round #323

    div1 C 这题的是给了一个无限循环的子数组,问有多少个 (l,s)l代表起点s代表长度的循环串,使得所有的在原串中的每位都小于等于另外这个串(l<=n,1<=s<n) 像这样,我 ...

  3. C#基础知识整理

    年时,北风吹雁雪纷纷,一条秋裤冻上头.冷的连手都懒得动,就随便翻翻书,也没有更新博客,如今年已过,开始投入到正常的工作状态中,趁现在需求还没有来,把C#基础知识梳理一下,其实一直以来就想这样做的,对于 ...

  4. 编程中的链式调用:Scala示例

    编程中的链式调用与Linux Shell 中的管道类似.Linux Shell 中的管道 ,会将管道连接的上一个程序的结果, 传递给管道连接的下一个程序作为参数进行处理,依次串联起N个实用程序形成流水 ...

  5. Linux Shell入门

    转自:http://www.mamicode.com/info-detail-605431.html

  6. Docker相关知识整理

    一.进入和退出docker容器 使用docker ps查看容器id docker exec -it containerId /bin/bash //进入容器 Ctrl + P + Q //退出容器 二 ...

  7. 转:三值逻辑与NULL的处理方式

    来自:<Microsoft SQL SERVER 2008技术内幕 T-SQL查询>P7 在SQL中谓词(逻辑表达式)的可能值为TRUE.FALSE和UNKNOWN.这就是所谓的三值逻辑, ...

  8. 校正PHP服务器时间不准的问题

    关于怎样解决PHP服务器时间不准的问题,得针对不同的情况进行不同的处理. 下面是经常遇到的情况,及应对办法. 1.PHP服务器时区不对,使用下面代码修正: <?php $timezone = & ...

  9. 关于Vue中的 render: h => h(App) 具体是什么含义?

    render: h => h(App) 是下面内容的缩写: render: function (createElement) { return createElement(App); } 进一步 ...

  10. 在html中引入css、js和jQuery的方法

    在html中引入css代码 在html中插入CSS样式表的方法有三种: 1.外部样式表(External style sheet):即所有的样式单独写在一个.css文件中,在html文件的head部分 ...