传送门

我居然忘写题解啦!(记忆废)

不管怎么说,这题还算是一道好题啊……你觉得敦爷出的题会有水题么

……

这题比较容易把人误导到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 完全图的最小生成树计数的更多相关文章

  1. 51Nod1601 完全图的最小生成树计数 Trie Prufer编码

    原文链接https://www.cnblogs.com/zhouzhendong/p/51Nod1601.html 题目传送门 - 51Nod1601 题意 题解 首先我们考虑如何求答案. 我们将所有 ...

  2. 51Nod 1601 完全图的最小生成树计数

    题目链接 分析: 这是一张完全图,并且边的权值是由点的权值$xor$得到的,所以我们考虑贪心的思想,考虑$kruskal$的过程选取最小的边把两个连通块合并,所以我们可以模仿$kruskal$的过程, ...

  3. 「51Nod 1601」完全图的最小生成树计数 「Trie」

    题意 给定\(n\)个带权点,第\(i\)个点的权值为\(w_i\),任意两点间都有边,边权为两端点权的异或值,求最小生成树边权和,以及方案数\(\bmod 10^9 + 7\) \(n \leq 1 ...

  4. 最小生成树计数 bzoj 1016

    最小生成树计数 (1s 128M) award [问题描述] 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一 ...

  5. 树的Prufer 编码和最小生成树计数

      Prufer数列 Prufer数列是无根树的一种数列.在组合数学中,Prufer数列由有一个对于顶点标过号的树转化来的数列,点数为n的树转化来的Prufer数列长度为n-2.它可以通过简单的迭代方 ...

  6. 【bzoj1016】 JSOI2008—最小生成树计数

    http://www.lydsy.com/JudgeOnline/problem.php?id=1016 (题目链接) 题意 求图的最小生成树计数. Solution %了下题解,发现要写矩阵树,15 ...

  7. [BZOJ]1016 JSOI2008 最小生成树计数

    最小生成树计数 题目描述 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一条边不同,则这两个最小生成树就是不同 ...

  8. bzoj1016 [JSOI2008]最小生成树计数

    1016: [JSOI2008]最小生成树计数 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 3517  Solved: 1396[Submit][St ...

  9. 【BZOJ】【1016】【JSOI2008】最小生成树计数

    Kruskal/并查集+枚举 唉我还是too naive,orz Hzwer 一开始我是想:最小生成树删掉一条边,再加上一条边仍是最小生成树,那么这两条边权值必须相等,但我也可以去掉两条权值为1和3的 ...

随机推荐

  1. django执行过程

  2. C#-WinForm-★★★★★跨窗体 构造函数传值 及应用—登录式窗口传值、如何关闭主页面时关闭应用程序、如何打开唯一窗口★★★★★

    构造函数可以传任意类型的值,并可以同时传多个值 结构函数传值的初步应用——简单的登陆式界面 现在我有两个窗体Form3和Form4,如下,如何点击Form3中的按钮后,打开Form4并将Form3中的 ...

  3. HTML-制作图片的自动播放和手动切换

    思路:将想要播放的图片放入集合中,设置一个div,将图片依次从集合中定时取出放到div中展示:设置一个变量,通过变量与集合元素索引联系起来,点击改变时,获取当前图片的索引以切换图片 整体代码: < ...

  4. 什么是hive

    Hadoop Hive概念学习系列之什么是Hive? 参考  <Hadoop大数据分析与挖掘实战>的在线电子书阅读                   http://yuedu.baidu ...

  5. PIE SDK与OpenCV结合说明文档

    1.功能简介 OpenCV是基于BSD许可(开源)发行的跨平台计算机视觉库,可以运行在Linux.Windows.Android和Mac OS操作系统上.它轻量级而且高效——由一系列 C 函数和少量 ...

  6. 使用express、react、webpack打包、socket.io、mongodb、ant.design、less、es6实现聊天室

    拿到一个项目,我们应该如何去完成这个项目呢. 是直接上手? 还是先进行分析,然后再去解决呢?毫无疑问,如果直接上手解决,那么可能会因为知道目标所在,而导致出现各种问题. 所以,我们应该系统的分析这个项 ...

  7. 【Lua】关于遍历指定路径下所有目录及文件

    关于Lua中如何遍历指定文件路径下的所有文件,需要用到Lua的lfs库. 首先创建一个temp.lua文件,用编辑器打开: 要使用lfs库,首先需要把lfs库加载进来 require("lf ...

  8. JVM的内存结构

    程序计数器 程序计数器(Program Counter Register)是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器.字节码解释器工作时就是通过改变这个计数器的值来选取下一条 ...

  9. 【c++】常识易混提示

    1. struct 和 class 唯一的区别:默认的成员保护级别和默认的派生保护级别不同(前者为public,后者为private). 2. int *p = new int[23];     de ...

  10. no jpeg in java.library.path;java.lang.NoClassDefFoundError: Could not initialize class sun.awt.image.codec.JPEGImageEncoderImpl

    no jpeg in java.library.path;java.lang.NoClassDefFoundError: Could not initialize class sun.awt.imag ...