题意 : 给你一棵树 。 树的每一个结点都有一个权值 。 问你有多少条路径权值的乘积是一个全然立方数 。

题目中给了你 K 个素数 ( K <= 30 ) , 全部权值都能分解成这k个素数

思路 : 一个全然立方数的素因子个数都是三的倍数 , 所以我们仅仅要求各个素数的个数即可了 , 而且我们仅仅关心个数对三的余数

所以我们能够用一个 长整形来表示每一个结点到根的各个素因子的个数( 三进制压缩 ) 。

只是由于用位运算会快一点 , 所以我用了四进制。即每两位表示一个素因子的个数 。

中间合并的时候计算刚好相加之后变成0的数,然后用map统计就能够了

注意 : 题目中 pi 的值 , 0 是取不到的(  纠结了好久该怎么处理 = = )

#include <stdio.h>
#include <string.h>
#include <map>
#include <algorithm>
#pragma comment(linker, "/STACK:102400000,102400000")
using namespace std; #define MAXN 50005
#define INF 0x3f3f3f3f
__int64 prime[35] ;
int n , k ; struct Tree{
struct Edge{
int to , nex ;
Edge(){}
Edge( int _to , int _nex ) {
to = _to ;
nex = _nex ;
}
}edge[MAXN*2] ; int head[MAXN] ;
int Index ; void init(){
memset( head , -1 , sizeof(head) ) ;
Index = 0 ;
} void add( int from , int to ) {
edge[Index] = Edge( to , head[from] ) ;
head[from] = Index ++ ;
}
} tree ; inline int get( __int64 val , int pos ) {
return ( val >> ( pos << 1 ) ) & 3 ;
} inline void set( __int64 & val , int pos , int k ) {
val |= ( (__int64)k << ( pos << 1 ) ) ;
} __int64 add( __int64 a , __int64 b ) {
__int64 ans = 0 ;
for( int i = 0 ; i < k ; i ++ ) {
int aa = get( a , i ) ;
int bb = get( b , i ) ;
aa += bb ;
aa %= 3 ; ;
set( ans , i , aa ) ;
}
return ans ;
} __int64 fan( __int64 a ) {
__int64 ans = 0 ;
for( int i = 0 ; i < k ; i ++ ) {
int tmp = get( a , i ) ;
tmp = 3 - tmp ;
set( ans , i , tmp % 3 ) ;
}
return ans ;
} bool vis[MAXN] ;
__int64 val[MAXN] ;
__int64 ans ; // 第i结点为根的子树的最大个数
int dp[MAXN] ;
// 第i结点为根的子树的大小
int sum[MAXN] ; int Sum ;
int Min , Minid ; void dfs1( int u , int f ) {
sum[u] = 1 ;
dp[u] = 0 ;
for( int i = tree.head[u] ; ~i ; i = tree.edge[i].nex ) {
int ch = tree.edge[i].to ;
if( ch == f || vis[ch] ) continue ;
dfs1( ch , u ) ;
sum[u] += sum[ch] ;
dp[u] = max( dp[u] , sum[ch] ) ;
}
} void dfs2( int u , int f ) {
int M = max( dp[u] , Sum - sum[u] ) ;
if( M < Min ) {
Min = M ;
Minid = u ;
}
for( int i = tree.head[u] ; ~i ; i = tree.edge[i].nex ) {
int ch = tree.edge[i].to ;
if( ch == f || vis[ch] ) continue ; dfs2( ch , u ) ;
}
} // 返回树的重心
int getRoot( int u ) {
// 第一遍dfs求出全部点的子树大小。和以孩子为跟的子树个数的最大值
dfs1( u , 0 ) ;
Sum = sum[u] ;
Min = 0x3f3f3f3f ;
Minid = -1 ;
// 第二次dfs求重心
/* 事实上假设整棵树求重心的话,我们一边dfs即可。由于这个是部分树,我们要dfs来确定哪些结点在当前树中 */
dfs2( u , 0 ) ;
return Minid ;
} int tot ;
__int64 Val[MAXN] ; void getVal( int u , int fa , __int64 vv ) {
Val[tot++] = add( vv , val[u] ) ;
for( int i = tree.head[u] ; ~i ; i = tree.edge[i].nex ) {
int to = tree.edge[i].to ;
if( vis[to] == true )
continue ;
if( to == fa ) continue ;
getVal( to , u , add( vv , val[u] ) ) ;
}
} map<__int64,int> mp ; __int64 cal( int root , __int64 val , __int64 root_val ) {
tot = 0 ;
getVal( root , 0 , val ) ;
//sort( Val , Val + tot ) ;
mp.clear() ;
__int64 ans = 0 ;
for( int i = 0 ; i < tot ; i ++ ) {
ans += mp[ add( root_val , fan(Val[i]) ) ] ;
mp[Val[i]] ++ ;
}
return ans ;
} void solve( int u ) {
// 得到当前子树的重心
int root = getRoot( u ) ;
// 计算以root为根的结果
ans += cal( root , 0 , val[root] ) ;
if( val[root] == 0 ) ans ++ ;
vis[root] = true ;
for( int i = tree.head[root] ; ~i ; i = tree.edge[i].nex ) {
int ch = tree.edge[i].to ;
if( vis[ch] ) continue ;
ans -= cal( ch , val[root] , val[root] ) ;
solve( ch ) ;
}
} int main(){
while( scanf( "%d" , &n ) != EOF ) {
scanf( "%d" , &k ) ;
for( int i = 0 ; i < k ; i ++ ) {
scanf( "%d" , &prime[i] ) ;
}
tree.init() ;
memset( vis , false , sizeof(vis) );
memset( val , 0 , sizeof(val) ) ;
for( int i = 1 ; i <= n ; i ++ ) {
__int64 tmp ;
scanf( "%I64d" , &tmp ) ;
for( int j = 0 ; j < k ; j ++ ) {
int cnt = 0 ;
while( tmp % prime[j] == 0 ) {
cnt ++ ;
tmp /= prime[j] ;
}
cnt %= 3 ;
set( val[i] , j , cnt ) ;
}
}
for( int i = 0 ; i < n - 1 ; i ++ ) {
int u , v ;
scanf( "%d%d" , &u , &v ) ;
tree.add( u , v ) ;
tree.add( v , u ) ;
}
ans = 0 ;
solve( 1 ) ;
printf( "%I64d\n" , ans ) ;
}
return 0 ;
}

