传送门

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

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

……

这题比较容易把人误导到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. inception_web

    1) wget --no-check-certificate https://bootstrap.pypa.io/ez_setup.py 2) python ez_setup.py --insecur ...

  2. WPF捕获全局未处理异常

    在WPF开发过程中我们一般都用try/catch块来捕获异常,但不是每个异常我们都能捕获,程序总会出现一些意想不到情况,抛出一些未捕获的异常,这时就要用到全局异常捕获,即在程序的最外层加上捕获未处理异 ...

  3. Cache缓存设计

    缓存的适用场景: 缓存的目的是提高访问速度,减少不必要的开销,提高性能.那什么样的场景适用于缓存呢.试想一个多项式的计算是一个CPU bound的操作,如果频繁调用同一个多项式的结果.显然缓存结果是一 ...

  4. Spark 错误日志中看到的一些问题

    2014-4-23 18:42:09 org.jivesoftware.spark.util.log.Log error 严重: Unable to contact shared group info ...

  5. C++ 创建类时常考虑的问题

    目录 继承 抽象接口 标准设计模式 初始化与析构模型 定义复制构造函数和赋值操作 模板 操作符 类型转换操作符 友元 非功能性约束 本文仅以c++为例,简要说明创建一个类时我们需要考虑的问题.创建一个 ...

  6. 感想篇:7)知其然与知其所以然,KnowHow与KnowWhy

    本章目的:探究--知其然与知其所以然,KnowHow与KnowWhy. 1.Know-How体系与代价: 100多年的汽车研发历史表明,企业只有开发过两代车以上才能逐步建立和完善Know-How体系. ...

  7. CDQZ Day5

    1DP #1题目名称 题目名称匹配块路径染色输入文件名 输入文件名match.in.in.inblock.inpath.inpaint.in输出文件名 输出文件名match.out.out.out.o ...

  8. mp4 格式无法使用html5的video标签播放

     只有视频编码为h264的视频才能在html5中使用video标签播放 我的解决方法为:下载魔影工厂,按照如下图所示步骤操作: width:600px;

  9. @Transcational特性

    捕获RuntimeException 捕获Error 并不捕获Checked Exception 在方法中使用@Transcational注解时候,通过throw new Exception(),在发 ...

  10. 【云计算】impala建表,文件关联,查询

    [cloudil@hadoop164 caixianfeng]$ hdfs dfs -mkdir -p /csv-pig/tab1 /csv-pig/tab1 tab1.csv: 1,true,123 ...