51Nod1601 完全图的最小生成树计数
我居然忘写题解啦!(记忆废)
不管怎么说,这题还算是一道好题啊……你觉得敦爷出的题会有水题么
……
这题比较容易把人误导到Boruvka算法之类的东西上去(我们机房去刚D题的人一开始大多也被误导了),但仔细思考之后是可以发现问题的特殊性质的。
听说很多人是从Kruskal算法想到这道题的做法的?好吧我并不是,那我就写写我的思考过程好了……
记得算导上有一道思考题,判断一个最小生成树算法的正确性。那个算法是这样的:把当前图的点集随意划分成两半,递归两半后选出连接两个点集的边中权值最小的一条,得到最后的最小生成树。
这个算法显然是错的,因为最终的最小生成树中可能有两条连接当前层两个点集的边。但本题有特殊性,边权都是端点点权的异或值,也许可以把这个算法改造一下用到这道题中。
考虑对一个点集求最小生成树,由于边权是端点点权的异或,因此可以把所有点按照最高位划分成两半,一半最高位为0,另一半最高位为1。这样递归两半之后只选一条连接两半的最小权边就可以得到一个生成树。可以证明它是最小生成树,并且证明起来并不难:如果某一组解中有两条边都连接了两半,由于这两条边边权的最高位一定是1,而位于两半内的边边权最高位一定是0,因此把这两条边中的一条替换成两半内的边得到的解一定比当前优。
有了正确性,算法也就成型了:每次把当前点集按最高位划分为两半后递归两半,然后若两边均非空则把任意一条连接两半的最小权边加入最小生成树即可。可能有多条边都是最小权边,显然方案数应该是每层的方案数之积。
实现的时候对所有权值建一棵01-Trie,那么选边的过程就相当于对每个点计算它的左子树和右子树之间的贡献,再dfs一遍左右子树即可。每个数最多被dfs到$\frac{32^2}2=512$次,因此复杂度为$O(512n)$。
注意一个细节:在划分过程中如果递归到了叶子节点且此处点数$>1$,则说明有多个点权值相同,显然这些点随便连就行了,那么答案就应该乘上对应点数的无向完全图生成树的数量。这个在OEIS上可以找到,通项是$n^{n-2}$。
#include<cstdio>
#include<cstring>
#include<cassert>
#include<algorithm>
using namespace std;
const int maxn=,maxm=maxn<<,p=1e9+;
void insert(int,int&);
void solve(int,int);
void dfs(int,int,int,int);
int qpow(int,int);
long long sum=;
int sm[maxm]={},ch[maxm][]={{}},root=,cnt=;
int n,a[maxn],x,ans=,mn,tmp;
signed main(){
scanf("%d",&n);
for(int i=;i<=n;i++){
scanf("%d",&x);
insert(,root);
}
solve(,root);
//assert(ans==1ll);
printf("%lld\n%d",sum,ans);
return ;
}
void insert(int k,int &rt){
if(!rt)rt=++cnt;
sm[rt]++;
if(k==-)return;
insert(k-,ch[rt][(x>>k)&]);
}
void solve(int k,int x){
if(sm[x]<=)return;//printf("solve(%d,%d)\n",k,x);
if(k==-){
ans=(long long)ans*qpow(sm[x],sm[x]-)%p;
return;
}
solve(k-,ch[x][]);
solve(k-,ch[x][]);
if(sm[ch[x][]]&&sm[ch[x][]]){
mn=;
tmp=;
dfs(k-,ch[x][],ch[x][],<<k);//printf("solve(%d,%d)\n",k,x);printf("mn=%d tmp=%d\n",mn,tmp);
sum+=mn;
ans=(long long)ans*tmp%p;
}
}
void dfs(int k,int x,int y,int now){
if(!sm[x]||!sm[y])return;
if(k==-){
if(now<mn){
mn=now;
tmp=(long long)sm[x]*sm[y]%p;
}
else if(now==mn)tmp=(tmp+(long long)sm[x]*sm[y]%p)%p;
return;
}
if(sm[ch[x][]]){
if(sm[ch[y][]])dfs(k-,ch[x][],ch[y][],now);
else dfs(k-,ch[x][],ch[y][],now|(<<k));
}
if(sm[ch[x][]]){
if(sm[ch[y][]])dfs(k-,ch[x][],ch[y][],now);
else dfs(k-,ch[x][],ch[y][],now|(<<k));
}
}
int qpow(int a,int b){
int ans=;
for(;b;b>>=,a=(long long)a*a%p)if(b&)ans=(long long)ans*a%p;
return ans;
}
51Nod1601 完全图的最小生成树计数的更多相关文章
- 51Nod1601 完全图的最小生成树计数 Trie Prufer编码
原文链接https://www.cnblogs.com/zhouzhendong/p/51Nod1601.html 题目传送门 - 51Nod1601 题意 题解 首先我们考虑如何求答案. 我们将所有 ...
- 51Nod 1601 完全图的最小生成树计数
题目链接 分析: 这是一张完全图,并且边的权值是由点的权值$xor$得到的,所以我们考虑贪心的思想,考虑$kruskal$的过程选取最小的边把两个连通块合并,所以我们可以模仿$kruskal$的过程, ...
- 「51Nod 1601」完全图的最小生成树计数 「Trie」
题意 给定\(n\)个带权点,第\(i\)个点的权值为\(w_i\),任意两点间都有边,边权为两端点权的异或值,求最小生成树边权和,以及方案数\(\bmod 10^9 + 7\) \(n \leq 1 ...
- 最小生成树计数 bzoj 1016
最小生成树计数 (1s 128M) award [问题描述] 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一 ...
- 树的Prufer 编码和最小生成树计数
Prufer数列 Prufer数列是无根树的一种数列.在组合数学中,Prufer数列由有一个对于顶点标过号的树转化来的数列,点数为n的树转化来的Prufer数列长度为n-2.它可以通过简单的迭代方 ...
- 【bzoj1016】 JSOI2008—最小生成树计数
http://www.lydsy.com/JudgeOnline/problem.php?id=1016 (题目链接) 题意 求图的最小生成树计数. Solution %了下题解,发现要写矩阵树,15 ...
- [BZOJ]1016 JSOI2008 最小生成树计数
最小生成树计数 题目描述 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一条边不同,则这两个最小生成树就是不同 ...
- bzoj1016 [JSOI2008]最小生成树计数
1016: [JSOI2008]最小生成树计数 Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 3517 Solved: 1396[Submit][St ...
- 【BZOJ】【1016】【JSOI2008】最小生成树计数
Kruskal/并查集+枚举 唉我还是too naive,orz Hzwer 一开始我是想:最小生成树删掉一条边,再加上一条边仍是最小生成树,那么这两条边权值必须相等,但我也可以去掉两条权值为1和3的 ...
随机推荐
- Python3之collections模块
简介 collections是Python内建的一个集合模块,提供了许多有用的集合类. namedtuple namedtuple 是一个函数,它用来创建一个自定义的元组对象,并且规定了元组元素的个数 ...
- luogu5282 【模板】快速阶乘算法
由于巨佬 shadowice1984 卡时限,本代码已经 T 请不要粘上去交 退役之后再写一个常数小的多项式取模吧 一句话题意:NP问题,求N!%P 吐槽:出题人太毒瘤...必须写任意模数NTT,而且 ...
- html头文件设置常用之<meta>设置缓存
<meta http-equiv="pragma" content="no-cache">,pragma与no-cache用于定义页面缓存,不缓存页 ...
- HNOI2018 退役记
HNOI2018 退役记 \(day0\): 除了切水题以外没有什么很重要的事. \(day1\): 进考场发现前面是\(yyb\)?\(orzyyb\) 试题解压密码终于没有奇怪的字符了,一遍打对. ...
- SCTP
流控制传输协议是为了替代UDP.TCP实现七号信令传输的. HS DPA high speed download packet access. HS UPA ... 通信人 Orthogonal f ...
- Q767 重构字符串
给定一个字符串S,检查是否能重新排布其中的字母,使得两相邻的字符不同. 若可行,输出任意可行的结果.若不可行,返回空字符串. 示例 1: 输入: S = "aab" 输出: &qu ...
- Java操作系统剪贴板(Clipboard)复制粘贴
Java操作系统剪贴板(Clipboard)复制粘贴
- pycharm 工具栏Tool中找不到Run manager.py Task
pycharm 工具栏Tool中找不到Run manager.py Task 在做Django项目的过程中, 无法进入pycharm提供的Run manager.py Task交互环境 出现这种问题是 ...
- Python操作Excel(将父子级表头生成树状结构)
import re class Node: ''' 容器,用来存储前后节点信息 ''' __slot__=[] def __init__(self,val,next_,pre,name,no): se ...
- 解决Eclipse中Android SDK Manager图标不见了的问题
在Eclipse中安装完ADT后,发现 Android SDK Manager图标不见了,Android Virtual Device Manager图标也不见了. 解决这个问题,可用如下方法: 打开 ...