HDU 4670 Cube number on a tree ( 树的点分治 )的更多相关文章

  1. hdu 4670 Cube number on a tree(点分治)

    Cube number on a tree Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 65535/65535 K (Java/ ...

  2. HDU 4670 Cube number on a tree

    divide and conquer on tree. #include <map> #include <vector> #include <cstdio> #in ...

  3. HDU4670 Cube number on a tree 树分治

    人生的第一道树分治,要是早点学我南京赛就不用那么挫了,树分治的思路其实很简单,就是对子树找到一个重心(Centroid),实现重心分解,然后递归的解决分开后的树的子问题,关键是合并,当要合并跨过重心的 ...

  4. HDU4670 cube number on a tree(点分治+三进制加法)

    The country Tom living in is famous for traveling. Every year, many tourists from all over the world ...

  5. hdu 4812 D Tree(树的点分治)

    D Tree Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 102400/102400 K (Java/Others) Total ...

  6. 【poj1741】Tree 树的点分治

    题目描述 Give a tree with n vertices,each edge has a length(positive integer less than 1001). Define dis ...

  7. POJ1741——Tree(树的点分治)

    1 /* *********************************************** 2 Author :kuangbin 3 Created Time :2013-11-17 1 ...

  8. HDU 5614 Baby Ming and Matrix tree 树链剖分

    题意: 给出一棵树,每个顶点上有个\(2 \times 2\)的矩阵,矩阵有两种操作: 顺时针旋转90°,花费是2 将一种矩阵替换为另一种矩阵,花费是10 树上有一种操作,将一条路经上的所有矩阵都变为 ...

  9. [hdu4670 Cube number on a tree]点分治

    题意:给一个N个带权节点的树,权值以给定的K个素数为因子,求路径上节点乘积为立方数的路径条数 思路:立方数的性质是每个因子的个数为3的倍数,那么每个因子只需要保存0-2三个状态即可,然后路径就可以转化 ...

随机推荐

  1. access2003的使用

    access2003中如何用sql语句创建表 http://zhidao.baidu.com/link?url=dinVbwoI20Xz__NbcIeBPdkjeXRWmZNB0xJvdr0eMBqN ...

  2. shell中如何取括号中的字符

    1. 使用grep(结果带括号,不知道有没有办法仅把括号中的内容匹配出来) $a='abc[edg]adfirpqu' $echo $a|grep -o '\[.*\]' #中括号的处理需要转义 [e ...

  3. js 动画性能分析 transfrom

    1.动画实现代码 (1)使用定位实现: <!DOCTYPE html> <html> <head> <meta charset="utf-8&quo ...

  4. Exif介绍

    Exif是一种图像文件格式,它的数据存储与JPEG格式是完全相同的.实际上Exif格式就是在JPEG格式头部插入了数码照片的信息,包括拍摄时的光圈.快门.白平衡.ISO.焦距.日期时间等各种和拍摄条件 ...

  5. Stage3d AGAL GPU处理照片 旧照片效果 sepia || pixelbender

    如果看不到下边的flash,请更新flash player到最新版本. 利用AGAL实现旧照片效果,大家可以对照一下之前一篇文章,关于图像处理(pixelbender).硬件处理肯定会更快,但这里无法 ...

  6. eclipse properties文件插件

      eclipse properties插件 CreateTime--2018年4月22日22:51:34 Author:Marydon 下载地址:properties文件插件.rar 1.将plug ...

  7. 怎么将txt文件转化为html格式的文件?--极为丑陋的方式

    # -*- coding: utf-8 -*- #python 27 #xiaodeng #怎么将txt文件转化为html格式的文件?--极为丑陋的方式 import os #找出所有的txt格式的文 ...

  8. 29、java中阻塞队列

    阻塞队列与普通队列的区别在于,当队列是空的时,从队列中获取元素的操作将会被阻塞,或者当队列是满时,往队列里添加元素的操作会被阻塞.试图从空的阻塞队列中获取元素的线程将会被阻塞,直到其他的线程往空的队列 ...

  9. HTTP头返回码分析

    http协议通讯时,在客户端发送请求后(request),服务器端返回的状态码解释(response) http状态码 1**:请求收到,继续处理 2**:操作成功收到,分析.接受           ...

  10. FFmpeg基础库编程开发学习笔记——视频常见格式

    声明一下:这些关于ffmpeg的文章仅仅是用于记录我的学习历程和以便于以后查阅,文章中的一些文字可能是直接摘自于其它文章.书籍或者文献,学习ffmpeg相关知识是为了使用在Android上,我也才是刚 ...