先贴上地址 https://vjudge.net/problem/HDU-5732

判断有根树同构:

1. 直接用括号最小表示法

2. 利用括号最小表示法的思想进行hash

判断无根树同构:

1. 找到树的重心.

2. 以重心为根, 把无根树转化成有根树. 按照有根树同构的方法判断是否同构.

同构的过程中,为什么可以sort.

我们知道,对于树来说,

树的节点绕着它的父节点旋转,树的结构就不会被改变的.

所以sort的过程就相当于把树的节点绕着它的父节点进行旋转.

// sort的话,可以这么理解:

我们是按照同样的规则(我们保证这个规则可以唯一确定一棵树,),

对两棵树进行操作,如果同构的话,那么结果应该是一样的. 如果不同构的画 那么结果就是不一样的

至于树的括号表达式, 可以见这几个博客

括号表达式       https://www.byvoid.com/zhs/blog/directed-tree-bracket-sequence

树同构              https://blog.csdn.net/u010152669/article/details/9116975

树的表示方法   https://www.cnblogs.com/jsawz/p/6807636.html

#include <cstdio>
#include <ctime>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <map>
#include <vector> using namespace std; typedef unsigned long long ull;
const int maxn = 1e5+;
ull x[maxn];
int ans[maxn]; struct Edge {
int lst;
int to;
}; class Tree {
public :
Edge edge[maxn<<];
int head[maxn];
int cn, csz, ccnt, mid[];
int rcd[maxn];
ull cnode[maxn];
ull vvvvv[];
map<string, int> id;
char name[maxn][]; inline void add(int u, int v) {
edge[csz].lst = head[u];
edge[csz].to = v;
head[u] = csz++;
} int dfs(int u, int fa) {
int i, v, res = , t1;
for (i=head[u]; i; i=edge[i].lst) {
v = edge[i].to;
if (v == fa) continue;
t1 = dfs(v, u);
res += t1;
if (rcd[u] < t1) rcd[u] = t1;
if (rcd[v] < cn-t1) rcd[v] = cn-t1;
}
return res + ;
} ull dfs2(int u, int fa, int deep) {
int i, v, res = , t1;
vector<ull> son;
for (i=head[u]; i; i=edge[i].lst) {
v = edge[i].to;
if (v == fa) continue;
son.push_back(dfs2(v, u, deep+));
}
sort(son.begin(), son.end()); // 判断树同构 注意要sort后乘一个随机数(或者一个质数的i次方). 为什么可以sort呢? 因为树同一层的节点围绕父节点旋转不会改变树的结构. 而sort相当于把节点绕父节点旋转了.
for (i=, t1=son.size(); i<t1; ++i)
res += son[i] * x[i+];
return cnode[u] = (res ? res : x[deep]);
} void init(int n) {
cn = n; csz = ; id.clear();
int i, u, v, cnt = ;
char s1[], s2[]; for (i=; i<=n; ++i) rcd[i] = head[i] = ; for (i=; i<n; ++i) {
scanf("%s%s", s1, s2);
if (!(u = id[s1])) {
strcpy(name[cnt], s1);
id[s1] = u = cnt++;
} if (!(v = id[s2])) {
strcpy(name[cnt], s2);
id[s2] = v = cnt++;
}
add(u, v); add(v, u);
} dfs(, -);
int mm = 0x3f3f3f3f;
for (i=; i<=n; ++i) {
if (mm > rcd[i]) {
mm = rcd[i];
ccnt = ;
mid[++ccnt] = i;
} else if (mm == rcd[i]) mid[++ccnt] = i;
} for (i=; i<=ccnt; ++i)
vvvvv[i] = dfs2(mid[i], -, );
}
}te1, te2; struct nobe {
int id;
ull val;
bool operator < (const nobe &a) const {
return val < a.val;
}
nobe () {}
nobe (int iid, ull vval) : id(iid), val(vval) {}
}; void dfs3(int u1, int u2, int fa1, int fa2) {
int i, v1, v2, tsz;
ans[u1] = u2;
vector<nobe> ve1, ve2;
for (i=te1.head[u1]; i; i=te1.edge[i].lst) {
v1 = te1.edge[i].to;
if (v1 == fa1) continue;
ve1.push_back(nobe(v1, te1.cnode[v1]));
}
for (i=te2.head[u2]; i; i=te2.edge[i].lst) {
v2 = te2.edge[i].to;
if (v2 == fa2) continue;
ve2.push_back(nobe(v2, te2.cnode[v2]));
}
sort(ve1.begin(), ve1.end());
sort(ve2.begin(), ve2.end());
for (i=, tsz=ve1.size(); i<tsz; ++i)
dfs3(ve1[i].id, ve2[i].id, u1, u2);
} int main()
{
int i, j, k, n;
srand(time(NULL));
for (i=; i<maxn; ++i) x[i] = rand();
while (~scanf("%d", &n)) {
te1.init(n);
te2.init(n);
for (i=; i<=te1.ccnt; ++i)
for (j=; j<=te2.ccnt; ++j) {
if (te1.vvvvv[i] == te2.vvvvv[j]) {
dfs3(te1.mid[i], te2.mid[j], te1.mid[i], te2.mid[j]);
for (k=; k<=n; ++k)
printf("%s %s\n", te1.name[k], te2.name[ans[k]]);
goto A;
}
}
A: ;
} return ;
}

