独钓寒江雪

题目链接https://www.lydsy.com/JudgeOnline/problem.php?id=3162


题解

首先,如果没有那个本质相同的限制这就是个傻逼题。

直接树形dp就好。

那么如果加上那个限制呢?

我们发现,无论最后怎么本质相同,树的重心一定不变。

故此,从重心开始去重即可。

参考:https://www.cnblogs.com/zhoushuyu/p/9295759.html

代码

#include <bits/stdc++.h>

#define N 500010 

using namespace std;

const int mod = 1000000007 ;

char *p1, *p2, buf[100000];

typedef unsigned long long ll;

ll bs1 = 20021214 ;

ll bs2 = 20030315 ;

#define nc() (p1 == p2 && (p2 = (p1 = buf) + fread(buf, 1, 100000, stdin), p1 == p2) ? EOF : *p1 ++ )

int rd() {
int x = 0;
char c = nc();
while (c < 48) {
c = nc();
}
while (c > 47) {
x = (((x << 2) + x) << 1) + (c ^ 48), c = nc();
}
return x;
} int n, inv[N], to[N << 1], nxt[N << 1], head[N], tot, sz[N], w[N], root, rt1, rt2, flag, f[2][N], tmp[N]; ll hs[N]; inline void add(int x, int y) {
to[ ++ tot] = y;
nxt[tot] = head[x];
head[x] = tot;
} void getroot(int p, int fa) {
sz[p] = 1;
w[p] = 0;
for (int i = head[p]; i; i = nxt[i]) {
if(to[i] != fa) {
getroot(to[i], p);
sz[p] += sz[to[i]];
w[p] = max(w[p], sz[to[i]]);
}
}
w[p] = max(w[p], n - sz[p]);
if (w[p] < w[root]) {
root = p;
}
} int C(int n, int m) {
int re = 1;
for (int i = n - m + 1; i <= n; i ++ ) {
re = (ll)re * i % mod;
}
for (int i = 1; i <= m; i ++ ) {
re = (ll)re * inv[i] % mod;
}
return re;
} inline bool cmp(int i, int j) {
return hs[i] < hs[j];
} void dfs(int p, int fa) {
sz[p] = f[0][p] = f[1][p] = 1;
for (int i = head[p]; i; i = nxt[i]) {
if (to[i] != fa) {
dfs(to[i], p);
sz[p] += sz[to[i]];
}
}
int len = 0;
for (int i = head[p]; i; i = nxt[i]) {
if (to[i] != fa) {
tmp[ ++ len] = to[i];
}
}
sort(tmp + 1, tmp + len + 1, cmp);
for (int i = 1, j = 1; i <= len; i = j) {
while (j <= len && hs[tmp[j]] == hs[tmp[i]]) {
j ++ ;
}
f[0][p] = (ll)f[0][p] * C(f[0][tmp[i]] + f[1][tmp[i]] + j - i - 1, j - i) % mod;
f[1][p] = (ll)f[1][p] * C(f[0][tmp[i]] + j - i - 1, j - i) % mod;
}
hs[p] = bs2 * len + sz[p];
for (int i = 1; i <= len; i ++ ) {
hs[p] = (hs[p] * bs1) ^ hs[tmp[i]];
}
} int main() {
n = rd();
inv[0] = inv[1] = 1;
for (int i = 2; i <= n; i ++ ) {
inv[i] = (ll)inv[mod % i] * (mod - mod / i) % mod;
}
for (int i = 1; i < n; i ++ ) {
int x = rd(), y = rd();
add(x, y), add(y, x);
}
w[0] = n;
getroot(1, 0);
getroot(root, 0);
for (int i = head[root], last = 0; i; last = i, i = nxt[i]) {
if (sz[to[i]] * 2 == n) {
n ++ ;
if (i == head[root]) {
head[root] = nxt[i];
}
else {
nxt[last] = nxt[i];
}
for (int j = head[to[i]], lst = 0; j; lst = j, j = nxt[j]) {
if (to[j] == root) {
if (j == head[to[i]]) {
head[to[i]] = nxt[j];
}
else {
nxt[lst] = nxt[j];
}
break;
}
}
add(n, root);
add(root, n);
add(n, to[i]);
add(to[i], n);
rt1 = root;
rt2 = to[i];
root = n;
flag = 1;
break;
}
}
dfs(root, 0);
if (!flag) {
cout << (f[0][root] + f[1][root]) % mod << endl ;
}
else if (hs[rt1] == hs[rt2]) {
cout << (f[0][root] - C(f[1][rt1] + 1, 2) + mod) % mod;
}
else {
cout << ((ll)f[0][rt1] * f[0][rt2] + (ll)f[0][rt1] * f[1][rt2] + (ll)f[1][rt1] * f[0][rt2]) % mod << endl ;
}
}

小结:树hash的时候,如果一个模数担心过不去,像我一样开两个就好了233

