hdu 4670 树的分治-求点对的个数
/*
树的分治
因为树的点权值可达到10^15,注意手动扩栈,还有int64
题意:给你一棵树,给你一些素数,给你每个点一个权值且每个权值均可由这些素数组成。现在定义任意任意两点的价值为他们路径上的权值相乘。求这样的点对的权值为立方数的个数
解:
如果直接求得话会超int64,不可行
由立方数的性质可得,一个数可有素数组成,对于这些素数可以分解为这些素数相乘的形式如,24=(2^3)*(3^1);如果是立方数的话那么他的各进制对3取余都为0.股24可写成01这种三进制形式
对于这些权值的乘法可有三进制想加可得。
接下来就是树的分治了
当然这里可以先求出一条子树上的各个点的权值乘积,然后和根节点和其他字树比较看是否可以互补那么就找到一对
可用map容器实现。因为他重点是比较到根节点和其他子树是否可以互补,进而递归下去,求出每个子树的这样的点对。
*/
#pragma comment(linker, "/STACK:1024000000,1024000000")
#include<stdio.h>
#include<string.h>
#include<algorithm>
#include<iostream>
#include<map>
#define ll __int64
#define N 51000
#define inf 0x3fffffff
using namespace std;
struct node {
ll u,v,next;
}bian[N*4];
ll yong,head[N];
ll num[N],nn,all[N][51],ff[N][51],minn,m,vis[N],fp[N],ma;
ll prime[51],len;
map<ll,ll>q;
void init() {
yong=0;
memset(head,-1,sizeof(head));
}
void addedge(ll u,ll v) {
bian[yong].u=u;
bian[yong].v=v;
bian[yong].next=head[u];
head[u]=yong++;
}
void dfs1(ll u,ll fa) {//找到每次根子树的的节点数目
ll i;
nn++;
for(i=head[u];i!=-1;i=bian[i].next) {
ll v=bian[i].v;
if(v!=fa&&!vis[v])
dfs1(v,u);
}
return ;
}
ll Max(ll v,ll vv) {
return v>vv?v:vv;
}
void dfs2(ll u,ll fa) {//找重心
num[u]=1;
ll i,tit=0;
for(i=head[u];i!=-1;i=bian[i].next) {
ll v=bian[i].v;
if(v!=fa&&!vis[v]) {
dfs2(v,u);
num[u]+=num[v];
tit=Max(tit,num[v]);
}
}
tit=Max(tit,nn-num[u]);
if(tit<minn) {
minn=tit;
ma=u;
}
return ;
}
void dfs4(ll u,ll fa) {
ll i;
//prllf("dfs4=%d",u);
if(fa!=-1) {
ll e=fp[fa];
for(i=1;i<=m;i++)
all[len][i]=(all[e][i]+ff[u][i])%3;
}
else {
for(i=1;i<=m;i++)
all[len][i]=ff[u][i];
}
fp[u]=len++;
for(i=head[u];i!=-1;i=bian[i].next) {
ll v=bian[i].v;
if(v!=fa&&!vis[v])
dfs4(v,u);
}
return ;
}
ll dfs3(ll u) {//求以当前子树为根的符合条件的对数
ll i,ans=0,j,k;
ll s1,s2;
s1=0;
q.clear();
for(i=1;i<=m;i++)
s1=s1*3+ff[u][i];
if(s1==0)ans++;
//prllf("ma=%d,s1=%d\n",ma,s1);
q[s1]=1;
for(i=head[u];i!=-1;i=bian[i].next) {
ll v=bian[i].v;
if(vis[v])continue;
len=0;
dfs4(v,-1);
// prllf("len=%d\n",len);
for(j=0;j<len;j++) {//进行互补
s1=0;
for(k=1;k<=m;k++)
s1=s1*3+(3-all[j][k])%3;
// prllf("zzs1=%d\n",s1);
ans+=q[s1];
}
for(j=0;j<len;j++) {
s2=0;
for(k=1;k<=m;k++)
s2=s2*3+(all[j][k]+ff[u][k])%3;//到根结点的权值,为下次互补做准备,同时避免重复
q[s2]++;
}
// prllf("aad%d\n",ans);
}
return ans;
}
ll dfs(ll u) {
minn=inf;
nn=0;
dfs1(u,-1);
dfs2(u,-1);//求重心
vis[ma]=1;
ll ans=dfs3(ma);
// prllf("ma=%d\n",ma);
//prllf("ans=%d\n",ans);
ll i;
for(i=head[ma];i!=-1;i=bian[i].next) {
ll v=bian[i].v;
if(!vis[v])
ans+=dfs(v);
}
return ans;
}
int main() {
ll n,i,j,k,kk;
while(scanf("%I64d",&n)!=EOF) {
init();
scanf("%I64d",&m);
for(i=1;i<=m;i++)
scanf("%I64d",&prime[i]);
for(i=1;i<=n;i++) {
scanf("%I64d",&k);
for(j=1;j<=m;j++) {
kk=0;
while(k%prime[j]==0) {
k/=prime[j];
kk++;
kk%=3;
}
ff[i][j]=kk;
}
}
for(i=1;i<n;i++) {
scanf("%I64d%I64d",&j,&k);
addedge(j,k);
addedge(k,j);
}
memset(vis,0,sizeof(vis));
printf("%I64d\n",dfs(1));
}
return 0;}
hdu 4670 树的分治-求点对的个数的更多相关文章
- hdu 4638 树状数组 区间内连续区间的个数(尽可能长)
Group Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Subm ...
- hdu 4871 树的分治+最短路记录路径
/* 题意:给你一些节点和一些边,求最短路径树上是k个节点的最长的路径数. 解:1.求出最短路径树--spfa加记录 2.树上进行操作--树的分治,分别处理子树进行补集等运算 */ #include& ...
- hdu 4670 树的点分治
思路:首先当然是要用树的点分治了.根节点为root,那么经过root的合法路径数求出来这题就解决了.因为我们可以用分治枚举根,最后将所有根的路径数加起来就是结果.当然这里的根不是整棵树的根,是子树根. ...
- HDU 1394 树状数组求逆序对
Minimum Inversion Number Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java ...
- hdu 1007 Quoit Design 分治求最近点对
Quoit Design Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Tot ...
- Hdu 2586 树链剖分求LCA
Code: #include<cstdio> #include<cstring> #include<vector> #include<algorithm> ...
- HDU X问题 中国剩余定理--求满足条件的个数
X问题 Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submiss ...
- Coconuts HDU - 5925 (二维离散化求连通块的个数以及大小)
题目链接: D - Coconuts HDU - 5925 题目大意:首先是T组测试样例,然后给你n*m的矩阵,原先矩阵里面都是白色的点,然后再输入k个黑色的点.这k个黑色的点可能会使得原先白色的点 ...
- hdu 1257 最少拦截系统 求连续递减子序列个数 (理解二分)
最少拦截系统 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Subm ...
随机推荐
- P4727 [HNOI2009]图的同构记数
传送门 如果我们把选出子图看成选出边,进而看成对边黑白染色,那么就是上一题的弱化版了,直接复制过来然后令\(m=2\)即可 不过直接交上去会T,于是加了几发大力优化 不知为何华丽的被小号抢了rank2 ...
- golang——database/sql包学习
1.database/sql包 sql包提供了保证SQL或类SQL数据库的泛用接口. 使用sql包时必须注入(至少)一个数据库驱动. (1)获取mysql driver:go get -v githu ...
- 【原创】Maven安装和配置
ι 版权声明:本文为博主原创文章,未经博主允许不得转载. 前提 利用maven进行java项目或J2EE项目开发,要求电脑已配置java开发环境(JDK) 下载 下载地址:http://maven.a ...
- Java多线程(二) synchronized 针对对象进行锁定
http://www.cnblogs.com/QQParadise/articles/5059824.html 1.方法内的变量为线程安全的 2.实例变量非线程安全的 public class Has ...
- 贪心 FZU 2013 A short problem
题目传送门 /* 题意:取长度不小于m的序列使得和最大 贪心:先来一个前缀和,只要长度不小于m,从m开始,更新起点k最小值和ans最大值 */ #include <cstdio> #inc ...
- 406 Queue Reconstruction by Height 根据身高重建队列
假设有打乱顺序的一群人站成一个队列. 每个人由一个整数对(h, k)表示,其中h是这个人的身高,k是排在这个人前面且身高大于或等于h的人数. 编写一个算法来重建这个队列.注意:总人数少于1100人.示 ...
- 为什么,博主我要写下这一系列windows实用网络?
发现,随着自身一路过来所学,无论在大数据领域.还是linux or windows里,菜鸟的我慢慢在长大.把自己比作一个园,面积虽在增加,涉及面增多,但圆外的东西,还是那么多. 现在,正值在校读研 ...
- JDBC性能优化
一.使用PreparedStatement的Batch功能 参见本人一下文章:http://blog.csdn.net/lmb55/article/details/50631062 二.选择合适的光标 ...
- iframe子页面让父页面跳转 parent.location.href
if ($roleNum < 9) { echo "<script > parent.location.href='admin_login.php' </script ...
- GAN生成图像论文总结
GAN Theory Modifyingthe Optimization of GAN 题目 内容 GAN DCGAN WGAN Least-square GAN Loss Sensi ...