/*
树的分治
因为树的点权值可达到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 树的分治-求点对的个数的更多相关文章

  1. hdu 4638 树状数组 区间内连续区间的个数(尽可能长)

    Group Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Subm ...

  2. hdu 4871 树的分治+最短路记录路径

    /* 题意:给你一些节点和一些边,求最短路径树上是k个节点的最长的路径数. 解:1.求出最短路径树--spfa加记录 2.树上进行操作--树的分治,分别处理子树进行补集等运算 */ #include& ...

  3. hdu 4670 树的点分治

    思路:首先当然是要用树的点分治了.根节点为root,那么经过root的合法路径数求出来这题就解决了.因为我们可以用分治枚举根,最后将所有根的路径数加起来就是结果.当然这里的根不是整棵树的根,是子树根. ...

  4. HDU 1394 树状数组求逆序对

    Minimum Inversion Number Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java ...

  5. hdu 1007 Quoit Design 分治求最近点对

    Quoit Design Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Tot ...

  6. Hdu 2586 树链剖分求LCA

    Code: #include<cstdio> #include<cstring> #include<vector> #include<algorithm> ...

  7. HDU X问题 中国剩余定理--求满足条件的个数

    X问题 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submiss ...

  8. Coconuts HDU - 5925 (二维离散化求连通块的个数以及大小)

    题目链接: D - Coconuts  HDU - 5925 题目大意:首先是T组测试样例,然后给你n*m的矩阵,原先矩阵里面都是白色的点,然后再输入k个黑色的点.这k个黑色的点可能会使得原先白色的点 ...

  9. hdu 1257 最少拦截系统 求连续递减子序列个数 (理解二分)

    最少拦截系统 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Subm ...

随机推荐

  1. [SDOI2011]消防(单调队列,树的直径,双指针)

     消防 2011年  时间限制: 2 s  空间限制: 256000 KB  题目等级 : 大师 Master   题目描述 Description 某个国家有n个城市,这n个城市中任意两个都连通且有 ...

  2. mysql——免安装配置

    1.下载 (1)下载地址:https://dev.mysql.com/downloads/mysql/ (2)选择下载 2.配置环境变量 (1)解压目录:D:\mysql-8.0.16-winx64 ...

  3. [CREC2007/CQOI2014]robotic sort

    Description 一个实验室里有n个长短不一的试管.你的任务是编写一段程序,用机器臂把它们按照高度从小到大的顺序排列. 对于高度相同的试管,排序前后的相对位置应保持不变.排序方法如图所示. 排序 ...

  4. Visual C++ Windows 桌面应用程序样例(摘抄)

    //================================== //Windows应用程序框架结构(例子) //参考:<Visual C++宝典>陈国建等编著 //======= ...

  5. 解决FormClosing事件点击关闭2次的问题

    以下代码:提示框会跳出2遍  private void mFrmmain_FormClosing(object sender, FormClosingEventArgs e) { if (Dialog ...

  6. JavaScript(十)基本包装类

    基本包装类都具有对象的基本方法     toString   和 valueOf Number 数字是原始类型,那为啥还有方法? 因为他在执行方法的时候会创建一个对应的包装类对象,这个对象有这种方法, ...

  7. HTTP、HTTP1.0、HTTP1.1、HTTP2.0——笔记

    笔记来源地址:https://mp.weixin.qq.com/s/T2IErLDxbWP1a-VbRkZZHg HTTP: HTTP是WWW数据通信的基础,是应用层协议. HTTP是干什么的?用来给 ...

  8. 解决:未能加载文件或程序集“System.Web.Http, Version=5.2.4.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35”或它的某一个依赖项。找到的程序集清单定义与程序集引用不匹配。 (异常来自 HRESULT:0x80131040)

    今天发布web API,调用接口报错了:未能加载文件或程序集“System.Web.Http, Version=5.2.4.0, Culture=neutral, PublicKeyToken=31b ...

  9. [转]汇编语言:MOVSB,MOVSW,MOVSD

    汇编语言:MOVSB,MOVSW,MOVSD 转自: http://blog.csdn.net/zhenyongyuan123/article/details/8364011   目前80386系列的 ...

  10. MFC_2.1使用单选和多选框

    使用单选和多选框 单选 1.拖控件 设置名字,CTRL+D设置顺序,属性设置第一个GROUP为TRUE: 2.设置第一个按钮绑定变量为 值 INT型 名称m_RadioIndxe; 3.设置单击响应内 ...