[JZOJ3234] 阴阳
题目
题目大意
有一棵树,每条边有\(0\)或\(1\)两种颜色。
求满足存在\((u,v)\)路径上的点\(x\)使得\((u,x)\)和\((x,v)\)路径上的两种颜色出现次数相同的点对\((u,v)\)数量。
思考历程
在看到这题之前我就已经知道这题是点分治了……
然而看了题目后,很长一段时间搞错了题目大意……
后来终于搞懂了题目大意,于是开始想正解。
想了半天,心态崩了……点分治我在差不多一年前才打过一题啊!
很久后搜题解,粗略看一看,没有一个看得懂……
于是又开始自己刚……然后刚出来了……
结果WA了……
调了不知道多久,终于AC……
正解
正解就是点分治啦……
显然的思路是将\(0\)颜色的边权值记为\(1\),\(1\)颜色的边权值记为\(-1\)。
如果\((u,v)\)满足条件,一定有点\(x\)使得\((u,x)\)和\((x,v)\)路径上的权值和为\(0\)。
现在设重心为\(root\),现在我们要求的路径必须经过\(root\)。
求出每个点到\(root\)路径上的权值和\(dis_x\)。
显然,如果\((u,v)\)满足条件,一定有\(dis_u+dis_v=0\)。
还有一个条件是存在\(u\)或\(v\)的祖先\(x\),使得\(dis_x=dis_u\)或\(dis_x=dis_v\)。
然后我们就搬出几个桶……
设\(tot_i\)表示\(dis\)值为\(i\)的点的数量,\(can_i\)表示\(dis\)值为\(i\)并且其祖先有\(dis\)值和它相同的点的数量。这两个都是前面计算过的\(root\)的所有子树的。同理,设\(sub_i\)和\(scn_i\),意义相同,表示现在正在计算的这个子树的。
\(sub_i\)的计算方法显然,至于\(scn_i\),我们在\(dfs\)的过程中再维护一个桶\(anc_i\),表示\(dis\)值为\(i\)的祖先有多少个,这样就可以快速判断了。
\(tot_i\)和\(can_i\)可以由后两者累加而得。
计算答案的时候,枚举\(dis\)值,用前面四个桶里的值来计算,具体来说,如果\((u,v)\)能被算入答案中,则\(u\)和\(v\)至少有一个在\(can\)或\(scn\)中。
注意,我们还没有计算\(v=root\)的情况。针对这种情况,我们维护一个\(cnt0\),在\(dfs\)的时候遇到\(anc_{dis_i}>1\)时,便意味着除了\(root\)之外,还有一个和它\(dis\)值相同的祖先,那就将其加入\(cnt0\)中。答案加上\(cnt0\)即可。
所以说实际上这是一道点分治的模板题啊……
代码
using namespace std;
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <climits>
#include <cassert>
#define N 100010
int n;
struct EDGE{
int to,len;
EDGE *las;
bool ok;
} e[N*2];
int ne;
EDGE *last[N];
inline void link(int u,int v,int len){
e[ne]={v,len,last[u],1};
last[u]=e+ne++;
}
#define rev(ei) (e+(int((ei)-e)^1))
long long ans;
int siz[N],all;
void get_siz(int x,int fa){
siz[x]=1;
for (EDGE *ei=last[x];ei;ei=ei->las)
if (ei->to!=fa && ei->ok){
get_siz(ei->to,x);
siz[x]+=siz[ei->to];
}
}
int find(int x,int fa){//找重心
bool bz=(all-siz[x]<<1<=all);
for (EDGE *ei=last[x];ei;ei=ei->las)
if (ei->to!=fa && ei->ok){
int tmp=find(ei->to,x);
if (tmp)
return tmp;
bz&=(siz[ei->to]<<1<=all);
}
if (bz)
return x;
}
int dis[N];
int _tot[N*2],*tot=_tot+N;//为了代码方便,所以用指针来处理了……
int _sub[N*2],*sub=_sub+N;
int _can[N*2],*can=_can+N;
int _scn[N*2],*scn=_scn+N;
int cnt0;
int _anc[N*2],*anc=_anc+N;
int mn,mx,smn,smx;//记录mn,mx是为了方便清空桶。
void get_dis(int x,int fa){
smn=min(smn,dis[x]),smx=max(smx,dis[x]);
sub[dis[x]]++;
if (anc[dis[x]]){
scn[dis[x]]++;
if (anc[dis[x]]>1 && dis[x]==0)
cnt0++;
}
anc[dis[x]]++;
for (EDGE *ei=last[x];ei;ei=ei->las)
if (ei->to!=fa && ei->ok){
dis[ei->to]=dis[x]+ei->len;
get_dis(ei->to,x);
}
anc[dis[x]]--;
}
void divide(int x){
get_siz(x,0);
all=siz[x];
int root=find(x,0);
dis[root]=0;
mn=INT_MAX,mx=INT_MIN;
anc[0]=1;
for (EDGE *ei=last[root];ei;ei=ei->las)
if (ei->ok){
smn=INT_MAX,smx=INT_MIN;
cnt0=0;
dis[ei->to]=dis[root]+ei->len;
get_dis(ei->to,root);
for (int j=smn;j<=smx;++j)
ans+=1ll*sub[j]*can[-j]+1ll*scn[j]*tot[-j]-1ll*scn[j]*can[-j];//容斥原理
ans+=cnt0;
for (int j=smn;j<=smx;++j){
tot[j]+=sub[j],sub[j]=0;
can[j]+=scn[j],scn[j]=0;
}
mn=min(mn,smn);
mx=max(mx,smx);
}
anc[0]=0;
for (int i=mn;i<=mx;++i)
tot[i]=can[i]=0;
for (EDGE *ei=last[root];ei;ei=ei->las)
if (ei->ok){
ei->ok=rev(ei)->ok=0;
divide(ei->to);
}
}
int main(){
scanf("%d",&n);
for (int i=1;i<n;++i){
int u,v,type;
scanf("%d%d%d",&u,&v,&type);
link(u,v,type==0?1:-1);
link(v,u,type==0?1:-1);
}
divide(1);
printf("%lld\n",ans);
return 0;
}
总结
点分治的题目还是打得太少了……
[JZOJ3234] 阴阳的更多相关文章
- 伏羲八卦、文王六十四卦、老子阴阳太极、西方哲学辩证与"解耦和复用”思想的异曲同工之妙
伏羲八卦.文王六十四卦.老子阴阳太极.西方哲学辩证与"解耦和复用”思想的异曲同工之妙 问题:任何程序语言在遇到复杂逻辑时,代码维护难度就会加大,如何处理该问题? 答案:重构,模块化. ...
- scheme 阴阳谜题
本篇分析continuation的一个著名例子"阴阳迷题",这是由David Madore先生提出的,原谜题如下: (let* ((yin ((lambda (foo) (disp ...
- 使用CSS达到阴阳八卦图等图形
CSS还是比較强大的,能够实现中国古典的"阴阳八卦图"等形状. 正方形 #rectangle { width: 200px; height: 100px; backgrount-c ...
- CSS3_边框 border 详解_一个 div 的阴阳图
(面试题) 怎么样通过 CSS 画一个三角形: 1. 元素的 width 和 height 设置为 0 2. 边框 足够大 3. 需要的三角形的部分, border-top-color 设置为 ...
- Python turtle绘制阴阳太极图代码解析
本文详细分析如何使用Python turtle绘制阴阳太极图,先来分解这个图形,图片中有四种颜色,每条曲线上的箭头表示乌龟移动的方向,首先从中心画一个半圆(红线),以红线所示圆的直径作半径画一个校园, ...
- UI设计教程分享:字体变形—阴阳收缩法
阴阳师中国古代对自然规律发展变化基础因素的描述,是古代美学逻辑思维.推理分析的核心要素,也是描述万物基本要素和成因的概念之一.阴阳代表事物的对立关系,它是自然界的客观规律,是万物运动变化的本源,是人类 ...
- css基本图形绘制(基本的矩形、圆形、椭圆、三角形、多边形,也包括稍微复杂一点的爱心、钻石、阴阳八卦等)
正方形: 代码: <style> .square { width: 100px; height: 100px; background-color: cornflowerblue; } &l ...
- python之demo2----改编自python官方提供的turtle_yinyang.py画阴阳的demo
""" 执行 python -m turtledemo 命令查看系统内置demo的源码 绘制:需要通过import turtle引入绘制图形库turtle库 改编自pyt ...
- Rust 阴阳谜题,及纯基于代码的分析与化简
Rust 阴阳谜题,及纯基于代码的分析与化简 雾雨魔法店专栏 https://zhuanlan.zhihu.com/marisa 来源 https://zhuanlan.zhihu.com/p/522 ...
随机推荐
- 38-Ubuntu-用户管理-03-usermod指定用户登录shell
简记: 所谓shell就是可以输入终端命令的窗口,shell是一个软件. 1.Ubuntu终端shell介绍 summmer@summmer-virtual-machine:~/桌面$ summmer ...
- HXY烧情侣
题目描述 众所周知,HXY已经加入了FFF团.现在她要开始喜(sang)闻(xin)乐(bing)见(kuang)地烧情侣了.这里有n座电影院,n对情侣分别在每座电影院里,然后电影院里都有汽油,但是要 ...
- SQL Server2012 Offset Fetch子句 分页查询
在本教程中,将学习如何使用SQL Server OFFSET FETCH子句来限制查询返回的行数.OFFSET和FETCH子句是ORDER BY子句的选项. 它们用于限制查询返回的行数.以下是OFFS ...
- 标准 IO fprintf 与 sprintf 函数使用
函数原型 fprintf int fprintf(FILE *stream, const char *format, ...); 把数据写到流中 int sprintf(char *str, con ...
- 从零开始搭搭建系统3.1——顶级pom制定
从零开始搭搭建系统3.1——顶级pom制定
- day01 mysql认识 安装 配置 起服务 密码 字符集 用户授权
day01 mysql 一.认识mysql 关系型数据库: 最流行的关系型数据库管理系统,支持大型数据库,处理上千万条记录 关系型: oracle, ...
- Dart编程变量
变量是"存储器中的命名空间",用于存储值.换句话说,它作为程序中值的容器.变量名称称为标识符.以下是标识符的命名规则 - 标识符不能是关键字. 标识符可以包含字母和数字. 标识符不 ...
- 阿里云POLARDB如何帮助百胜软件应对数据库的“巅峰时刻”
POLARDB是阿里云自研的下一代关系型云数据库,100%兼容MySQL,存储容量最高可达100TB,性能最高提升至MySQL的6倍,适用于企业多样化的数据库应用场景.POLARDB采用存储和计算分离 ...
- css制作旋转风车(transform 篇)
做这个案例之前首先要大概了解CSS的transform的属性 transform 属性向元素应用 2D 或 3D 转换.该属性允许我们对元素进行旋转.缩放.移动或倾斜. 看看效果图 打开的时候自动旋转 ...
- NX二次开发-UFUN设置环境变量UF_set_variable
NX9+VS2012 #include <uf.h> #include <stdio.h> UF_initialize(); //UFUN方式 //设置环境变量 int a = ...