无根树同构_hash的更多相关文章

  1. 3194. 【HNOI模拟题】化学(无标号无根树计数)

    Problem 求\(n\)个点的每个点度数不超过\(4\)的无标号无根树个数. Data constraint \(1\le n\le 500\) Solution 尝试着把问题一般化.我们来考虑一 ...

  2. BZOJ 4754 [JSOI2016]独特的树叶 | 树哈希判同构

    题目链接 这道题是一道判断无根树同构的模板题,判断同构主要的思路就是哈希. 一遇到哈希题,一百个人能有一百零一种哈希方式,这篇题解随便选用了一种--类似杨弋<Hash在信息学竞赛中的一类应用&g ...

  3. 【CF1252F】Regular Forestation(重心,树同构)

    题意:给定一棵n个点的树,问删去某个点之后所有的树同构,这样分割出来的树最多能有几棵 n<=4000 思路:分割成至少两个size相等的联通块之后size必定小于n/2,与树的重心的定义相同 预 ...

  4. 【BZOJ4337】树的同构(树同构,哈希)

    题意: 树是一种很常见的数据结构. 我们把N个点,N-1条边的连通无向图称为树. 若将某个点作为根,从根开始遍历,则其它的点都有一个前驱,这个树就成为有根树. 对于两个树T1和T2,如果能够把树T1T ...

  5. ustc 1117

    无根树同构 有两种方法,一种是确定其中一棵树,另一棵树枚举根节点. 一种是,利用拓扑排序,先确定其中一棵树.另一棵树,若拓扑后剩两个节点,则枚举这两个节点为根结点,否则,只需做一次.注意,无根树节点入 ...

  6. 树hash

    判断树的同构,采用树hash的方式. 树hash定义在有根树上.判断无根树同构的时候,可以比较重心为根的hash值或者比较每个点为根的hash值. h[x]表示x为根的子树的hash,g[x]表示x为 ...

  7. uva12489 Combating cancer(树同构)

    转载请注明出处: http://www.cnblogs.com/fraud/          ——by fraud https://uva.onlinejudge.org/index.php?opt ...

  8. 4337: BJOI2015 树的同构

    题解: 树的同构的判定 有根树从根开始进行树hash 先把儿子的f进行排序 $f[i]=\sum_{j=1}^{k} { f[j]*prime[j]} +num[i]$(我没有仔细想这样是不是树是唯一 ...

  9. [BJOI2015]树的同构

    嘟嘟嘟 判断树的同构的方法就是树上哈希. 如果树是一棵有根树,那么只要从根节点出发dfs,每一个节点的哈希值等于按传统方式算出来的子树的哈希值的结果.需要注意的是,算完子树的哈希值后要先排序再加起来, ...

随机推荐

  1. iOS性能优化总结

    iOS性能优化总结.关于 iOS 性能优化梳理: 基本工具.业务优化.内存优化.卡顿优化.布局优化.电量优化. 安装包瘦身.启动优化.网络优化等. 关于iOS 性能优化梳理: 基本工具.业务优化.内存 ...

  2. 百度AI认为最漂亮的中国女星是----范冰冰

    一.程序说明 1.1 程序说明 之前写调用百度AI接口的程序,然后刷到了两条明星的新闻,就想到了写个给明星颜值排下名的程序. 程序的关键点是两个,第一个是百度AI接口的调用这点其实直接使用早前实现的类 ...

  3. Swagger 路径过滤 -PreSerializeFilters

    Swagger 默认显示所有api, 如果要做路径过滤,可以这样做. //过滤,只显示部分api app.UseSwagger(c=> { c.PreSerializeFilters.Add(( ...

  4. samba共享文件夹设置

    sudo apt-get install samba(4) mkdir /home/用户名/share (新建share文件夹) sudo cp /etc/samba/smb.conf /etc/sa ...

  5. laravel的日志服务

    日志服务使用: $app=app(); $log=$app->make('log'); $log->info('post_index',['data'=>'this is post ...

  6. CSS技巧-文字分散对齐的方法

    下面的代码可以在IE中实现文字分散对齐: <table width="300" align="center">    <tr>      ...

  7. 什么是Java优先级队列(Priority Queue)?

    PriorityQueue是一个基于优先级堆的无界队列.它的元素是按照自然顺序排序的.在创建元素的时候,我们给它一个一个负责排序的比较器.PriorityQueue不允许null值,因为 它们没有自然 ...

  8. SQL-38 创建视图

    题目描述 针对actor表创建视图actor_name_view,只包含first_name以及last_name两列,并对这两列重新命名,first_name为first_name_v,last_n ...

  9. 第二节 java流程控制(循环结构)

     1.for循环 for(初始化表达式;循环条件表达式;循环后的操作表达式){ 执行语句 } 2.while循环 while(条件表达式){ 执行语句 } while循环特点是只有条件满足才会执行我们 ...

  10. win8 下面 listen 的队列长度貌似无效了 上c/s 代码 并附截图,有图有真相

    #include <WinSock2.h> #include <stdio.h> #include <windows.h> DWORD ServerRoutine( ...