[BJOI2015]树的同构 && 树哈希教程
有根树的哈希
离散数学中对树哈希的描述在这里。大家可以看看。
判断有根树是否同构,可以考虑将有根树编码。而编码过程中,要求保留树形态的特征,同时忽略子树顺序的不同。先来看一看这个方法:
不妨令一棵树的编码是个字符串\(T\)。
对于一个点\(u\),先求出\(u\)所有\(son_u\)的编码\(f_{son_u}\),然后将这些编码按字典序从小到大排序得到\(g_{1\cdots k}\)。那么\(f_u="0"+\sum\limits g_i+"1"\)。
令\(T\)的编码为根节点的编码。
通过解码的方式可以验证这个算法的正确性。
这个只是树的编码,并不算是树的哈希。这个编码保证了正确性。而实际上,我们应用的时候,通常不会,也不允许这样做(时间空间都吃不消)。我们通常会采用数值的操作和取模的方法。而实际上,这个数值的操作要尽可能满足保留树形态的特征,同时忽略子树顺序的不同。加法、异或、排序等都是可以的。
树哈希的方法非常多,OIwiki上有三种常见的做法。
无根树的哈希
一般的,通过选定根将无根树转成有根树,从而实现无根树的编码。而根一般选定为数的中心。如果有两个中心,就选定编码较小中心为根。对于哈希也是差不多的道理。
对于这道题,由于数据范围十分小,所以直接暴力编码,map判重即可。
#include <bits/stdc++.h>
using namespace std;
const int Maxn = 60;
struct edge {
int To, Next;
edge() {}
edge( int _To, int _Next ) : To( _To ), Next( _Next ) {}
};
int Start[ Maxn ], Used;
edge Edge[ Maxn << 1 ];
inline void AddEdge( int x, int y ) {
Edge[ ++Used ] = edge( y, Start[ x ] );
Start[ x ] = Used;
return;
}
map< string, int > Map;
string A[ Maxn ], B[ Maxn ], S;
int n, m, T[ Maxn ];
int Cnt, Rt[ Maxn ];
int Dfs1( int u, int Fa ) {
int Size = 1, IsR = 1;
for( int t = Start[ u ]; t; t = Edge[ t ].Next ) {
int v = Edge[ t ].To;
if( v == Fa ) continue;
int T = Dfs1( v, u );
if( T > n / 2 ) IsR = 0;
Size += T;
}
if( n - Size > n / 2 ) IsR = 0;
if( IsR ) Rt[ ++Cnt ] = u;
return Size;
}
void Cal( int u, int Fa ) {
for( int t = Start[ u ]; t; t = Edge[ t ].Next ) {
int v = Edge[ t ].To;
if( v == Fa ) continue;
Cal( v, u );
}
A[ u ] = "0";
int Cnt = 0;
for( int t = Start[ u ]; t; t = Edge[ t ].Next ) {
int v = Edge[ t ].To;
if( v == Fa ) continue;
B[ ++Cnt ] = A[ v ];
}
sort( B + 1, B + Cnt + 1 );
for( int i = 1; i <= Cnt; ++i )
A[ u ] = A[ u ] + B[ i ];
A[ u ] = A[ u ] + "1";
return;
}
int main() {
scanf( "%d", &m );
for( int i = 1; i <= m; ++i ) {
memset( Start, 0, sizeof( Start ) );
Used = 0;
scanf( "%d", &n );
for( int j = 1; j <= n; ++j ) {
int x;
scanf( "%d", &x );
if( x == 0 ) continue;
AddEdge( x, j );
AddEdge( j, x );
}
Cnt = 0;
Dfs1( 1, 0 );
Cal( Rt[ 1 ], 0 );
S = A[ Rt[ 1 ] ];
for( int j = 2; j <= Cnt; ++j ) {
Cal( Rt[ j ], 0 );
if( A[ Rt[ j ] ] < S )
S = A[ Rt[ j ] ];
}
if( Map.find( S ) == Map.end() ) Map[ S ] = i;
printf( "%d\n", Map[ S ] );
}
return 0;
}
[BJOI2015]树的同构 && 树哈希教程的更多相关文章
- bzoj4337: BJOI2015 树的同构 树哈希判同构
题目链接 bzoj4337: BJOI2015 树的同构 题解 树哈希的一种方法 对于每各节点的哈希值为hash[x] = hash[sonk[x]] * p[k]; p为素数表 代码 #includ ...
- BZOJ 4337: BJOI2015 树的同构 树hash
4337: BJOI2015 树的同构 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=4337 Description 树是一种很常见的数 ...
- BZOJ4337:[BJOI2015]树的同构(树hash)
Description 树是一种很常见的数据结构. 我们把N个点,N-1条边的连通无向图称为树. 若将某个点作为根,从根开始遍历,则其它的点都有一个前驱,这个树就成为有根树. 对于两个树T1和T2,如 ...
- [BZOJ4337][BJOI2015]树的同构(树的最小表示法)
4337: BJOI2015 树的同构 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1023 Solved: 436[Submit][Status ...
- 【BZOJ4474】isomorphism(树的同构,哈希)
题意:一个无向树的度数为 2的结点称为假结点,其它结点称为真结点.一个无向树的简化树其结点由原树的全体真结点组成,两个真结点之间有边当且仅当它们在原树中有边,或者在原树中有一条联结这两个结点的路,其中 ...
- BZOJ.4337.[BJOI2015]树的同构(树哈希)
BZOJ 洛谷 \(Description\) 给定\(n\)棵无根树.对每棵树,输出与它同构的树的最小编号. \(n及每棵树的点数\leq 50\). \(Solution\) 对于一棵无根树,它的 ...
- BZOJ4337 树的同构 (树哈希)(未完成)
样例迷,没过 交了30pts #include <cstdio> #include <iostream> #include <cstring> #include & ...
- [BJOI2015]树的同构
嘟嘟嘟 判断树的同构的方法就是树上哈希. 如果树是一棵有根树,那么只要从根节点出发dfs,每一个节点的哈希值等于按传统方式算出来的子树的哈希值的结果.需要注意的是,算完子树的哈希值后要先排序再加起来, ...
- 【BZOJ4337】BJOI2015 树的同构 括号序列
[BZOJ4337]BJOI2015 树的同构 Description 树是一种很常见的数据结构. 我们把N个点,N-1条边的连通无向图称为树. 若将某个点作为根,从根开始遍历,则其它的点都有一个前驱 ...
随机推荐
- 《深入理解 Java 虚拟机》学习 -- 垃圾收集器
<深入理解 Java 虚拟机>学习 -- 垃圾收集器 1. Serial 收集器(新生代) 含义: 单线程收集器. 缺点: 进行垃圾收集时,必须暂停其他所有的工作线程. 优点: 简单而高效 ...
- 10.Bash的安装
10.Bash的安装本节提供了在 Bash支持的不同系统上的基本安装指导.本版本支持 GNU操作系统,几乎每个 UNIX版本,以及几个非 UNIX 系统,例如 BeOS 和 Interix.还有针对 ...
- Tomcat中的服务器组件和 服务组件
开始学习Tocmat时,都是学习如何通过实例化一个连接器 和 容器 来获得一个Servlet容器,并将连接器 和 servlet容器相互关联,但是之前学习的都只有一个连接器可以使用,该连接器服务80 ...
- IIS Express启动不了的的解决方案
netsh http show iplisten netsh http delete iplisten ipaddress=11.22.33.44 (where 11.22.33.44 is the ...
- C#中word文档转html
var path = Request.Url.Host + ":" + Request.Url.Port + list[i].AnnexPath; //html保存路径 strin ...
- C# WebApi日期格式化
WebApi中日期格式化:在WebApiConfig文件中加入如下代码即可,之前遇到的问题,日期中总带有T,现在记录一下解决的方法. 代码: private static void ReturnDat ...
- 百度URL链接中文转码
百度搜索链接规则为: http://www.baidu.com/s?wd=[搜索词目]&cl=3 有多个搜索词通过加号进行链接: http://www.baidu.com/s?wd=keywo ...
- MySQL四舍五入函数ROUND(x)、ROUND(x,y)和TRUNCATE(x,y)
MySQL四舍五入函数ROUND(x) ROUND(x)函数返回最接近于参数x的整数,对x值进行四舍五入. 实例: 使用ROUND(x)函数对操作数进行四舍五入操作.SQL语句如下: mysql> ...
- 实现单点登录功能的思路以及kafka同步数据
单点登录以及用户数据同步思路与方案 当公司业务分布于多个子系统时, 同一用户在A系统注册,即可在其他所有关联系统使用, 并支持登录A系统后,自动在其他系统登录,退出同理. 在A平台修改通用的用户数据( ...
- 关于STM32运行时程序卡在B.处的解决方法
文章转载自:https://blog.csdn.net/u014470361/article/details/78780444 背景: 程序运行时,发现程序卡死在B.处. 解决方法: 程序卡死在B.处 ...