JZOJ 3234. 阴阳
阴阳
题面


分析
个人认为是极好的题,很容易写
如果你学点分治是无奈背板的,那就做做这道题,加深你对点分治的理解
一般的,处理树上大规模统计问题,我们分治的关键是找一棵子树的重心
找到分治中心,即新一轮的根节点,然后处理子树节点经过根节点时的答案,接着对子树继续分治下去
那么我们看这题,让黑白的各自变为 \(1\) 或 \(-1\)
合法的路径是这样的:能找到一个分割点使路径两端各自的和分别为 \(0\)
那么我们思考在统计经过根节点的答案时,只有不同子树的两路径合并时或一棵子树中某节点到根距离为 \(0\) 且其路径上有分割点才可能产生答案
也就是说,我们要找的路径和必须为 \(0\) 且有分割点
找路径和为 \(0\) 很简单,记录子树节点到根的距离 \(dis\)(可能为负,所以让他加上 \(n\) 避免出错),两路径合并时,设其两端为 \(x,y\),\(dis[x]+dis[y] = 0\) 是必要条件
重点就是如何判断两路径是否有分割点,是的话我们就可以统计答案
那么我们回到 \(dis[x]+dis[y] = 0\) 这句话
\(dis[y]=-dis[x]\),若 \(dis[y]\) 他到根节点中 值是非第一次出现的,那么可以直接贡献答案
我们只需要记下这类点的个数就好了,即开桶以值为下标记个数,我们称其为二类桶
???那一类桶呢?
既然二类桶是非第一次出现,那一类桶就记第一次出现的个数
我们可以用 \(flag[x]=1\) 表示 \(dis[x]\) 已经出现过
因为是遍历子树是深搜的顺序,所以我们再开个桶标记在他到根节点路径中 \(dis\) 出现的次数,这样就可以很好的判断 \(flag\) 是否需要标记为 \(1\)
那么考虑当前节点如何计算答案
- 若他的 \(dis\) 到根节点是第一次出现,那么他可以直接加上二类桶。注意 \(dis\) 为 \(0\) 的话他又可以加一类桶,因为此时的根可以为路径的分割点
- 若他的 \(dis\) 到根节点不是第一次出现,那么他可以加上一、二类桶,因为他先前的 \(dis\) 到他的距离为 \(0\),即先前的 \(dis\) 可做分割点。当然,此时他 \(dis\) 若为 \(0\) 同理说明他到根也是合法路径,所以答案加 \(1\)。记得标记当前点的 \(flag\) 为 \(1\)
每次统计完一个子树,就要再遍历一遍这个子树更新一类、二类桶的信息
换重心时我们不能直接把数组清 \(0\),为了保证复杂度,我们统计完所有子树的答案后再遍历一遍所有子树,将一类、二类桶的信息退回,\(flag\) 和 \(dis\) 清 \(0\)
完结撒花
\(Code\)
#include<cstdio>
#include<iostream>
using namespace std;
typedef long long LL;
const int N = 1e5 + 5;
int n , buc[N << 1] , buc1[N << 1][3] , flag[N] , dis[N] , h[N] , size , siz[N] , son[N] , rt , use[N] , tot;
LL ans;
struct edge{
int to , nxt , w;
}e[2 * N];
inline void add(int x , int y , int z)
{
e[++tot].nxt = h[x];
e[tot].to = y;
e[tot].w = z;
h[x] = tot;
}
inline void getrt(int x , int fa)
{
son[x] = 0 , siz[x] = 1;
for(register int i = h[x]; i; i = e[i].nxt)
{
int v = e[i].to;
if (v == fa || use[v]) continue;
getrt(v , x);
siz[x] += siz[v];
son[x] = max(son[x] , siz[v]);
}
son[x] = max(son[x] , size - siz[x]);
rt = son[x] < son[rt] ? x : rt;
}
inline void getdis(int x , int fa)
{
for(register int i = h[x]; i; i = e[i].nxt)
{
int v = e[i].to;
if (v == fa || use[v]) continue;
dis[v] = dis[x] + e[i].w;
getdis(v , x);
}
}
inline void fill(int x , int fa)
{
++buc1[dis[x] + n][flag[x]];
for(register int i = h[x]; i; i = e[i].nxt)
{
int v = e[i].to;
if (v == fa || use[v]) continue;
fill(v , x);
}
}
inline void clear(int x , int fa)
{
--buc1[dis[x] + n][flag[x]];
for(register int i = h[x]; i; i = e[i].nxt)
{
int v = e[i].to;
if (v == fa || use[v]) continue;
clear(v , x);
}
dis[x] = flag[x] = 0;
}
inline LL dfs(int x , int fa)
{
LL res = 0;
if (buc[dis[x] + n]) flag[x] = 1 , res += buc1[-dis[x] + n][0] + buc1[-dis[x] + n][1] + (!dis[x] ? 1 : 0);
else{
if (!dis[x]) res += buc1[-dis[x] + n][0];
res += buc1[-dis[x] + n][1];
}
++buc[dis[x] + n];
for(register int i = h[x]; i; i = e[i].nxt)
{
int v = e[i].to;
if (v == fa || use[v]) continue;
res += dfs(v , x);
}
--buc[dis[x] + n];
return res;
}
inline LL calc(int x)
{
getdis(x , 0);
LL res = 0;
for(register int i = h[x]; i; i = e[i].nxt)
{
int v = e[i].to;
if (use[v]) continue;
res += dfs(v , x);
fill(v , x);
}
for(register int i = h[x]; i; i = e[i].nxt)
{
int v = e[i].to;
if (use[v]) continue;
clear(v , x);
}
return res;
}
inline void divide(int x)
{
use[x] = 1 , ans += calc(x);
for(register int i = h[x]; i; i = e[i].nxt)
{
int v = e[i].to;
if (use[v]) continue;
size = siz[v] , rt = 0;
getrt(v , x) , divide(rt);
}
}
int main()
{
scanf("%d" , &n);
int u , v , w;
for(register int i = 1; i < n; i++)
{
scanf("%d%d%d" , &u , &v , &w);
w = w ? 1 : -1;
add(u , v , w) , add(v , u , w);
}
son[0] = 2e9 , size = n;
getrt(1 , 0) , divide(rt);
printf("%lld" , ans);
}
JZOJ 3234. 阴阳的更多相关文章
- 伏羲八卦、文王六十四卦、老子阴阳太极、西方哲学辩证与"解耦和复用”思想的异曲同工之妙
伏羲八卦.文王六十四卦.老子阴阳太极.西方哲学辩证与"解耦和复用”思想的异曲同工之妙 问题:任何程序语言在遇到复杂逻辑时,代码维护难度就会加大,如何处理该问题? 答案:重构,模块化. ...
- bzoj 3234: [Ahoi2013]立方体
题目链接:http://www.lydsy.com:808/JudgeOnline/problem.php?id=3234 题意:求长方体交的表面积. 思路:flood-fill const int ...
- 【HDOJ】3234 Exclusive-OR
并查集.对于对元素赋值操作,更改为I p n v.令val[n]=0(任何数与0异或仍为原值).考虑fa[x] = fx, fa[y] = fy.如果使得fa[fx] = fy, 那么val[fx] ...
- scheme 阴阳谜题
本篇分析continuation的一个著名例子"阴阳迷题",这是由David Madore先生提出的,原谜题如下: (let* ((yin ((lambda (foo) (disp ...
- 使用CSS达到阴阳八卦图等图形
CSS还是比較强大的,能够实现中国古典的"阴阳八卦图"等形状. 正方形 #rectangle { width: 200px; height: 100px; backgrount-c ...
- (jzoj snow的追寻)线段树维护树的直径
jzoj snow的追寻 DFS序上搞 合并暴力和,记录最长链和当前最远点,距离跑LCA # include <stdio.h> # include <stdlib.h> # ...
- [jzoj]3506.【NOIP2013模拟11.4A组】善良的精灵(fairy)(深度优先生成树)
Link https://jzoj.net/senior/#main/show/3506 Description 从前有一个善良的精灵. 一天,一个年轻人B找到她并请他预言他的未来.这个精灵透过他的水 ...
- [jzoj]3468.【NOIP2013模拟联考7】OSU!(osu)
Link https://jzoj.net/senior/#main/show/3468 Description osu 是一款群众喜闻乐见的休闲软件. 我们可以把osu的规则简化与改编成以下的样子: ...
- [jzoj]5478.【NOIP2017提高组正式赛】列队
Link https://jzoj.net/senior/#main/show/5478 Description Sylvia 是一个热爱学习的女孩子. 前段时间,Sylvia 参加了学校 ...
- [jzoj]1115.【HNOI2008】GT考试
Link https://jzoj.net/senior/#main/show/1115 Description 申准备报名参加GT考试,准考证号为n位数X1X2X3...Xn-1Xn(0<=X ...
随机推荐
- uniCloud云开发入门以及对传统开发方式的思考
事情缘由 作为选修了移动互联网应用的一员,老师讲的什么JS基础,还有ES6和uniapp,当然是没怎么听,因为是之前大二的时候都大概看过. 但是快到期末,老师讲了云开发,并且布置了与此相关的大作业,自 ...
- SpringCloud Alibaba(三) - GateWay网关
1.基本环境搭建 1.1 依赖 <!-- Gatway 网关会和springMvc冲突,不能添加web依赖 --> <dependency> <groupId>or ...
- 4.1:简单python爬虫
简单python爬虫 在创建的python文件中输入下列代码: # coding:utf-8 import requests from bs4 import BeautifulSoup def spi ...
- 【每日一题】【哈希表,返回结果的下标】2022年1月18日-NC61 两数之和
描述给出一个整型数组 numbers 和一个目标值 target,请在数组中找出两个加起来等于目标值的数的下标,返回的下标按升序排列.(注:返回的数组下标从1开始算起) 算法: import java ...
- 可视化编排的数据集成和分发开源框架Nifi轻松入门-上
@ 目录 概述 定义 dataflow面临挑战 特性 核心概念 架构 高级概述 安装 部署 常见处理器 入门示例 概述 定义 Nifi 官网地址 https://nifi.apache.org/ Ni ...
- ping localhost时出现::1的原因以及解决办法
ping localhost时出现: 在cmd中ping localhost解析出来的是ipv6的::1的原因是windows有个优先解析列表,当ipv6的优先级高于ipv4时,就会出现这种情况. 具 ...
- Pytorch框架详解之一
Pytorch基础操作 numpy基础操作 定义数组(一维与多维) 寻找最大值 维度上升与维度下降 数组计算 矩阵reshape 矩阵维度转换 代码实现 import numpy as np a = ...
- week_8
Andrew Ng 机器学习笔记 ---By Orangestar Week_7_Unsupervised Learning While supervised learning algorithms ...
- c++随笔测试(Corner of cpp)
在c++17下,程序的输出是什么?(有可能编译出错,有可能输出未知,有可能是未定义行为) 点击查看代码 #include<iostream> void foo(unsigned int) ...
- python里面一些零碎知识点
1. Python中反斜杠可以用在一行结尾做续行符使用. 2. pytorch中,一般来说如果对tensor的一个函数后加上了下划线,则表明这是一个in-place类型.in-place类型是指,当在 ...