2019.01.09 bzoj3697: 采药人的路径(点分治)
传送门
点分治好题。
题意:给出一棵树,边分两种,求满足由两条两种边数相等的路径拼成的路径数。
思路:
考虑将边的种类转化成边权−1-1−1和111,这样就只用考虑由两条权值为000的路径拼成的路径数。
然后发现对于一个点ppp,经过它的这样的路径满足如下至少一个条件:
- ∃v\exist v∃v满足dist(v,p)=0&&∃v0∈roadv,pdist(v,p)=0\&\&\exist v_0\in road_{v,p}dist(v,p)=0&&∃v0∈roadv,p满足dist(v0,v)=0dist(v_0,v)=0dist(v0,v)=0
- ∃v1,v2\exist v_1,v_2∃v1,v2满足$dist(v_1,p)=0,dist(v_2,p)=0
- ∃v1,v2\exist v_1,v_2∃v1,v2满足dist(v1,p)+dist(v2,p)=0&&∃v0∈roadv1,pdist(v_1,p)+dist(v_2,p)=0\&\&\exist v_0\in road_{v_1,p}dist(v1,p)+dist(v2,p)=0&&∃v0∈roadv1,p满足distv0,p=distv1,pdist{v_0,p}=dist_{v_1,p}distv0,p=distv1,p
- ∃v1,v2\exist v_1,v_2∃v1,v2满足dist(v1,p)+dist(v2,p)=0&&∃v0∈roadv2,pdist(v_1,p)+dist(v_2,p)=0\&\&\exist v_0\in road_{v_2,p}dist(v1,p)+dist(v2,p)=0&&∃v0∈roadv2,p满足distv0,p=distv2,pdist{v_0,p}=dist_{v_2,p}distv0,p=distv2,p
然后点分容斥处理即可,注意不要忘记第一种情况的讨论。
代码:
#include<bits/stdc++.h>
#define fi first
#define se second
#define ri register int
using namespace std;
const int N=1e5+5;
int n,siz[N],msiz[N],rt=0,all=0,lim;
bool vis[N];
typedef pair<int,int> pii;
typedef long long ll;
vector<pii>e[N];
ll ans=0,cnt[2][N][2]={0};
inline int read(){
int ans=0;
char ch=getchar();
while(!isdigit(ch))ch=getchar();
while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
return ans;
}
void findroot(int p,int fa){
siz[p]=1,msiz[p]=0;
for(ri i=0,v;i<e[p].size();++i){
if((v=e[p][i].fi)==fa||vis[v])continue;
findroot(v,p),siz[p]+=siz[v],msiz[p]=max(msiz[p],siz[v]);
}
msiz[p]=max(msiz[p],all-siz[p]);
if(msiz[p]<msiz[rt])rt=p;
}
void getdis(int p,int fa,int dist,int down,int up){
if(dist>=down&&dist<=up){
if(dist>=0)++cnt[0][dist][1];
if(dist<0)++cnt[1][-dist][1];
}
else{
if(dist>=0)++cnt[0][dist][0];
if(dist<0)++cnt[1][-dist][0];
}
down=min(down,dist),up=max(up,dist),lim=max(lim,max(-down,up));
for(ri i=0,v;i<e[p].size();++i){
if((v=e[p][i].fi)==fa||vis[v])continue;
getdis(v,p,dist+e[p][i].se,down,up);
}
}
inline void calc(int p,int pre,int tmp){
ll sum=0;
lim=0;
if(tmp==1)getdis(p,0,pre,1,-1);
else getdis(p,0,pre,0,0);
sum+=cnt[0][0][1]*(cnt[0][0][1]-1)/2,cnt[0][0][0]=cnt[0][0][1]=0;
for(ri i=1;i<=lim;++i){
sum+=cnt[0][i][0]*cnt[1][i][1];
sum+=cnt[0][i][1]*cnt[1][i][0];
sum+=cnt[0][i][1]*cnt[1][i][1];
cnt[0][i][0]=cnt[0][i][1]=cnt[1][i][0]=cnt[1][i][1]=0;
}
ans+=sum*tmp;
}
void solve(int p,int fa,int tim,int dist){
if(!dist){
if(tim>=2)++ans;
++tim;
}
for(ri i=0,v;i<e[p].size();++i)if((v=e[p][i].fi)!=fa&&!vis[v])solve(v,p,tim,dist+e[p][i].se);
}
void dfs(int p){
vis[p]=1,solve(p,0,0,0),calc(p,0,1);
for(ri i=0,v;i<e[p].size();++i){
if(vis[v=e[p][i].fi])continue;
calc(v,e[p][i].se,-1),rt=0,all=siz[v],findroot(v,0),dfs(rt);
}
}
int main(){
freopen("lx.in","r",stdin);
n=read();
for(ri i=1,u,v,w;i<n;++i)u=read(),v=read(),w=read()?1:-1,e[u].push_back(pii(v,w)),e[v].push_back(pii(u,w));
rt=0,all=n,msiz[rt]=n,findroot(1,0),dfs(rt);
cout<<ans;
return 0;
}
2019.01.09 bzoj3697: 采药人的路径(点分治)的更多相关文章
- BZOJ3697采药人的路径——点分治
题目描述 采药人的药田是一个树状结构,每条路径上都种植着同种药材.采药人以自己对药材独到的见解,对每种药材进行了分类.大致分为两类,一种是阴性的,一种是阳性的.采药人每天都要进行采药活动.他选择的路径 ...
- BZOJ3697:采药人的路径(点分治)
Description 采药人的药田是一个树状结构,每条路径上都种植着同种药材. 采药人以自己对药材独到的见解,对每种药材进行了分类.大致分为两类,一种是阴性的,一种是阳性的. 采药人每天都要进行采药 ...
- [bzoj3697]采药人的路径——点分治
Brief Description 采药人的药田是一个树状结构,每条路径上都种植着同种药材. 采药人以自己对药材独到的见解,对每种药材进行了分类.大致分为两类,一种是阴性的,一种是阳性的. 采药人每天 ...
- 2019.01.09 bzoj2599: [IOI2011]Race(点分治)
传送门 题意:给一棵树,每条边有权.求一条路径,权值和等于K,且边的数量最小. 思路: 考虑点分治如何合并. 我们利用树形dpdpdp求树的直径的方法,边dfsdfsdfs子树边统计答案即可. 代码: ...
- AWS re:Invent(2019.01.09)
时间:2019.01.09地点:北京国际饭店
- 【BZOJ3697】采药人的路径 点分治
[BZOJ3697]采药人的路径 Description 采药人的药田是一个树状结构,每条路径上都种植着同种药材.采药人以自己对药材独到的见解,对每种药材进行了分类.大致分为两类,一种是阴性的,一种是 ...
- [bzoj3697]采药人的路径_点分治
采药人的路径 bzoj-3697 题目大意:给你一个n个节点的树,每条边分为阴性和阳性,求满足条件的链的个数,使得这条链上阴性的边的条数等于阳性的边的条数,且这条链上存在一个节点,这个节点到一个端点的 ...
- bzoj千题计划248:bzoj3697: 采药人的路径
http://www.lydsy.com/JudgeOnline/problem.php?id=3697 点分治 路径0改为路径-1 g[i][0/1] 和 f[i][0/1]分别表示当前子树 和 已 ...
- #学习笔记# VALSE 2019.01.09 朱俊彦 --- Learning to Synthesize Images, Videos, and 3D Objects
视频类型:VALSE-webinar 报告时间:2019年01月09日 报告人:MIT朱俊彦 报告题目:Learning to Synthesize Images, Videos, and 3D Ob ...
随机推荐
- linux环境下的c++编程
就C++开发工具而言,与Windows下微软(VC, VS2005等)一统天下相比,Linux/Unix下C++开发,可谓五花八门,各式各样.Emacs, vi, eclipse, anjuta,kd ...
- 约束布局 ConstraintLayout
app:layout_constraintVertical_bias="0.5"app:layout_constraintHorizontal_bias="0.5&quo ...
- oracle中job定时任务96
.INTERVAL参数常用值示例 每天午夜12点 ''TRUNC(SYSDATE + 1)'' 每天早上8点30分 ''TRUNC(SYSDATE + 1) + ...
- Django的Rbac介绍3
今天的博客主要是记录一下如何实现左侧菜单,这里我们想实现的效果就是,如果用户有查看用户的权限,则显示查看用户的左侧菜单,如果用户有查看角色的权限,则显示查看角色的左侧菜单,如果两者都有,则需要显示两个 ...
- TestSuite测试报告生成
简介:HTMLTestRuner介绍 1, 无法使用pip安装,手工下载 2, python3和python2语法不一致导致了HTMLTestRunner在py3中不兼容 解决办法:导入下面的HTML ...
- 如何开发简单的javaweb项目,jsp+javabean+servlet
一.相关的软件下载和环境配置 1.下载并配置JDK. 2.下载eclipse. 3.下载并配置apache-tomcat(服务器). 4.下载MySQL(数据库). 5.下载Navicat for M ...
- swift - 基础属性 - 属性写法
var num1 : Int = 0 var num2 : Int = 5 /// 1.计算属性 var num3 : Int{ return num1 + num2 } /// 2.闭包属性 pri ...
- Httpclient 表单,json,multipart/form-data 提交 ---总结常用的方法
最近在项目中,一直在使用HttpClient 中的方法,这里我进行一些方法的汇总,也是结合了一些大牛写的代码,以备不时之需 官话:HttpClient 是Apache Jakarta Common 下 ...
- MATLAB单步调试
我用的是matlab R2012b 1.先设置断点:点击菜单栏中"EDITOR"--"Breakpoints"--"set",出现以下对话框 ...
- centos 命令学习
关机&重启 shutdown -h 10 #计算机将于10分钟后关闭,且会显示在登录用户的当前屏幕中 shutdown -h now #计算机会立刻关机 shut ...