[bzoj3162]独钓寒江雪_树hash_树形dp的更多相关文章

  1. BZOJ3162 独钓寒江雪(哈希+树形dp)

    数独立集显然是可以树形dp的,问题在于本质不同. 假设已经给树确立了一个根并且找到了所有等效(注意是等效而不是同构)子树,那么对转移稍加修改使用隔板法就行了. 关键在于找等效子树.首先将树的重心(若有 ...

  2. BZOJ_2097_[Usaco2010 Dec]Exercise 奶牛健美操_二分答案+树形DP

    BZOJ_2097_[Usaco2010 Dec]Exercise 奶牛健美操_二分答案+树形DP Description Farmer John为了保持奶牛们的健康,让可怜的奶牛们不停在牧场之间 的 ...

  3. BZOJ_1999_[Noip2007]Core树网的核_单调队列+树形DP

    BZOJ_1999_[Noip2007]Core树网的核_单调队列+树形DP Description 设T=(V, E, W) 是一个无圈且连通的无向图(也称为无根树),每条边带有正整数的权,我们称T ...

  4. 【NOI P模拟赛】仙人掌(圆方树,树形DP)

    题面 n n n 个点, m m m 条边. 1 ≤ n ≤ 1 0 5 , n − 1 ≤ m ≤ 2 × 1 0 5 1\leq n\leq 10^5,n-1\leq m\leq 2\times1 ...

  5. bzoj 1040 [ZJOI2008]骑士(基环外向树,树形DP)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1040 [题意] 给一个基环森林,每个点有一个权值,求一个点集使得点集中的点无边相连且权 ...

  6. 洛谷4895 独钓寒江雪 (树哈希+dp+组合)

    qwq 首先,如果是没有要求本质不同的话,那么还是比较简单的一个树形dp 我们令\(dp[i][0/1]\)表示是否\(i\)的子树,是否选\(i\)这个点的方案数. 一个比较显然的想法. \(dp[ ...

  7. 青云的机房组网方案(简单+普通+困难)(虚树+树形DP+容斥)

    题目链接 1.对于简单的版本n<=500, ai<=50 直接暴力枚举两个点x,y,dfs求x与y的距离. 2.对于普通难度n<=10000,ai<=500 普通难度解法挺多 ...

  8. 洛谷AT2046 Namori(思维,基环树,树形DP)

    洛谷题目传送门 神仙思维题还是要写点东西才好. 树 每次操作把相邻且同色的点反色,直接这样思考会发现状态有很强的后效性,没办法考虑转移. 因为树是二分图,所以我们转化模型:在树的奇数层的所有点上都有一 ...

  9. Luogu4630 APIO2018 Duathlon 圆方树、树形DP

    传送门 要求的是一条按顺序经过\(s,t,c\)三个点的简单路径.简单路径的计数问题不难想到点双联通分量,进而使用圆方树进行求解. 首先将原图缩点,对于一个大小为\(size\)的点双联通分量内,在这 ...

随机推荐

  1. is(expr|obj|ele|fn)

    is(expr|obj|ele|fn) 概述 根据选择器.DOM元素或 jQuery 对象来检测匹配元素集合,如果其中至少有一个元素符合这个给定的表达式就返回true.大理石平台支架 如果没有元素符合 ...

  2. 诊断和修复Web测试记录器(Web Test Recorder)问题

    http://www.cnblogs.com/oscarxie/articles/1045430.html Database LoadTest2010 script C:\Program Files ...

  3. [Luogu] 无线网络发射器选址

    https://www.luogu.org/problemnew/show/P2038 二维前缀和 #include <iostream> #include <cstdio> ...

  4. 顺序表应用1:多余元素删除之移位算法(SDUT 3324)

    Problem Description 一个长度不超过10000数据的顺序表,可能存在着一些值相同的"多余"数据元素(类型为整型),编写一个程序将"多余"的数据 ...

  5. MySQL 5.7半同步复制技术

    一.复制架构衍生史 在谈这个特性之前,我们先来看看MySQL的复制架构衍生史. 在2000年,MySQL 3.23.15版本引入了Replication.Replication作为一种准实时同步方式, ...

  6. 网站快速收录/站点快速收录/seo快排技术/seo快速排名/泛域名快速收录/泛目录快速收录

    泛目录技术是目前最快速最有效的办法,增加站点的收录方面,这里推荐莲花泛目录 莲花泛目录,完善的技术支持,代码亲自编写,独立研发.业界领先. (这个是seo关键词,请无视,直接看下面的泛目程序介绍)网站 ...

  7. mybatis之<trim

    1.<trim prefix="" suffix="" suffixOverrides="" prefixOverrides=&quo ...

  8. C#_switch语句,for循环,do while循环,while循环

    1:switch语句,代码如下: using System; using System.Collections.Generic; using System.Linq; using System.Tex ...

  9. GDB之调试器用法

    GDB 完成的作用: 启动程序,可以按照工程师自定义的要求随心所欲的运行程序 让被调试的程序在工程师指定的断点处停住,断点可以是条件表达式 当程序被停住时,可以检查此时程序中所发生的事,并追索上文 动 ...

  10. No Desktop License Servers available to provide a license

    远程桌面连接失败,提示:“no Remote Desktop License Servers available to provide a license” 原因:没有remote desktop l ...